我是 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