cli icon indicating copy to clipboard operation
cli copied to clipboard

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

Open ngustavo opened this issue 4 years ago • 23 comments

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.

ngustavo avatar Aug 12 '21 23:08 ngustavo

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.

Kulwch avatar Aug 15 '21 17:08 Kulwch

For now, I created this temporary solution.

ngustavo avatar Aug 15 '21 20:08 ngustavo

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 !

Kulwch avatar Aug 16 '21 08:08 Kulwch

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;
})();

NixonAzeredo avatar Jan 05 '22 12:01 NixonAzeredo

@ephys can you look into this?

WikiRik avatar Jan 13 '22 11:01 WikiRik

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

ephys avatar Jan 13 '22 11:01 ephys

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.

ngustavo avatar Jan 20 '22 19:01 ngustavo

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)

ephys avatar Jan 21 '22 09:01 ephys

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.

ngustavo avatar Feb 08 '22 12:02 ngustavo

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)

ephys avatar Feb 08 '22 12:02 ephys

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

mirzagirl avatar Dec 17 '22 08:12 mirzagirl

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.

Yxn16a avatar Apr 21 '23 22:04 Yxn16a

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?

inbal-samucha avatar May 21 '23 12:05 inbal-samucha

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.

jleweli avatar Jun 26 '23 09:06 jleweli

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 avatar Jun 26 '23 16:06 Yxn16a

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: @.***>

Yxn16a avatar Jun 26 '23 16:06 Yxn16a

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.

jleweli avatar Jun 26 '23 19:06 jleweli

ping

mahnunchik avatar Oct 20 '23 03:10 mahnunchik

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;

heyimhere avatar Feb 03 '24 02:02 heyimhere

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...

sarakabariti avatar Apr 08 '24 21:04 sarakabariti

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:

  • replace dynamic require with dynamic import. Keep in mind that it's async but if you use ESM you have access to top level await
  • __dirname and __filename can be replaced with import.meta.dirname and import.meta.filename respectively (node 21), or with this snippet of code:
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;

ephys avatar Apr 09 '24 07:04 ephys