1

我正在尝试为新的 Hapi 应用程序设置测试模式。我过去在 Express 中使用过 Mocha 和 Chai,但我正在尝试使用 Lab 和 Code 留在 Hapi 生态系统中。我还使用 Bookshelf 和 Knex 来处理数据库交互。

所以我有一个简单的健康端点我想测试。

'use strict';

const controller = require('../controller/healthController');

module.exports = [
  {
    method: 'GET',
    path: '/health',
    config: {
      handler: controller.health,
      description: 'The health endpoint returns 200',
      tags: ['api', 'health']
    }
  }
];

在处理程序中,它只是进行快速查询以确保它可以连接到数据库。

'use strict';

const bookshelf = require('../config/db');
const knex = bookshelf.knex;

module.exports = {
  health: function (request, reply) {
    knex.raw('SELECT version()').then(() => {
      reply('service is running');
    }).catch((err) => {
      reply({err: err, code: 500});
    });
  }
};

据我了解,需要服务器然后使用 server.inject 实际上并没有启动服务器,所以我不相信我应该有一个 db 连接,这意味着我应该将它模拟出 db 调用。对我来说奇怪的是这个测试通过了:

'use strict';
const Code = require('code');
const Lab = require('lab');

const lab = exports.lab = Lab.script();
const describe = lab.describe;
const it = lab.test;
const expect = Code.expect;
const before = lab.before;

let server;

describe('health controller', () => {

  before((done) => {
    server = require('../../server');
    done();
  });

  it('health check replies 200 when successful call to db', (done) => {

    const options = {
      method: 'GET',
      url: '/health'
    };

    server.inject(options, (res) => {
      expect(res.payload).to.include('is running');
      expect(res.statusCode).to.equal(200);

      done();
    });

  });

});

所以我有两个问题。首先,我觉得上面的测试不应该真正通过。除非它正在加载所有内容并因此连接到我想的数据库。也许我应该只测试控制器/处理程序方法?但我还没有找到任何这样的例子。

其次,无论如何我都试图将 knex.raw 调用存根,当我尝试像下面那样做时,我得到一个 500 错误。

'use strict';
const Code = require('code');
const Lab = require('lab');
const Sinon = require('sinon');

const lab = exports.lab = Lab.script();
const describe = lab.describe;
const it = lab.test;
const expect = Code.expect;
const before = lab.before;

let server;
let knex = require('../../app/config/db').knex;

describe('health controller', () => {

  before((done) => {
    server = require('../../server');
    done();
  });

  it('health check replies 200 when successful call to db', (done) => {

    const stub = Sinon.stub(knex, 'raw').returns({});

    const options = {
      method: 'GET',
      url: '/health'
    };

    server.inject(options, (res) => {
      expect(res.payload).to.include('is running');
      expect(res.statusCode).to.equal(200);
      expect(stub.calledOnce).to.be.true();
      done();
    });

  });

});

我不确定为什么会这样。

4

2 回答 2

1

我认为Sinon.stub(knex, 'raw').resolves({});是一个更好的解决方案

于 2018-03-14T23:04:09.477 回答
0

server.inject就像您向真实服务器发出请求一样工作。因此,如果在运行测试时您的数据库已启动,端点将从数据库返回数据,就像您手动启动服务器时一样。

Sinon.stub(knex, 'raw').returns({});不会工作。knex.raw(…)预计会返回一个 Promise,而不是一个空对象。请尝试以下方法:

Sinon.stub(knex, 'raw').returns(Promise.resolve({}));

附带说明:记住server.stop()在每次测试后调用,以确保测试之间没有状态持续存在。一般来说,我认为您可以查看Hapi University 存储库中的示例测试文件。

于 2017-01-29T21:05:16.577 回答