You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
554 lines
12 KiB
554 lines
12 KiB
'use strict'
|
|
|
|
const { Readable } = require('stream')
|
|
const test = require('tap').test
|
|
const sget = require('simple-get').concat
|
|
const Fastify = require('../')
|
|
|
|
process.removeAllListeners('warning')
|
|
|
|
function endRouteHook (doneOrPayload, done) {
|
|
if (typeof doneOrPayload === 'function') {
|
|
doneOrPayload()
|
|
} else {
|
|
done()
|
|
}
|
|
}
|
|
|
|
function testExecutionHook (hook) {
|
|
test(`${hook}`, t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
|
|
fastify.post('/', {
|
|
[hook]: (req, reply, doneOrPayload, done) => {
|
|
t.pass('hook called')
|
|
endRouteHook(doneOrPayload, done)
|
|
}
|
|
}, (req, reply) => {
|
|
reply.send(req.body)
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
const payload = JSON.parse(res.payload)
|
|
t.same(payload, { hello: 'world' })
|
|
})
|
|
})
|
|
|
|
test(`${hook} option should be called after ${hook} hook`, t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
const checker = Object.defineProperty({ calledTimes: 0 }, 'check', {
|
|
get: function () { return ++this.calledTimes }
|
|
})
|
|
|
|
fastify.addHook(hook, (req, reply, doneOrPayload, done) => {
|
|
t.equal(checker.check, 1)
|
|
endRouteHook(doneOrPayload, done)
|
|
})
|
|
|
|
fastify.post('/', {
|
|
[hook]: (req, reply, doneOrPayload, done) => {
|
|
t.equal(checker.check, 2)
|
|
endRouteHook(doneOrPayload, done)
|
|
}
|
|
}, (req, reply) => {
|
|
reply.send({})
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
})
|
|
})
|
|
|
|
test(`${hook} option could accept an array of functions`, t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
const checker = Object.defineProperty({ calledTimes: 0 }, 'check', {
|
|
get: function () { return ++this.calledTimes }
|
|
})
|
|
|
|
fastify.post('/', {
|
|
[hook]: [
|
|
(req, reply, doneOrPayload, done) => {
|
|
t.equal(checker.check, 1)
|
|
endRouteHook(doneOrPayload, done)
|
|
},
|
|
(req, reply, doneOrPayload, done) => {
|
|
t.equal(checker.check, 2)
|
|
endRouteHook(doneOrPayload, done)
|
|
}
|
|
]
|
|
}, (req, reply) => {
|
|
reply.send({})
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
})
|
|
})
|
|
|
|
test(`${hook} option does not interfere with ${hook} hook`, t => {
|
|
t.plan(7)
|
|
const fastify = Fastify()
|
|
const checker = Object.defineProperty({ calledTimes: 0 }, 'check', {
|
|
get: function () { return ++this.calledTimes }
|
|
})
|
|
|
|
fastify.addHook(hook, (req, reply, doneOrPayload, done) => {
|
|
t.equal(checker.check, 1)
|
|
endRouteHook(doneOrPayload, done)
|
|
})
|
|
|
|
fastify.post('/', {
|
|
[hook]: (req, reply, doneOrPayload, done) => {
|
|
t.equal(checker.check, 2)
|
|
endRouteHook(doneOrPayload, done)
|
|
}
|
|
}, handler)
|
|
|
|
fastify.post('/no', handler)
|
|
|
|
function handler (req, reply) {
|
|
reply.send({})
|
|
}
|
|
|
|
fastify.inject({
|
|
method: 'post',
|
|
url: '/'
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
t.equal(checker.calledTimes, 2)
|
|
|
|
checker.calledTimes = 0
|
|
|
|
fastify.inject({
|
|
method: 'post',
|
|
url: '/no'
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
t.equal(checker.calledTimes, 1)
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
function testBeforeHandlerHook (hook) {
|
|
test(`${hook} option should be unique per route`, t => {
|
|
t.plan(4)
|
|
const fastify = Fastify()
|
|
|
|
fastify.post('/', {
|
|
[hook]: (req, reply, done) => {
|
|
req.hello = 'earth'
|
|
done()
|
|
}
|
|
}, (req, reply) => {
|
|
reply.send({ hello: req.hello })
|
|
})
|
|
|
|
fastify.post('/no', (req, reply) => {
|
|
reply.send(req.body)
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
const payload = JSON.parse(res.payload)
|
|
t.same(payload, { hello: 'earth' })
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/no',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
const payload = JSON.parse(res.payload)
|
|
t.same(payload, { hello: 'world' })
|
|
})
|
|
})
|
|
|
|
test(`${hook} option should handle errors`, t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
|
|
fastify.post('/', {
|
|
[hook]: (req, reply, done) => {
|
|
done(new Error('kaboom'))
|
|
}
|
|
}, (req, reply) => {
|
|
reply.send(req.body)
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
const payload = JSON.parse(res.payload)
|
|
t.equal(res.statusCode, 500)
|
|
t.same(payload, {
|
|
message: 'kaboom',
|
|
error: 'Internal Server Error',
|
|
statusCode: 500
|
|
})
|
|
})
|
|
})
|
|
|
|
test(`${hook} option should handle throwing objects`, t => {
|
|
t.plan(4)
|
|
const fastify = Fastify()
|
|
|
|
const myError = { myError: 'kaboom' }
|
|
|
|
fastify.setErrorHandler(async (error, request, reply) => {
|
|
t.same(error, myError, 'the error object throws by the user')
|
|
reply.send({ this: 'is', my: 'error' })
|
|
})
|
|
|
|
fastify.get('/', {
|
|
[hook]: async () => {
|
|
// eslint-disable-next-line no-throw-literal
|
|
throw myError
|
|
}
|
|
}, (req, reply) => {
|
|
t.fail('the handler must not be called')
|
|
})
|
|
|
|
fastify.inject({
|
|
url: '/',
|
|
method: 'GET'
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
t.equal(res.statusCode, 500)
|
|
t.same(res.json(), { this: 'is', my: 'error' })
|
|
})
|
|
})
|
|
|
|
test(`${hook} option should handle throwing objects by default`, t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
|
|
fastify.get('/', {
|
|
[hook]: async () => {
|
|
// eslint-disable-next-line no-throw-literal
|
|
throw { myError: 'kaboom', message: 'i am an error' }
|
|
}
|
|
}, (req, reply) => {
|
|
t.fail('the handler must not be called')
|
|
})
|
|
|
|
fastify.inject({
|
|
url: '/',
|
|
method: 'GET'
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
t.equal(res.statusCode, 500)
|
|
t.same(res.json(), { myError: 'kaboom', message: 'i am an error' })
|
|
})
|
|
})
|
|
|
|
test(`${hook} option should handle errors with custom status code`, t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
|
|
fastify.post('/', {
|
|
[hook]: (req, reply, done) => {
|
|
reply.code(401)
|
|
done(new Error('go away'))
|
|
}
|
|
}, (req, reply) => {
|
|
reply.send(req.body)
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
const payload = JSON.parse(res.payload)
|
|
t.equal(res.statusCode, 401)
|
|
t.same(payload, {
|
|
message: 'go away',
|
|
error: 'Unauthorized',
|
|
statusCode: 401
|
|
})
|
|
})
|
|
})
|
|
|
|
test(`${hook} option should keep the context`, t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
|
|
fastify.decorate('foo', 42)
|
|
|
|
fastify.post('/', {
|
|
[hook]: function (req, reply, done) {
|
|
t.equal(this.foo, 42)
|
|
this.foo += 1
|
|
done()
|
|
}
|
|
}, function (req, reply) {
|
|
reply.send({ foo: this.foo })
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
const payload = JSON.parse(res.payload)
|
|
t.same(payload, { foo: 43 })
|
|
})
|
|
})
|
|
|
|
test(`${hook} option should keep the context (array)`, t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
|
|
fastify.decorate('foo', 42)
|
|
|
|
fastify.post('/', {
|
|
[hook]: [function (req, reply, done) {
|
|
t.equal(this.foo, 42)
|
|
this.foo += 1
|
|
done()
|
|
}]
|
|
}, function (req, reply) {
|
|
reply.send({ foo: this.foo })
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
const payload = JSON.parse(res.payload)
|
|
t.same(payload, { foo: 43 })
|
|
})
|
|
})
|
|
}
|
|
|
|
testExecutionHook('preHandler')
|
|
testExecutionHook('onSend')
|
|
testExecutionHook('onRequest')
|
|
testExecutionHook('onResponse')
|
|
testExecutionHook('preValidation')
|
|
testExecutionHook('preParsing')
|
|
// hooks that comes before the handler
|
|
testBeforeHandlerHook('preHandler')
|
|
testBeforeHandlerHook('onRequest')
|
|
testBeforeHandlerHook('preValidation')
|
|
testBeforeHandlerHook('preParsing')
|
|
|
|
test('preValidation option should be called before preHandler hook', t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
|
|
fastify.addHook('preHandler', (req, reply, done) => {
|
|
t.ok(req.called)
|
|
done()
|
|
})
|
|
|
|
fastify.post('/', {
|
|
preValidation: (req, reply, done) => {
|
|
req.called = true
|
|
done()
|
|
}
|
|
}, (req, reply) => {
|
|
reply.send(req.body)
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
const payload = JSON.parse(res.payload)
|
|
t.same(payload, { hello: 'world' })
|
|
})
|
|
})
|
|
|
|
test('preSerialization option should be able to modify the payload', t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
|
|
fastify.get('/only', {
|
|
preSerialization: (req, reply, payload, done) => {
|
|
done(null, { hello: 'another world' })
|
|
}
|
|
}, (req, reply) => {
|
|
reply.send({ hello: 'world' })
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'GET',
|
|
url: '/only'
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
t.equal(res.statusCode, 200)
|
|
t.same(JSON.parse(res.payload), { hello: 'another world' })
|
|
})
|
|
})
|
|
|
|
test('preParsing option should be called before preValidation hook', t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
|
|
fastify.addHook('preValidation', (req, reply, done) => {
|
|
t.ok(req.called)
|
|
done()
|
|
})
|
|
|
|
fastify.post('/', {
|
|
preParsing: (req, reply, done) => {
|
|
req.called = true
|
|
done()
|
|
}
|
|
}, (req, reply) => {
|
|
reply.send(req.body)
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
const payload = JSON.parse(res.payload)
|
|
t.same(payload, { hello: 'world' })
|
|
})
|
|
})
|
|
|
|
test('preParsing option should be able to modify the payload', t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
|
|
fastify.post('/only', {
|
|
preParsing: (req, reply, payload, done) => {
|
|
const stream = new Readable()
|
|
stream.receivedEncodedLength = parseInt(req.headers['content-length'], 10)
|
|
stream.push(JSON.stringify({ hello: 'another world' }))
|
|
stream.push(null)
|
|
done(null, stream)
|
|
}
|
|
}, (req, reply) => {
|
|
reply.send(req.body)
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/only',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
t.equal(res.statusCode, 200)
|
|
t.same(JSON.parse(res.payload), { hello: 'another world' })
|
|
})
|
|
})
|
|
|
|
test('onRequest option should be called before preParsing', t => {
|
|
t.plan(3)
|
|
const fastify = Fastify()
|
|
|
|
fastify.addHook('preParsing', (req, reply, done) => {
|
|
t.ok(req.called)
|
|
done()
|
|
})
|
|
|
|
fastify.post('/', {
|
|
onRequest: (req, reply, done) => {
|
|
req.called = true
|
|
done()
|
|
}
|
|
}, (req, reply) => {
|
|
reply.send(req.body)
|
|
})
|
|
|
|
fastify.inject({
|
|
method: 'POST',
|
|
url: '/',
|
|
payload: { hello: 'world' }
|
|
}, (err, res) => {
|
|
t.error(err)
|
|
const payload = JSON.parse(res.payload)
|
|
t.same(payload, { hello: 'world' })
|
|
})
|
|
})
|
|
|
|
test('onTimeout on route', t => {
|
|
t.plan(4)
|
|
const fastify = Fastify({ connectionTimeout: 500 })
|
|
|
|
fastify.get('/timeout', {
|
|
async handler (request, reply) { },
|
|
onTimeout (request, reply, done) {
|
|
t.pass('onTimeout called')
|
|
done()
|
|
}
|
|
})
|
|
|
|
fastify.listen(0, (err, address) => {
|
|
t.error(err)
|
|
t.teardown(() => fastify.close())
|
|
|
|
sget({
|
|
method: 'GET',
|
|
url: `${address}/timeout`
|
|
}, (err, response, body) => {
|
|
t.type(err, Error)
|
|
t.equal(err.message, 'socket hang up')
|
|
})
|
|
})
|
|
})
|
|
|
|
test('onError on route', t => {
|
|
t.plan(3)
|
|
|
|
const fastify = Fastify()
|
|
|
|
const err = new Error('kaboom')
|
|
|
|
fastify.get('/',
|
|
{
|
|
onError (request, reply, error, done) {
|
|
t.match(error, err)
|
|
done()
|
|
}
|
|
},
|
|
(req, reply) => {
|
|
reply.send(err)
|
|
})
|
|
|
|
fastify.inject('/', (err, res) => {
|
|
t.error(err)
|
|
t.same(JSON.parse(res.payload), {
|
|
error: 'Internal Server Error',
|
|
message: 'kaboom',
|
|
statusCode: 500
|
|
})
|
|
})
|
|
})
|