繁体   English   中英

使用带有nodeunit的DB连接/驱动程序编写单元测试

[英]Writing unit tests with DB connections/drivers with nodeunit

我正在尝试使用nodeunit为我的简单数据库层SimpleDbLayer编写一组测试单元。 我发现这很困难。

问题:

1)我想一次连接到数据库,并使用该连接。 我不能真正使用setUp和tearDown,因为它们分别在每次测试之前和之后运行。 作为“解决方案”,我在范围内使用模块范围的变量(请参见下面的代码)

2)我的简单数据库层有几个数据库驱动程序。 MongoMixin只是其中之一-我正在开发MariaMixin和PostgresMixin。 理想情况下,我将为每个驱动程序运行一次这些测试,以查看它们是否有效。 有没有一种简便的好方法?

3)我遇到的情况是我的测试代码有误,而我得到的只是关于test.done的投诉。 在这里我需要注意一些技巧吗?

4)我做错什么了吗? (请参见代码)

var 
  dummy

, declare = require('simpledeclare')
, SimpleDbLayer = require('./SimpleDbLayer')

, MongoMixin = require('./MongoMixin.js')

, mw = require('mongowrapper')
, async = require('async')
;

var db, layer;


exports.createLayer = {

  databaseConnect: function( test ){
    mw.connect('mongodb://localhost/hotplate', {}, function( err, returnedDb ){
      test.ifError( err );
      db = returnedDb;
      test.done( null );
    });
  },


  makeLayer: function( test ){

    var C = declare( [ SimpleDbLayer, MongoMixin ] );
    layer = new C( 'test', {  name: true, surname: true, age: true }, db );

    test.ok( layer );
    test.done(); 
  },

  deleteAll: function( test ){
    layer.delete( { }, { multi: true }, function( err, howmany ){
      test.ifError( err );
      test.done();
    });
  },

  insert: function( test ){
    var people = [
      { name: 'Chiara',    surname: 'Mobily',     age: 24 },
      { name: 'Tony',      surname: 'Mobily',     age: 37 },
      { name: 'Sara',      surname: 'Connor',     age: 14 },
      { name: 'Daniela',   surname: 'Mobily',     age: 64 },
    ];

    returnedPeople = [];

    var functions = [];

    // Populate the database
    people.forEach( function( person ){

      functions.push( function( done ){
        layer.insert( person, { returnRecord: true }, function( err, person ){
          test.ifError( err );
          returnedPeople.push( person );
          done( null );
        })
      })

    })

    async.series( functions, function( err, res ){
      test.ifError( err );
      test.done();
    });

  },

  databaseDisconnect: function( test ){
    db.close();
    test.done();
  },

} 

回答我自己。 解决方案很简单:对于每个驱动程序,这是test.js文件:

var driver = require('./ specificDriver'); var simpledblayerTests = require(“ ./lib/simpledblayer/test.js”);

var tests = simpledblayerTests.get(

  function getDbInfo( done ) {
    mw.connect('mongodb://localhost/tests', {}, function( err, db ){
      if( err ){
        throw new Error("MongoDB connect: could not connect to database");
      } else {
        done( null, db, driver );
      }
    });
  },

  function closeDb( db, done ) {
    db.close( done );
  }
);


for(var test in tests) {
    exports[ test ] = tests[ test ];
}

基本上,由实际的“主”模块导出的get()函数(稍后会详细介绍)具有两个参数:两个用于打开和关闭数据库连接的函数。

get()函数返回一堆准备好导出的函数-构成单元测试的函数。

主模块将具有以下内容:

exports.get = function( getDbInfo, closeDb ){

  var tests;
  var g = {};

  var startup = function( test ){
    var self = this;

    test.doesNotThrow( function(){

      getDbInfo( function( err, db, driver ){
        if( err ){
          throw( new Error("Could not connect to db, aborting all tests") );
          process.exit();
        }

        // Set the important g.driver variables (db and driver)
        g.db = db;
        d.driver = driver

        test.done();
      });
    });
  }


  var finish = function( test ){
    var self = this;
    closeDb( g.db, function( err ){
      if( err ){
        throw( new Error("There was a problem disconnecting to the DB") );
      }
      test.done();
    });
  };

  tests = {

    startup: startup,

    // ...
    // Your tests here. All functions will have access to g.db and g.driver 
    // ...

    finish: finish

  }

  return tests;
}

这样,每个模块都将运行相同的测试。 但是, driver变量和用于连接/断开数据库的功能将很简单。

使用实际代码查看它:

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM