【问题标题】:module.filename is undefined (Sequelize, Node.js)module.filename 未定义(Sequelize,Node.js)
【发布时间】:2020-09-13 03:09:39
【问题描述】:

我正在阅读this Sequelize 和 Node.js 的教程。我在第 4 行收到以下错误:

TypeError:路径必须是字符串。收到未定义

const fs = require('fs');
const path = require('path');
const Sequelize = require('sequelize');
const basename = path.basename(module.filename);
const env = process.env.NODE_ENV || 'development';
const config = require(`${__dirname}/../config/config.json`)[env];
const db = {};

let sequelize;
if (config.use_env_variable) {
  sequelize = new Sequelize(process.env[config.use_env_variable]);
} else {
  sequelize = new Sequelize(
    config.database, config.username, config.password, config
  );
}
fs
  .readdirSync(__dirname)
  .filter(file =>
    (file.indexOf('.') !== 0) &&
    (file !== basename) &&
    (file.slice(-3) === '.js'))
  .forEach(file => {
    const model = sequelize.import(path.join(__dirname, file));
    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;

似乎 module.filename 未定义。我不知道为什么会这样(我是 Node 新手),与教程和我的应用程序的唯一区别是我不使用 express api(但 GraphQL)并且我的 sequelize 自动生成的文件夹位于../ postgres/ 而不是 ../

编辑:

如果我运行教程存储库并打印 module.filename 我会得到 /Users/Creece/Downloads/postgres-express-node-tutorial-master/server/models/index.js

现在,如果我手动分配 module.filename,如下所示:module.filename = '/Users/Creece/Downloads/express_graphql_hmr_article_boilterplate-master/server/postgres/models/index.js';(在分配基本名称常量之前),我在 const model = ... 行上收到一个新的奇怪错误:

Error: Cannot find module '/Users/Creece/Downloads/express_graphql_hmr_article_boilerplate-master/server/dist/postgres/models/todo.js'

请注意,上述路径上的文件夹“dist”是 webpack 的(虽然不确定,但它不是我的)。

编辑2:(关于我对@NikMartin 回答的评论

这是 Sequelize 自动生成的文件 ./postgres/models/todo.js:

module.exports = (sequelize, DataTypes) => {

  const Todo = sequelize.define('Todo', {
    title: {
      type: DataTypes.STRING,
      allowNull: false,
    },
  });



  Todo.associate = (models) => {
    Todo.hasMany(models.TodoItem, {
      foreignKey: 'todoId',
      as: 'todoItems',
    });
  };
  return Todo;

};

解决方案(基于接受的答案):

const fs = require('fs');
const path = require('path');
const Sequelize = require('sequelize');
//const basename = path.basename(__filename);
const env = process.env.NODE_ENV || 'development';
const config = require(`${__dirname}/../config/config.json`)[env];
const db = {};

let sequelize;
if (config.use_env_variable) {
  sequelize = new Sequelize(process.env[config.use_env_variable]);
} else {
  sequelize = new Sequelize(
    config.database, config.username, config.password, config
  );
}

const models = [
  require('./Todo')(sequelize, Sequelize),
  require('./TodoItem')(sequelize, Sequelize),
];

models.forEach(model => {
  db[model.name] = model;
});

models.forEach(model => {

  if (db[model.name].associate) {
    console.log("entered");
    db[model.name].associate(db);
  }
});

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

module.exports = db;

【问题讨论】:

  • 第 4 行 module.filename 未定义。 module中如何获取文件名
  • 你的节点版本是多少? node -v
  • @FrançoisP。版本是 6.10.0

标签: javascript node.js postgresql postgresql-9.6


【解决方案1】:

关于module.filename 问题 - 您可以阅读此讨论或仅此评论:https://github.com/webpack/webpack/issues/1599#issuecomment-328540024

引用:

aprilmintacpineda 发表评论

对于那些遇到此问题的人,还需要注意的是,如果您捆绑了文件,module.filename 将是未定义的。在我的情况下,它是未定义的,我通过简单地用 __filename 替换它来修复它,这会产生相同的结果。

所以您应该可以将path.basename(module.filename); 替换为path.basename(__filename);

但据我了解,它仍然是 /dist 文件夹中的捆绑 js 文件,并且您的模型位于 /postgres/models 文件夹中。

我在这里看到两个选项:

1) 不要自动加载模型,而是直接需要所有必要的模型。

假设这个文件的路径是 /postgres/models/index.js 并且实际模型在同一个文件夹中 未测试

const Sequelize = require('sequelize');
const env = process.env.NODE_ENV || 'development';
const config = require(`${__dirname}/../config/config.json`)[env];
const db = {};

let sequelize;
if (config.use_env_variable) {
  sequelize = new Sequelize(process.env[config.use_env_variable]);
} else {
  sequelize = new Sequelize(
    config.database, config.username, config.password, config
  );
}

// removed autoloading block

// setting the list of used models
const models = [
  require('./TodoItem')(sequelize, Sequelize),
  require('./Comment')(sequelize, Sequelize)
];

// registering models by their names
models.forEach(model => {
  db[model.name] = model;
});

// using registered models to set up relations
models.forEach(model => {
  if (model.associate) {
    model.associate(db);
  }
});

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

module.exports = db;

这意味着灵活性降低,但您的捆绑代码将开始使用捆绑模型。我想 dist 代码需要一些“外部”(缺少内部模型)非配置 js 文件才能工作并不是最好的事情。

2) 不是从当前文件夹而是从定义的文件夹加载文件。

  • /postgres/models/index.js 从模型文件夹移出到 /postgres/index.js(因此模型文件夹将只有实际模型文件)
  • 从文件中删除basename
  • 而不是.readdirSync(__dirname) 使用.readdirSync(__dirname + "/../postgres/models")
  • 删除(file !== basename) &&检查(因为它是不必要的)

无论哪种方式,替换模块的属性 (module.filename) 都不是解决方案。

【讨论】:

  • 首先感谢您花时间帮助我解决这个问题。很多天后,我自己仍然无法弄清楚。将 module.filename 更改为 __filename 或使用您提出的第二个选项让我收到此错误:错误:找不到模块 '/Users/Creece/Downloads/express_graphql_hmr_article_boilerplate-master/server/dist/postgres/models/todo.js' 使用第一个选项,我得到:错误:Todo.hasMany 调用的东西不是 Sequelize.Model 的子类
  • 添加到我上面的评论中,似乎第一个选项确实有效(我在控制台上看到了“INSERT INTO ...”sql 内容)。现在唯一的问题是“Todo.hasMany”函数返回错误。我正在使用该特定代码更新我的问题。
  • @SkyPower,我不知道你为什么得到“/d‌​ist/postgres/models/‌​todo.js”而不是“/d‌​ist/../postgres/models/ ‌​todo.js”在#2,因为.readdirSync(__dirname + "/../postgres/models")据我所知应该在/dist目录中。关于#1 - 是的,db 对象应该在关联之前拥有所有模型,以修复解决方案。如果我错过了其他事情,请告诉我。
【解决方案2】:

这可以这样简化:

const config = require('../config/config')['development']
const {Sequelize} = require('sequelize');
const sequelize = new Sequelize(config['dbUrl']);
const db = {};
const models = [];

sequelize.authenticate()
    .then(() => console.log('Connection has been established successfully.'))
    .catch(err => console.error('Unable to connect to the database:', err));

const modules = [
    require('./user'),
    require('./post'),
    require('./comment')
]

modules.forEach(module => {
    const model = module(sequelize, Sequelize)
    db[model.name] = model
    models.push(model)
});

models.forEach(model => {
    if (db[model.name].associate) {
        db[model.name].associate(db)
    }
});

module.exports = {...db, sequelize, Sequelize};

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 2017-04-07
    • 2017-01-27
    • 2015-11-19
    • 1970-01-01
    • 2015-01-04
    • 2017-05-22
    • 2018-07-30
    • 1970-01-01
    相关资源
    最近更新 更多