Node.js 使用Sequelize为测试构建、植入和销毁PostgreSQL

Node.js 使用Sequelize为测试构建、植入和销毁PostgreSQL,node.js,postgresql,testing,mocha.js,sequelize.js,Node.js,Postgresql,Testing,Mocha.js,Sequelize.js,我正在尝试为每个测试自动构建、播种和销毁我的数据库。我正在使用PostgreSQL、Mocha和Sequelize 我找到了一个库:它让我在某种程度上达到了目的,但最终它非常不一致,偶尔会抛出约束错误:未处理的拒绝SequelizeUniqueConstraintError:Validation error,即使我没有对模型进行任何验证 下面是我做测试的方式 const sequelize = new Sequelize('test_db', 'db', null, { logging: f

我正在尝试为每个测试自动构建、播种和销毁我的数据库。我正在使用PostgreSQL、Mocha和Sequelize

我找到了一个库:它让我在某种程度上达到了目的,但最终它非常不一致,偶尔会抛出约束错误:
未处理的拒绝SequelizeUniqueConstraintError:Validation error
,即使我没有对模型进行任何验证

下面是我做测试的方式

const sequelize = new Sequelize('test_db', 'db', null, {
  logging: false,
  host: 'localhost',
  port: '5432',
  dialect: 'postgres',
  protocol: 'postgres'
})

describe('/auth/whoami', () => {
  beforeEach((done) => {
    Fixtures.loadFile('test/fixtures/data.json', models)
      .then(function(){
         done()
      })
  })

  afterEach((done) => {
    sequelize.sync({
      force: true
    }).then(() => {
      done()
    })
  })

  it('should connect to the DB', (done) => {
    sequelize.authenticate()
      .then((err) => {
        expect(err).toBe(undefined)
        done()
      })
  })

  it('should test getting a user', (done) => {
    models.User.findAll({
      attributes: ['username'],
    }).then((users) => {
      users.forEach((user) => {
        console.log(user.password)
      })
      done()
    })
  })
})
我的模型定义如下:

var Sequelize = require('sequelize'),
    db = require('./../utils/db')

var User = db.define('User', {
  username: {
    type: Sequelize.STRING(20),
    allowNull: false,
    notEmpty: true
  },
  password: {
    type: Sequelize.STRING(60),
    allowNull: false,
    notEmpty: true
  }
})

module.exports = User
错误日志:

Fixtures: reading file test/fixtures/data.json...
Executing (default): CREATE TABLE IF NOT EXISTS "Users" ("id"   SERIAL , "username" VARCHAR(20) NOT NULL, "password" VARCHAR(60) NOT NULL, "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL, "updatedAt" TIMESTAMP WITH TIME ZONE NOT NULL, PRIMARY KEY ("id"));
Executing (default): SELECT "id", "username", "password", "createdAt", "updatedAt" FROM "Users" AS "User" WHERE "User"."id" = 1 AND "User"."username" = 'Test User 1' AND "User"."password" = 'testpassword';
Executing (default): SELECT i.relname AS name, ix.indisprimary AS primary, ix.indisunique AS unique, ix.indkey AS indkey, array_agg(a.attnum) as column_indexes, array_agg(a.attname) AS column_names, pg_get_indexdef(ix.indexrelid) AS definition FROM pg_class t, pg_class i, pg_index ix, pg_attribute a WHERE t.oid = ix.indrelid AND i.oid = ix.indexrelid AND a.attrelid = t.oid AND t.relkind = 'r' and t.relname = 'Users' GROUP BY i.relname, ix.indexrelid, ix.indisprimary, ix.indisunique, ix.indkey ORDER BY i.relname;
Executing (default): INSERT INTO "Users" ("id","username","password","createdAt","updatedAt") VALUES (1,'Test User 1','testpassword','2016-04-29 23:15:08.828 +00:00','2016-04-29 23:15:08.828 +00:00') RETURNING *;
Unhandled rejection SequelizeUniqueConstraintError: Validation error
这曾经奏效,但再也不会奏效。在每次测试之前,是否有一种更可靠的方法让我从一个完全干净的DB开始,用测试数据填充以进行操作



此外,如果有人也知道为什么即使我启用了
logging:false
,我仍然得到
console.logs()
,那将不胜感激。

您粘贴的错误似乎表明相同的数据被多次插入,导致
id
列上发生冲突

我希望调用
sequelize.sync({force:true})
会在每次运行时为您清除所有数据库表,但情况似乎并非如此。您可以尝试在每个hook之前将调用移动到一个
,以确保要运行的第一个测试也有一个新的数据库

在我正在处理的应用程序中,我们不会为每个测试重新同步数据库,而是在开始时执行一次,并在测试之间截断表。我们使用的清理函数如下所示:

function cleanup() {
    return User.destroy({ truncate: true, cascade: true });
}
create
方法完成从json装置加载数据并将其插入数据库的工作

function create() {
    var users = require('./fixtures/user.json');
    return User.bulkCreate(users);
}
通过省略
sequelize fixture
并自己处理事情,您可能能够简化依赖关系并提高稳定性

另外,还有一个无关的建议:Sequelize的方法返回Mocha可以本地处理的承诺,因此无需在测试和设置/拆卸代码中使用
done
回调:

 it('should connect to the DB', () => {
   return sequelize.authenticate()
 })
如果承诺被拒绝,测试将失败

此外,目前建议不要使用箭头功能:

不鼓励将箭头功能传递给摩卡。他们对
this
值的词法绑定使他们无法访问Mocha上下文,以及类似
this.timeout(1000)的语句在箭头函数内不起作用


您的模型是如何定义的?logging false表示Sequelize不会将SQL打印到控制台。在您的情况下,您应该启用log以获取有关您的问题的所有信息。@denisazevedo我已经用我的模型定义和日志更新了我的帖子。我试图禁用日志记录,但由于某些原因,它仍然会生成日志,因此我仍然可以看到它们。