4

我是 Mocha 的新手,对 Node/Express 只有一点经验。当我通过我的 Express 应用程序访问它时,我的 DbProvider 模块运行良好(mongodb)。现在我想测试它。我已经阅读了 Mocha 网站和一些我可以找到的教程。但是我很难找到一个我可以遵循的真实世界的例子(非常感谢任何链接!)。

这是我编写测试文件的失败尝试:

var DbProvider = require('../db').DbProvider;
var assert     = require('assert');
var dbProvider = new DbProvider('localhost', 27017, 'mydb');
var util       = require('util');

console.log(util.inspect(dbProvider));

describe('DbProvider', function(){
  describe('findAllNotes', function(){
    it('should return some notes', function(){
      dbProvider.findAllNotes({}, function (err, result){
        assert(result.length > 0);
      });
    })
  })
})

我得到的输出是这样的:

$ mocha
{}

  ✖ 1 of 1 test failed:

  1) DbProvider findAllNotes should return some notes:
     TypeError: Cannot call method 'collection' of undefined
      at DbProvider.doOperation (/Users/frode/Node/json/db.js:46:11)
      at DbProvider.findAllNotes (/Users/frode/Node/json/db.js:56:8)
      at Context.<anonymous> (/Users/frode/Node/json/test/test.js:15:18)
(cutting out the rest)

看来我没有成功创建dbProvider。这在我的应用程序中完美运行......我怎样才能使它工作?(也许还有:我设置它的方式一般好吗?)

编辑:这是 db.js 文件:

// Database related
'use strict';

var MongoClient       = require('mongodb').MongoClient;
var BSON              = require('mongodb').BSONPure;
var ObjectID          = require('mongodb').ObjectID;
var checkForHexRegExp = new RegExp("^[0-9a-fA-F]{24}$");
var Validator         = require('validator').Validator
var fieldMaxLength    = 1024;
//var util              = require('util');

var DbProvider = function(host, port, database) {
  var dbUrl = "mongodb://"+host+":"+port+"/"+database;
  var self = this;
  MongoClient.connect(dbUrl, function(err, db) {
    self.db = db;
  });
};

// Do some basic validation on the data we get from the client/user
var validateParams = function(params, callback) {
  // Let´ do a quick general sanity check on the length on all fields
  for(var key in params) {
    if(params[key].length > fieldMaxLength) callback(new Error('Field ' + key + ' is too long.'));
  }
  // and the let us check some specific fields better
  if (params._id) {
    if(checkForHexRegExp.test(params._id)) {
      // In case of '_id' we also need to convert it to BSON so that mongodb can use it.
      params._id = new BSON.ObjectID(params._id);
    } else {
      var err = {error: 'Wrong ID format'};
    }
  }
  if(err) callback(err);
}

// Generalized function to operations on the database
// Todo: Generalize even more when user authenication is implemented
DbProvider.prototype.doOperation = function(collection, operation, params, callback) {
  validateParams(params, callback);
  var operationCallback = function(err, result) {
    callback(err, result);
  };
  this.db.collection(collection, function(err, collection) {
    if(operation==='find') {
      collection.find().toArray(operationCallback);
    } else {
      collection[operation](params, operationCallback);
    }
  });
}

DbProvider.prototype.findAllNotes = function(params, callback) {
  this.doOperation('notes', 'find', params, callback);
};

DbProvider.prototype.findNoteById = function(params, callback) {
  this.doOperation('notes', 'findOne', params, callback);
};

DbProvider.prototype.saveNote = function(params, callback) {
  params.created_at = new Date();
  this.doOperation('notes', 'save', params, callback);
};

DbProvider.prototype.deleteNote = function(params, callback) {
  this.doOperation('notes', 'remove', params, callback);
};

DbProvider.prototype.findUser = function(params, callback) {
  this.doOperation('users', 'findOne', params, callback);
};

exports.DbProvider = DbProvider;

解决方案:

在 Benjamin 告诉我处理 mongodb 连接到数据库的异步性质后,并受到他关于如何调整代码的建议的启发,我将构造函数 DbProvider 分为两部分。第一部分,构造函数 DbProvider 现在只是将 db 参数保存到一个变量中。第二部分,一个新函数,DbProvider.connect 执行实际的异步连接。见下文。

var DbProvider = function(host, port, database) {
  this.dbUrl = "mongodb://"+host+":"+port+"/"+database;
};

DbProvider.prototype.connect = function(callback) {
  var self = this;
  MongoClient.connect(this.dbUrl, function(err, db) {
    self.db = db;
    callback();
  });
};

所以我现在可以像这样进行 Mocha 测试(并且异步测试也需要包含“完成”,就像你在下面的代码中看到的那样):

var assert     = require('assert');
var DbProvider = require('../db').DbProvider;
var dbProvider = new DbProvider('localhost', 27017, 'nki');

describe('DbProvider', function(){
  describe('findAllNotes', function(){
    it('should return some notes', function(done){
      dbProvider.connect(function(){
        dbProvider.findAllNotes({}, function (err, result){
          assert(result.length > 0);
          done();
        });
      });
    })
  })
})

请注意,实际测试(“应该返回一些注释”)并不值得骄傲。我在这里想要的是进行设置,以便我能够测试一些东西。现在我终于可以做到这一点,我需要编写好的测试(在有一个测试数据库的行中,清除它,测试插入文档,测试搜索文档,等等......)。

在我的 Express 应用程序中,我曾经这样设置数据库:

var DbProvider    = require('./db').DbProvider;
// Setup db instance
var dbProvider = new DbProvider(
  process.env.mongo_host       || 'localhost',
  process.env.mongo_port       || 27017,
  process.env.mongo_db         || 'nki'
);

现在我做同样的事情,但另外,我调用了新的连接函数:

// Connect to db. I use (for now) 1 connection for the lifetime of this app.
// And I do not use a callback when connecting here (we do in the testing)
dbProvider.connect(function(){});

Benjamin 实际上指出,在 Express 应用程序中像这样设置数据库可能没问题,但不是最佳实践。但是,在我弄清楚真正的最佳实践是什么之前,我将保留这段代码。下面是一些关于我发现的主题的链接(但我还没有得出自己将如何解决它的结论):
Node.js 上 MongoDB 连接的最佳实践是什么?
[node-mongodb-native] MongoDB 初学者最佳实践

如果你喜欢,非常欢迎你在 github 上关注/fork/whatever 这个项目。我的目标是尽我所能将其做好生产准备。链接是 https://github.com/frodefi/node-mongodb-json-server

4

2 回答 2

1

MongoClient.connect异步的。

从文档:

回调(函数)——这将在执行此方法后被调用。如果发生错误,第一个参数将包含 Error 对象,否则为 null。而第二个参数将包含初始化的 db 对象,如果发生错误,则为 null。

这意味着DbProvider.db尚未在测试中设置,这就是您获得undefined.

在这里:

MongoClient.connect(dbUrl, function(err, db) {
    self.db = db;
});

您告诉它“连接发生后更新 self.db”,这至少是在此事件循环之后的一个事件循环滴答声(但可能更多)。在您的代码中,您在创建实例后立即mocha执行您的.describe和方法,这意味着它尚未初始化。.itDbProvider

我建议您重新考虑DbProvider以返回回调而不是构造函数。也许是这样的:

var getDbProvider = function(host, port, database,callback) {
  var dbUrl = "mongodb://"+host+":"+port+"/"+database;
  MongoClient.connect(dbUrl, function(err, db) {
    self.db = db;
    callback(db);
  });
};

这也意味着将所有DBProvider方法移动到一个对象(也许回调将返回一个dbprovider对象而不仅仅是一个数据库?)。

使用单元测试解决了另一个错误:)

于 2013-03-23T16:15:19.043 回答
0

This is what I used: https://github.com/arunoda/mocha-mongo

It has set of testing helpers for mongodb

于 2013-04-19T05:34:46.103 回答