UNPKG

fastify

Version:

Fast and low overhead web framework, for Node.js

428 lines (351 loc) 12.2 kB
'use strict' const stream = require('node:stream') const t = require('tap') const split = require('split2') const pino = require('pino') const Fastify = require('../../fastify') const helper = require('../helper') const { once, on } = stream const { request } = require('./logger-test-utils') t.test('logging', (t) => { t.setTimeout(60000) let localhost let localhostForURL t.plan(13) t.before(async function () { [localhost, localhostForURL] = await helper.getLoopbackHost() }) t.test('The default 404 handler logs the incoming request', async (t) => { const lines = ['incoming request', 'Route GET:/not-found not found', 'request completed'] t.plan(lines.length + 1) const stream = split(JSON.parse) const logger = pino({ level: 'trace' }, stream) const fastify = Fastify({ logger }) t.teardown(fastify.close.bind(fastify)) await fastify.ready() { const response = await fastify.inject({ method: 'GET', url: '/not-found' }) t.equal(response.statusCode, 404) } for await (const [line] of on(stream, 'data')) { t.equal(line.msg, lines.shift()) if (lines.length === 0) break } }) t.test('should not rely on raw request to log errors', async (t) => { const lines = [ { msg: /Server listening at/ }, { level: 30, msg: 'incoming request' }, { res: { statusCode: 415 }, msg: 'something happened' }, { res: { statusCode: 415 }, msg: 'request completed' } ] t.plan(lines.length + 1) const stream = split(JSON.parse) const fastify = Fastify({ logger: { stream, level: 'info' } }) t.teardown(fastify.close.bind(fastify)) fastify.get('/error', function (req, reply) { t.ok(req.log) reply.status(415).send(new Error('something happened')) }) await fastify.ready() await fastify.listen({ port: 0, host: localhost }) await request(`http://${localhostForURL}:` + fastify.server.address().port + '/error') for await (const [line] of on(stream, 'data')) { t.match(line, lines.shift()) if (lines.length === 0) break } }) t.test('should log the error if no error handler is defined', async (t) => { const lines = [ { msg: /Server listening at/ }, { msg: 'incoming request' }, { level: 50, msg: 'a generic error' }, { res: { statusCode: 500 }, msg: 'request completed' } ] t.plan(lines.length + 1) const stream = split(JSON.parse) const fastify = Fastify({ logger: { stream, level: 'info' } }) t.teardown(fastify.close.bind(fastify)) fastify.get('/error', function (req, reply) { t.ok(req.log) reply.send(new Error('a generic error')) }) await fastify.ready() await fastify.listen({ port: 0, host: localhost }) await request(`http://${localhostForURL}:` + fastify.server.address().port + '/error') for await (const [line] of on(stream, 'data')) { t.match(line, lines.shift()) if (lines.length === 0) break } }) t.test('should log as info if error status code >= 400 and < 500 if no error handler is defined', async (t) => { const lines = [ { msg: /Server listening at/ }, { msg: 'incoming request' }, { level: 30, msg: 'a 400 error' }, { res: { statusCode: 400 }, msg: 'request completed' } ] t.plan(lines.length + 1) const stream = split(JSON.parse) const fastify = Fastify({ logger: { stream, level: 'info' } }) t.teardown(fastify.close.bind(fastify)) fastify.get('/400', function (req, reply) { t.ok(req.log) reply.send(Object.assign(new Error('a 400 error'), { statusCode: 400 })) }) fastify.get('/503', function (req, reply) { t.ok(req.log) reply.send(Object.assign(new Error('a 503 error'), { statusCode: 503 })) }) await fastify.ready() await fastify.listen({ port: 0, host: localhost }) await request(`http://${localhostForURL}:` + fastify.server.address().port + '/400') for await (const [line] of on(stream, 'data')) { t.match(line, lines.shift()) if (lines.length === 0) break } }) t.test('should log as error if error status code >= 500 if no error handler is defined', async (t) => { const lines = [ { msg: /Server listening at/ }, { msg: 'incoming request' }, { level: 50, msg: 'a 503 error' }, { res: { statusCode: 503 }, msg: 'request completed' } ] t.plan(lines.length + 1) const stream = split(JSON.parse) const fastify = Fastify({ logger: { stream, level: 'info' } }) t.teardown(fastify.close.bind(fastify)) fastify.get('/503', function (req, reply) { t.ok(req.log) reply.send(Object.assign(new Error('a 503 error'), { statusCode: 503 })) }) await fastify.ready() await fastify.listen({ port: 0, host: localhost }) await request(`http://${localhostForURL}:` + fastify.server.address().port + '/503') for await (const [line] of on(stream, 'data')) { t.match(line, lines.shift()) if (lines.length === 0) break } }) t.test('should not log the error if error handler is defined and it does not error', async (t) => { const lines = [ { msg: /Server listening at/ }, { level: 30, msg: 'incoming request' }, { res: { statusCode: 200 }, msg: 'request completed' } ] t.plan(lines.length + 2) const stream = split(JSON.parse) const fastify = Fastify({ logger: { stream, level: 'info' } }) t.teardown(fastify.close.bind(fastify)) fastify.get('/error', function (req, reply) { t.ok(req.log) reply.send(new Error('something happened')) }) fastify.setErrorHandler((err, req, reply) => { t.ok(err) reply.send('something bad happened') }) await fastify.ready() await fastify.listen({ port: 0, host: localhost }) await request(`http://${localhostForURL}:` + fastify.server.address().port + '/error') for await (const [line] of on(stream, 'data')) { t.match(line, lines.shift()) if (lines.length === 0) break } }) t.test('reply.send logs an error if called twice in a row', async (t) => { const lines = [ 'incoming request', 'request completed', 'Reply was already sent, did you forget to "return reply" in "/" (GET)?', 'Reply was already sent, did you forget to "return reply" in "/" (GET)?' ] t.plan(lines.length + 1) const stream = split(JSON.parse) const logger = pino(stream) const fastify = Fastify({ logger }) t.teardown(fastify.close.bind(fastify)) fastify.get('/', (req, reply) => { reply.send({ hello: 'world' }) reply.send({ hello: 'world2' }) reply.send({ hello: 'world3' }) }) const response = await fastify.inject({ method: 'GET', url: '/' }) const body = await response.json() t.same(body, { hello: 'world' }) for await (const [line] of on(stream, 'data')) { t.same(line.msg, lines.shift()) if (lines.length === 0) break } }) t.test('should not log incoming request and outgoing response when disabled', async (t) => { t.plan(1) const stream = split(JSON.parse) const fastify = Fastify({ disableRequestLogging: true, logger: { level: 'info', stream } }) t.teardown(fastify.close.bind(fastify)) fastify.get('/500', (req, reply) => { reply.code(500).send(Error('500 error')) }) await fastify.ready() await fastify.inject({ method: 'GET', url: '/500' }) // no more readable data t.equal(stream.readableLength, 0) }) t.test('should not log incoming request, outgoing response and route not found for 404 onBadUrl when disabled', async (t) => { t.plan(1) const stream = split(JSON.parse) const fastify = Fastify({ disableRequestLogging: true, logger: { level: 'info', stream } }) t.teardown(fastify.close.bind(fastify)) await fastify.ready() await fastify.inject({ method: 'GET', url: '/%c0' }) // no more readable data t.equal(stream.readableLength, 0) }) t.test('defaults to info level', async (t) => { const lines = [ { reqId: /req-/, req: { method: 'GET' }, msg: 'incoming request' }, { reqId: /req-/, res: { statusCode: 200 }, msg: 'request completed' } ] t.plan(lines.length * 2 + 1) const stream = split(JSON.parse) const fastify = Fastify({ logger: { stream } }) t.teardown(fastify.close.bind(fastify)) fastify.get('/', function (req, reply) { t.ok(req.log) reply.send({ hello: 'world' }) }) await fastify.ready() await fastify.listen({ port: 0 }) await request(`http://${localhostForURL}:` + fastify.server.address().port) let id for await (const [line] of on(stream, 'data')) { // we skip the non-request log if (typeof line.reqId !== 'string') continue if (id === undefined && line.reqId) id = line.reqId if (id !== undefined && line.reqId) t.equal(line.reqId, id) t.match(line, lines.shift()) if (lines.length === 0) break } }) t.test('test log stream', async (t) => { const lines = [ { msg: /^Server listening at / }, { reqId: /req-/, req: { method: 'GET' }, msg: 'incoming request' }, { reqId: /req-/, res: { statusCode: 200 }, msg: 'request completed' } ] t.plan(lines.length + 3) const stream = split(JSON.parse) const fastify = Fastify({ logger: { stream, level: 'info' } }) t.teardown(fastify.close.bind(fastify)) fastify.get('/', function (req, reply) { t.ok(req.log) reply.send({ hello: 'world' }) }) await fastify.ready() await fastify.listen({ port: 0, host: localhost }) await request(`http://${localhostForURL}:` + fastify.server.address().port) let id for await (const [line] of on(stream, 'data')) { if (id === undefined && line.reqId) id = line.reqId if (id !== undefined && line.reqId) t.equal(line.reqId, id) t.match(line, lines.shift()) if (lines.length === 0) break } }) t.test('test error log stream', async (t) => { const lines = [ { msg: /^Server listening at / }, { reqId: /req-/, req: { method: 'GET' }, msg: 'incoming request' }, { reqId: /req-/, res: { statusCode: 500 }, msg: 'kaboom' }, { reqId: /req-/, res: { statusCode: 500 }, msg: 'request completed' } ] t.plan(lines.length + 4) const stream = split(JSON.parse) const fastify = Fastify({ logger: { stream, level: 'info' } }) t.teardown(fastify.close.bind(fastify)) fastify.get('/error', function (req, reply) { t.ok(req.log) reply.send(new Error('kaboom')) }) await fastify.ready() await fastify.listen({ port: 0, host: localhost }) await request(`http://${localhostForURL}:` + fastify.server.address().port + '/error') let id for await (const [line] of on(stream, 'data')) { if (id === undefined && line.reqId) id = line.reqId if (id !== undefined && line.reqId) t.equal(line.reqId, id) t.match(line, lines.shift()) if (lines.length === 0) break } }) t.test('should not log the error if request logging is disabled', async (t) => { t.plan(4) const stream = split(JSON.parse) const fastify = Fastify({ logger: { stream, level: 'info' }, disableRequestLogging: true }) t.teardown(fastify.close.bind(fastify)) fastify.get('/error', function (req, reply) { t.ok(req.log) reply.send(new Error('a generic error')) }) await fastify.ready() await fastify.listen({ port: 0, host: localhost }) await request(`http://${localhostForURL}:` + fastify.server.address().port + '/error') { const [line] = await once(stream, 'data') t.type(line.msg, 'string') t.ok(line.msg.startsWith('Server listening at'), 'message is set') } // no more readable data t.equal(stream.readableLength, 0) }) })