sequelize / cli

The Sequelize CLI
MIT License
2.52k stars 527 forks source link

models/index.js doesn't work with ES6 modules #960

Open ngustavo opened 2 years ago

ngustavo commented 2 years ago

What you are doing?

Trying to load models with models/index.js doesn't work with ES6 modules activated with "type": "module"

const basename = path.basename(__filename);
// {...}
const model = require(path.join(__dirname, file))(sequelize, Sequelize.DataTypes);

Variables like __filename and __dirname don't work anymore with modules.

What do you expect to happen?

There should be an option to choose between Modules or CommonJS.

What is actually happening?

There's only CommonJS support.

Kulwch commented 2 years ago

Hi, Having the same problem. Tried a long list of things to fix the problem, can't solve it. I'm stuck in a project i'm supposed to deliver in 2 weeks... Is there any chance someone has an idea of what to do? I searched for hours on the web, didn't find a useful thing.

ngustavo commented 2 years ago

For now, I created this temporary solution.

Kulwch commented 2 years ago

As a matter of fact, i occured this error since i didn't delete a model i created manually before using sequelize-cli... Thanks anyway for your answer, i'll keep it in mind !

NixonAzeredo commented 2 years ago

This is my solution for the moment

import { readdirSync } from "fs";
import { basename, dirname } from "path";
import { Sequelize, DataTypes } from "sequelize";
import { fileURLToPath } from 'url';
import database from "../config/database.js";

const __filename = fileURLToPath(import.meta.url);
const __dirname = dirname(__filename);

const db = {};
const sequelize = new Sequelize(database.development);

export default (async () => {
  const files = readdirSync(__dirname)
    .filter(
      (file) => file.indexOf('.') !== 0
      && file !== basename(__filename)
      && file.slice(-3) === '.js',
    );

  for await (const file of files) {
    const model = await import(`./${file}`);
    const namedModel = model.default(sequelize, DataTypes);
    db[namedModel.name] = namedModel;
  }

  Object.keys(db).forEach((modelName) => {
    if (db[modelName].associate) {
      db[modelName].associate(db);
    }
  });

  db.sequelize = sequelize;
  db.Sequelize = Sequelize;

  return db;
})();
WikiRik commented 2 years ago

@ephys can you look into this?

ephys commented 2 years ago

I'm not sure how to go about fixing this.

If we generate CJS code, it's not going to work for users that use ESM. If we generate ESM, it's not going to work for users that use CJS.

I suppose we could prompt during generation which module system to use

ngustavo commented 2 years ago

I guess it's possible to use *.mjs, since it's a standard, just to differentiate. Then, at generation time, only copy those with the corresponding extension. There's also the package.json option, "type": "module". This would imply having premade packages, as CRA does. The last option is doing weird string interpolation.

ephys commented 2 years ago

If we were to generate .mjs, users using commonjs would not be able to require those files But I think we could just maintain both a mjs & cjs version and prompt the user to choose one (with the default being guessed from package.json)

ngustavo commented 2 years ago

That's what I meant by "differentiate" and "corresponding extension". It's the easiest way to go. I'm not sure if Node can figure it out on its own though when given duplicate files.

ephys commented 2 years ago

Well import requires specifying the file extension and require is unable to load esm so it should, but it would be cleaner to emit only one of them by asking the user the choose It shouldn't be a difficult to implement but I'm mostly focused on the main library right now - if someone wants to give a shot at a PR

(The CLI code is also in dire need of an update and I'd like to migrate it to ESM before doing anything else too)

mirzagirl commented 1 year ago

model.default ......error saying TypeError: model.default is not a function please help

Yxn16a commented 1 year ago

This is my solution for the moment

import { readdirSync } from "fs";
import { basename, dirname } from "path";
import { Sequelize, DataTypes } from "sequelize";
import { fileURLToPath } from 'url';
import database from "../config/database.js";

const __filename = fileURLToPath(import.meta.url);
const __dirname = dirname(__filename);

const db = {};
const sequelize = new Sequelize(database.development);

export default (async () => {
  const files = readdirSync(__dirname)
    .filter(
      (file) => file.indexOf('.') !== 0
      && file !== basename(__filename)
      && file.slice(-3) === '.js',
    );

  for await (const file of files) {
    const model = await import(`./${file}`);
    const namedModel = model.default(sequelize, DataTypes);
    db[namedModel.name] = namedModel;
  }

  Object.keys(db).forEach((modelName) => {
    if (db[modelName].associate) {
      db[modelName].associate(db);
    }
  });

  db.sequelize = sequelize;
  db.Sequelize = Sequelize;

  return db;
})();

This works, you just have to check if the file were imported before you call "model.default(sequelize,DataTypes) because you are awaiting for the files to be read other than that you are good to go" Also if the "const model = await import('./${file}') does not work just use const model = await import(path.resolve("src", "models", ${file})); after you have imported path from path then you should be good.

inbal-samucha commented 1 year ago

I write like this but it is'nt load the models and then give my the error: TypeError: model.default is not a function

how can i fix it?

jleweli commented 1 year ago

I write like this but it is'nt load the models and then give my the error: TypeError: model.default is not a function

how can i fix it?

I had the same issue. In my case models/index.js was loaded (which is the model loader and not a model. So the error message was correct). I wrote an additional .filter line to get it out of my file list. If this does not help you have to make sure that you are only importing models.

I have a different problem. Following the solution of @NixonAzeredo I am trying to load models in my controller files: const User = sequelize.models.user; I get undefined doing so. The sequelize object which is exported from my models/index.js does not yet contain the models and I am wondering how I can fix this. I think this has to do with the async model loading. I tried out various things, but without any success, yet.

Yxn16a commented 1 year ago

This works, what changes was to include path to the models. but remember to define your sequelize database connection somewhere. I am saying this in case you copy this code block. Other than that this should work. Hopefully this helps


import { readdirSync } from "fs";
// import path  here 
import path from "path";
import { basename, dirname } from "path";
import { Sequelize, DataTypes } from "sequelize";
import { fileURLToPath } from "url";
import sequelize from "../config/connection.js";

const __filename = fileURLToPath(import.meta.url);
const __dirname = dirname(__filename);
const db = {};
export default (async () => {
  const files = readdirSync(__dirname).filter(
    (file) =>
      file.indexOf(".") !== 0 &&
      file !== basename(__filename) &&
      file.slice(-3) === ".js"
  );

  for await (const file of files) {
// use path here to access your models from models directory then await for it @
    const model = await import(path.resolve("models", `${file}`));
    if (model.default) {
      const namedModel = await model.default(sequelize, DataTypes);
      db[namedModel.name] = namedModel;
    }
  }

  Object.keys(db).forEach((modelName) => {
    if (modelName) {
      if (db[modelName].associate) {
        db[modelName].associate(db);
      }
    }
  });

  db.sequelize = sequelize;
  db.Sequelize = Sequelize;
  return db;
})();
Yxn16a commented 1 year ago

I did post a new code snippet; just check it out and let me know if it works.

On Mon, Jun 26, 2023 at 11:57 AM jleweli @.***> wrote:

I write like this but it is'nt load the models and then give my the error: TypeError: model.default is not a function

how can i fix it?

I had the same issue. In my case models/index.js was loaded (which is the model loader and not a model. So the error message was correct). I wrote an additional .filter line to get it out of my file list. If this does not help you have to make sure that you are only importing models.

I have a different problem. Following the solution of @NixonAzeredo https://github.com/NixonAzeredo I am trying to load models in my controller files: const User = sequelize.models.user; I get undefined doing so. The sequelize object which is exported from my models/index.js does not yet contain the models and I am wondering how I can fix this. I think this has to do with the async model loading. I tried out various things, but without any success, yet.

— Reply to this email directly, view it on GitHub https://github.com/sequelize/cli/issues/960#issuecomment-1607120078, or unsubscribe https://github.com/notifications/unsubscribe-auth/AWGZUKDHC6QVDED6DEZD5LLXNFMKLANCNFSM5CCM5NIQ . You are receiving this because you commented.Message ID: @.***>

jleweli commented 1 year ago

Thanks a lot for your effort. I tried it out and can confirm that it works. Unfortunately I ran into another issue: migrations seem not to work with ES6 modules out of the box. Possible solutions seem quite hacky. I decided to keep using CommonJS syntax and wait until there is better support for ES6.

mahnunchik commented 8 months ago

ping

heyimhere commented 5 months ago

This works on my Nextjs 14.1.0, Node 19 and Sequelize 6.35

'use strict';

const fs = require('fs');
const path = require('path');
const Sequelize = require('sequelize');
const process = require('process');
const env = process.env.NODE_ENV || 'development';
const config = require(__dirname + '/../config/config.js')[env];
const db = {};

const models = process.cwd() + '/db/models/' || __dirname;
const basename = path.basename(models + "index.js");

let sequelize;
if (config.use_env_variable) {
  sequelize = new Sequelize(process.env[config.use_env_variable], config);
} else {
  sequelize = new Sequelize({database: config.database, username: config.username,
    password: config.password, config: config, dialect: "mysql", host: config.host,
    dialectModule: require("mysql2")});
}

fs
  .readdirSync(models)
  .filter(file => {
    return (
      file.indexOf('.') !== 0 &&
      file !== basename &&
      file.slice(-3) === '.js' &&
      file.indexOf('.test.js') === -1
    );
  })
  .forEach(file => {
    const model = require(`./${file}`)(sequelize, Sequelize.DataTypes);
    db[model.name] = model;
  });

Object.keys(db).forEach(modelName => {
  if (db[modelName].associate) {
    db[modelName].associate(db);
  }
});

db.sequelize = sequelize;
db.Sequelize = Sequelize;

module.exports = db;
sarakabariti commented 3 months ago

I just started learning node.js/express/Sequelize, I ran into this issue and spent hours trying to figure it out. I'm not very experienced and not sure if my solution is the best.. But what worked for me was to add a package.json file that just has the line {"type": "commonjs"} in each folder generated by Sequelize CLI (config, migrations, models). So I ended up with 3 of those files, one in each folder.

This fixed the running migrate issue and importing the db and models in the rest of my files that use ES6. If there's a better solution please let me know...

ephys commented 3 months ago

models/index.js is going away in Sequelize 7, as the new model initialization makes it irrelevant

If you want to use this file in ESM, here is what you need to do:

import { fileURLToPath } from 'node:url';
import path from 'node:path';

const __filename = fileURLToPath(import.meta.url);
const __dirname = path.dirname(fileURLToPath(import.meta.url));

Basically https://github.com/sequelize/cli/issues/960#issuecomment-1005660879, with a small tweak, is the solution you should use if you want an equivalent to the CJS file for ESM. This is what we would generate if we were to update the CLI to support generating it as ESM.

import { readdirSync } from "fs";
import { basename, dirname } from "path";
import { Sequelize, DataTypes } from "sequelize";
import { fileURLToPath } from 'url';
import database from "../config/database.js";

const __filename = fileURLToPath(import.meta.url);
const __dirname = dirname(__filename);

const db = {};
const sequelize = new Sequelize(database.development);

const files = readdirSync(__dirname)
  .filter(
    (file) => file.indexOf('.') !== 0
    && file !== basename(__filename)
    && file.slice(-3) === '.js',
  );

await Promise.all(files.map(async file => {
  const model = await import(`./${file}`);
  if (!model.default) {
    return;
  }

  const namedModel = model.default(sequelize, DataTypes);
  db[namedModel.name] = namedModel;
}))

Object.keys(db).forEach((modelName) => {
  if (db[modelName].associate) {
    db[modelName].associate(db);
  }
});

db.sequelize = sequelize;
db.Sequelize = Sequelize;

return db;