koa-lite/test/context/onerror.js
2018-07-01 01:29:19 +08:00

226 lines
5.5 KiB
JavaScript

'use strict';
const assert = require('assert');
const request = require('supertest');
const Koa = require('../..');
const context = require('../helpers/context');
describe('ctx.onerror(err)', () => {
beforeEach(() => {
global.console = jest.genMockFromModule('console');
});
afterEach(() => {
global.console = require('console');
});
it('should respond', () => {
const app = new Koa();
app.use((ctx, next) => {
ctx.body = 'something else';
ctx.throw(418, 'boom');
});
const server = app.listen();
return request(server)
.get('/')
.expect(418)
.expect('Content-Type', 'text/plain; charset=utf-8')
.expect('Content-Length', '4');
});
it('should unset all headers', async () => {
const app = new Koa();
app.use((ctx, next) => {
ctx.set('Vary', 'Accept-Encoding');
ctx.set('X-CSRF-Token', 'asdf');
ctx.body = 'response';
ctx.throw(418, 'boom');
});
const server = app.listen();
const res = await request(server)
.get('/')
.expect(418)
.expect('Content-Type', 'text/plain; charset=utf-8')
.expect('Content-Length', '4');
assert.equal(res.headers.hasOwnProperty('vary'), false);
assert.equal(res.headers.hasOwnProperty('x-csrf-token'), false);
});
it('should set headers specified in the error', async () => {
const app = new Koa();
app.use((ctx, next) => {
ctx.set('Vary', 'Accept-Encoding');
ctx.set('X-CSRF-Token', 'asdf');
ctx.body = 'response';
throw Object.assign(new Error('boom'), {
status: 418,
expose: true,
headers: {
'X-New-Header': 'Value'
}
});
});
const server = app.listen();
const res = await request(server)
.get('/')
.expect(418)
.expect('Content-Type', 'text/plain; charset=utf-8')
.expect('X-New-Header', 'Value');
assert.equal(res.headers.hasOwnProperty('vary'), false);
assert.equal(res.headers.hasOwnProperty('x-csrf-token'), false);
});
it('should ignore error after headerSent', done => {
const app = new Koa();
app.on('error', err => {
assert.equal(err.message, 'mock error');
assert.equal(err.headerSent, true);
done();
});
app.use(async ctx => {
ctx.status = 200;
ctx.set('X-Foo', 'Bar');
ctx.flushHeaders();
await Promise.reject(new Error('mock error'));
ctx.body = 'response';
});
request(app.callback())
.get('/')
.expect('X-Foo', 'Bar')
.expect(200, () => {});
});
describe('when invalid err.status', () => {
describe('not number', () => {
it('should respond 500', () => {
const app = new Koa();
app.use((ctx, next) => {
ctx.body = 'something else';
const err = new Error('some error');
err.status = 'notnumber';
throw err;
});
const server = app.listen();
return request(server)
.get('/')
.expect(500)
.expect('Content-Type', 'text/plain; charset=utf-8')
.expect('Internal Server Error');
});
});
describe('when ENOENT error', () => {
it('should respond 404', () => {
const app = new Koa();
app.use((ctx, next) => {
ctx.body = 'something else';
const err = new Error('test for ENOENT');
err.code = 'ENOENT';
throw err;
});
const server = app.listen();
return request(server)
.get('/')
.expect(404)
.expect('Content-Type', 'text/plain; charset=utf-8')
.expect('Not Found');
});
});
describe('not http status code', () => {
it('should respond 500', () => {
const app = new Koa();
app.use((ctx, next) => {
ctx.body = 'something else';
const err = new Error('some error');
err.status = 9999;
throw err;
});
const server = app.listen();
return request(server)
.get('/')
.expect(500)
.expect('Content-Type', 'text/plain; charset=utf-8')
.expect('Internal Server Error');
});
});
});
describe('when non-error thrown', () => {
it('should response non-error thrown message', () => {
const app = new Koa();
app.use((ctx, next) => {
throw 'string error'; // eslint-disable-line no-throw-literal
});
const server = app.listen();
return request(server)
.get('/')
.expect(500)
.expect('Content-Type', 'text/plain; charset=utf-8')
.expect('Internal Server Error');
});
it('should use res.getHeaderNames() accessor when available', () => {
let removed = 0;
const ctx = context();
ctx.app.emit = () => {};
ctx.res = {
getHeaderNames: () => ['content-type', 'content-length'],
removeHeader: () => removed++,
end: () => {},
emit: () => {}
};
ctx.onerror(new Error('error'));
assert.equal(removed, 2);
});
it('should stringify error if it is an object', done => {
const app = new Koa();
app.on('error', err => {
assert.equal(err, 'Error: non-error thrown: {"key":"value"}');
done();
});
app.use(async ctx => {
throw {key: 'value'}; // eslint-disable-line no-throw-literal
});
request(app.callback())
.get('/')
.expect(500)
.expect('Internal Server Error', () => {});
});
});
});