20

我正在使用 ExpressJS、NodeJS、Mongoose 和 Mocha 开发 REST API。

问题是我有一个 app.coffee 文件,它负责设置 ExpressJS 并连接到 Mongoose。我设置的方式是首先连接 Mongoose,如果连接成功,则启动 ExpressJS 应用程序。

问题是在设置 Mocha 时,我需要确保 app.coffee 中存在的 ExpressJS 应用程序在执行任何测试用例之前完全成功启动,包括所有异步代码。

为此,我创建了一个 test_helper.coffee 并将以下代码放入其中,但是,即使 app.coffee 中的代码尚未完全执行,测试用例也会开始执行,这实际上是有意义的:

before (done) ->
  require(__dirname + '/../src/app')
  done()

简而言之,我想确保 ExpressJS 应用程序在执行任何测试用例之前已完全完成设置。

我怎么能这样做?

4

6 回答 6

15

我迟到了,但我发现为快速应用程序设置 mocha 测试套件的最佳方法是让我的 app.js 或 server.js 文件导出应用程序对象,如下所示:

var mongoose = require('mongoose');
var express = require('express');
require('express-mongoose');

var env = process.env.NODE_ENV || 'development';
var config = require('./config/config')[env];

var models = require('./app/models');
var middleware = require('./app/middleware');
var routes = require('./app/routes');

var app = express();

app.set('port', process.env.PORT || config.port || 3000);
app.set('views', __dirname + '/app/views');
app.set('view engine', 'jade');

// database
mongoose.connect(config.db);

// middleware
middleware(app);

// Application routes
routes(app);

app.listen(app.get('port'));
console.log('Express server listening on port ' + app.get('port'));

// export app so we can test it
exports = module.exports = app;

确保您的配置文件具有不同的环境,例如开发、测试、生产设置:

var path = require('path');
var rootPath = path.normalize(__dirname + '/..');

module.exports = {
  development: {
    db: 'mongodb://localhost/my_dev_db',
    port: 3000
  },
  test: {
    db: 'mongodb://localhost/my_test_db',
    port: 8888
  },
  production: {
    // ...
  }
}

然后在您的测试文件中,您可以继续并要求您的应用程序,它将连接到正确的数据库和正确的端口:

var should = require('chai').should();
var request = require('supertest');
var mongoose = require('mongoose');

var app = require('../app');
var agent = request.agent(app);

var User = mongoose.model('User');

    // get users
    describe('GET /api/users', function() {
      it('returns users as JSON', function(done) {
        agent
        .get('/api/users')
        .expect(200)
        .expect('Content-Type', /json/)
        .end(function(err, res) {
          if (err) return done(err);
          res.body.should.have.property('users').and.be.instanceof(Array);
          done();
        });
      });
    });

最后,要启动整个怪物,您将其包含在您的package.json(确保在您的 devDependencies 中有 nodemon 和 mocha):

"scripts": {
    "start": "NODE_ENV=development ./node_modules/.bin/nodemon app.js",
    "test": "NODE_ENV=test ./node_modules/.bin/mocha --reporter spec test/**.js"
  }

现在您可以npm test使用npm start.

希望能帮助到你!ps:我从这个惊人的例子中学到的大部分东西: https ://github.com/madhums/node-express-mongoose-demo

于 2013-10-15T08:55:06.003 回答
3

我在使用 jasmine/supertest 测试我的 express 应用程序时遇到了同样的问题。我通过在应用程序准备好时发出并且仅在之后运行我的测试来解决了这个问题。这是我的目录结构

- server.js
- spec
    - setup.spec.js
    - test.spec.js

server.js您的服务器设置好并准备好运行测试时,添加app.emit('started');. 并确保导出您的应用程序!您可以在setup.spec.js其中观察要发出的事件。

服务器.js

const express = require('express');
const app = express();
module.exports = app; // for testing

setTimeout(() => {
    app.listen(process.env.PORT || 3000);
});

setup.spec.js

const server = require('../index');

beforeAll(done => server.on('started', done));

test.spec.js

const request = require('supertest');
const server = require('../index');

describe('test', () => {
    it('test server works', done => {
        request(server).get('/test').expect(200);
     });
});

同样的想法也适用于摩卡咖啡。这是一篇解释 mocha 的文章。你应该beforeAll改成before.

于 2017-05-30T21:54:32.733 回答
1

可能有更直接的方法,但我去 Grunt 是为了自动化我的功能测试。有一个 express 和 mocha 插件可以实现您的目标。我的咕噜文件:

'use strict';

module.exports = function (grunt) {
grunt.initConfig({
    express: {
        options: {}
      , test: {
            options: {
                script: './app.js'
            }
        }
    }
  , simplemocha: {
        options: {
            globals: ['should']
          , timeout: 8000
          , ignoreLeaks: false
          , ui: 'bdd'
          , reporter: 'tap'
        }
      , all: { src: ['tests/*.test.js'] }
    }
})

grunt.loadNpmTasks('grunt-express-server')
grunt.loadNpmTasks('grunt-simple-mocha')

grunt.registerTask('default', ['express:test', 'simplemocha', 'express:test:stop'])
}

奖励:添加 'grunt' 作为 git 预提交挂钩。这样你就不能在没有通过所有测试的情况下提交

于 2013-09-23T09:43:07.437 回答
0

您无需侦听端口即可测试您的应用程序。您可以在您的应用程序中使用测测试库,它应该没问题。

不过,您可能需要连接到数据库或 redis 客户端。您可以在configure您的应用程序的方法中执行此操作。由于节点缓存模块,您可以在不同的测试模块中要求应用程序模块而无需重新连接。

于 2013-09-23T09:22:16.943 回答
0

app.listen方法接受一个回调,该回调会在一切准备就绪时运行。因此,您需要能够在那里传递完成的回调。就像是

before (done) ->
  var app = require(__dirname + '/../src/app')
  app.listen(3000, done)
于 2017-07-05T13:47:10.513 回答
0

基本上你需要做两件事。

  1. 确保数据库在服务器侦听之前连接
  2. 确保在运行测试之前启动应用程序。

框架已经有了一种通过使用事件来处理这个问题的方法。

将此添加到您的数据库连接中,这会指示 mongoose 在完成连接时发出“就绪”。

mongoose.connection.once('open', function() { 
   // All OK - fire (emit) a ready event. 
   console.log('Connected to MongoDB');
   app.emit('ready'); 
});

然后指示 express 在收听之前等待“就绪”。完成后,它会发出:'appStarted'。

app.on('ready', function() { 
    app.listen(PORT, function(){ 
        console.log("Server listing on port:",PORT); 
        app.emit("appStarted");
    }); 
}); 

这都是 Mocha 的好习惯,以确保您的服务器顺利启动。完成之前的 Mocha 集成(我喜欢使用 Promises):

before(function() {        
    return new Promise((resolve,reject) => {
        app.on("appStarted", function(){
            return resolve();
        }); 
    });
});
于 2018-12-25T20:42:24.607 回答