我正在寻找一种在测试之间完全重置我的快速服务器的正确方法!看来这对我来说不仅仅是一个问题,许多其他用户也提出了同样的问题, 很多博客文章都写在论点上。建议的解决方案对我来说不起作用也不令人满意。这里有另一个类似的问题和一篇很好描述问题的文章,并提出了一些解决方案:
堆栈溢出 Closing an express server after running jasmine specs
博客: https://glebbahmutov.com/blog/how-to-correctly-unit-test-express-server/
这里专门创建一个包来解决这个问题: https://www.npmjs.com/package/server-destroy
现在,重现我的病情的最小工作示例。 在测试代码中,创建了一个快速服务器;当在某个端点上调用时,服务器会递增一个值并返回它:
( function() {
'use strict'
const enableDestroy = require( 'server-destroy' )
const app = require( 'express' )()
const http = require( 'http' )
let val = 0
app.use( '/inc', (req, res) => {
val ++
res.send(val.toString())
} )
const server = http.createServer( app )
server.listen( 3000 )
enableDestroy(server);
module.exports = server
} )()
测试包括两个相同的测试用例;它们都调用端点上的服务器,并检查返回的值。
提供before_each
和after_each
部分是为了确保在单个测试用例运行之前创建新连接,然后关闭,以确保两个测试用例之间的独立性:
const chai = require( 'chai' )
const chaiHttp = require( 'chai-http' )
const expect = chai.expect
chai.use( chaiHttp )
let server
describe( 'first test group', () => {
beforeEach( () => {
server = require( './server' )
} ),
afterEach( ( done ) => {
server.destroy( done )
delete require.cache[require.resolve( './server' )]
} ),
it( 'should respond 1', ( done ) => {
chai.request( server )
.get( '/inc' )
.set( 'Connection', 'close' )
.end( ( err, res ) => {
expect( res.text ).to.be.equal( '1' )
done()
} )
} ),
it( 'should respond 1', ( done ) => {
chai.request( server )
.get( '/inc' )
.set( 'Connection', 'close' )
.end( ( err, res ) => {
expect( res.text ).to.be.equal( '1' )
done()
} )
} )
} )
测试失败,因为第一次测试后服务器未运行。请注意,在after_each
部分,我强制缓存清理完全丢失了最后一个服务器实例。如果运行单个测试用例,则测试成功:
first test group
✓ should respond 1
1) "after each" hook for "should respond 1"
1 passing (70ms)
1 failing
1) first test group
"after each" hook for "should respond 1":
Error: Not running
at Server.close (net.js:1620:12)
at emitCloseNT (net.js:1671:8)
at _combinedTickCallback (internal/process/next_tick.js:135:11)
at process._tickCallback (internal/process/next_tick.js:180:9)
我使用的配置:
我该如何解决这个问题?错误信息意味着什么?
答案 0 :(得分:1)
它有点修改
我没有使用server-destroy
,因为server.close
工作正常
<强> server.js 强>
// ( function() { // no need for this
'use strict'
//const enableDestroy = require( 'server-destroy' )
const app = require( 'express' )()
const http = require( 'http' )
let val = 0
app.use( '/inc', (req, res) => {
val ++
res.send(val.toString())
} )
const server = http.createServer( app )
server.listen( 3000 )
// enableDestroy(server);
module.exports = server
// } )()
<强> test.js 强>
const chai = require( 'chai' )
const chaiHttp = require( 'chai-http' )
const expect = chai.expect
chai.use( chaiHttp )
let server
describe( 'first test group', () => {
beforeEach( () => {
server = require( './server' )
} ),
afterEach( ( done ) => {
// UPDATE DON'T CLOSE THE SERVER
delete require.cache[require.resolve( './server' )]
done()
//server.close( () => {
// delete require.cache[require.resolve( './server' )]
// done()
//})
} ),
it( 'should respond 1', ( done ) => {
chai.request( server )
.get( '/inc' )
.set( 'Connection', 'close' )
.end( ( err, res ) => {
expect( res.text ).to.be.equal( '1' )
done()
} )
} ),
it( 'should respond 1', ( done ) => {
chai.request( server )
.get( '/inc' )
.set( 'Connection', 'close' )
.end( ( err, res ) => {
expect( res.text ).to.be.equal( '1' )
done()
} )
} )
} )
答案 1 :(得分:0)
这是从最初提出问题开始的一段时间。
在chai-http文档中,我可以找到它具有使用request方法中的keepOpen()
方法打开服务器连接的机制。 (实际上它会返回一个代理)
const chai = require("chai");
const chaiHttp = require("chai-http");
chai.use(chaiHttp);
const chaiAppServer = chai.request(server).keepOpen();
https://www.chaijs.com/plugins/chai-http/#integration-testing
但是正如它所提到的,我们必须手动关闭服务器连接。
describe("TEST Case group", () => {
after(() => {
chaiAppServer.close();
});
//..... integration test cases
});