fastify
Version:
Fast and low overhead web framework, for Node.js
382 lines (324 loc) • 11.5 kB
JavaScript
'use strict'
const { test, before } = require('node:test')
const Fastify = require('../..')
const helper = require('../helper')
const http = require('node:http')
const pino = require('pino')
const split = require('split2')
const deepClone = require('rfdc')({ circles: true, proto: false })
const { deepFreezeObject } = require('../../lib/initialConfigValidation').utils
const { buildCertificate } = require('../build-certificate')
process.removeAllListeners('warning')
let localhost
let localhostForURL
before(async function () {
await buildCertificate();
[localhost, localhostForURL] = await helper.getLoopbackHost()
})
test('Fastify.initialConfig is an object', t => {
t.plan(1)
t.assert.ok(typeof Fastify().initialConfig === 'object')
})
test('without options passed to Fastify, initialConfig should expose default values', t => {
t.plan(1)
const fastifyDefaultOptions = {
connectionTimeout: 0,
keepAliveTimeout: 72000,
maxRequestsPerSocket: 0,
requestTimeout: 0,
bodyLimit: 1024 * 1024,
caseSensitive: true,
allowUnsafeRegex: false,
disableRequestLogging: false,
ignoreTrailingSlash: false,
ignoreDuplicateSlashes: false,
maxParamLength: 100,
onProtoPoisoning: 'error',
onConstructorPoisoning: 'error',
pluginTimeout: 10000,
requestIdHeader: false,
requestIdLogLabel: 'reqId',
http2SessionTimeout: 72000,
exposeHeadRoutes: true,
useSemicolonDelimiter: false
}
t.assert.deepStrictEqual(Fastify().initialConfig, fastifyDefaultOptions)
})
test('Fastify.initialConfig should expose all options', t => {
t.plan(22)
const serverFactory = (handler, opts) => {
const server = http.createServer((req, res) => {
handler(req, res)
})
return server
}
const versionStrategy = {
name: 'version',
storage: function () {
const versions = {}
return {
get: (version) => { return versions[version] || null },
set: (version, store) => { versions[version] = store }
}
},
deriveConstraint: (req, ctx) => {
return req.headers.accept
},
validate () { return true }
}
let reqId = 0
const options = {
http2: true,
https: {
key: global.context.key,
cert: global.context.cert
},
ignoreTrailingSlash: true,
ignoreDuplicateSlashes: true,
maxParamLength: 200,
connectionTimeout: 0,
keepAliveTimeout: 72000,
bodyLimit: 1049600,
onProtoPoisoning: 'remove',
serverFactory,
caseSensitive: true,
allowUnsafeRegex: false,
requestIdHeader: 'request-id-alt',
pluginTimeout: 20000,
useSemicolonDelimiter: false,
querystringParser: str => str,
genReqId: function (req) {
return reqId++
},
loggerInstance: pino({ level: 'info' }),
constraints: {
version: versionStrategy
},
trustProxy: function myTrustFn (address, hop) {
return address === '1.2.3.4' || hop === 1
}
}
const fastify = Fastify(options)
t.assert.strictEqual(fastify.initialConfig.http2, true)
t.assert.strictEqual(fastify.initialConfig.https, true, 'for security reason the key cert is hidden')
t.assert.strictEqual(fastify.initialConfig.ignoreTrailingSlash, true)
t.assert.strictEqual(fastify.initialConfig.ignoreDuplicateSlashes, true)
t.assert.strictEqual(fastify.initialConfig.maxParamLength, 200)
t.assert.strictEqual(fastify.initialConfig.connectionTimeout, 0)
t.assert.strictEqual(fastify.initialConfig.keepAliveTimeout, 72000)
t.assert.strictEqual(fastify.initialConfig.bodyLimit, 1049600)
t.assert.strictEqual(fastify.initialConfig.onProtoPoisoning, 'remove')
t.assert.strictEqual(fastify.initialConfig.caseSensitive, true)
t.assert.strictEqual(fastify.initialConfig.useSemicolonDelimiter, false)
t.assert.strictEqual(fastify.initialConfig.allowUnsafeRegex, false)
t.assert.strictEqual(fastify.initialConfig.requestIdHeader, 'request-id-alt')
t.assert.strictEqual(fastify.initialConfig.pluginTimeout, 20000)
t.assert.ok(fastify.initialConfig.constraints.version)
// obfuscated options:
t.assert.strictEqual(fastify.initialConfig.serverFactory, undefined)
t.assert.strictEqual(fastify.initialConfig.trustProxy, undefined)
t.assert.strictEqual(fastify.initialConfig.genReqId, undefined)
t.assert.strictEqual(fastify.initialConfig.childLoggerFactory, undefined)
t.assert.strictEqual(fastify.initialConfig.querystringParser, undefined)
t.assert.strictEqual(fastify.initialConfig.logger, undefined)
t.assert.strictEqual(fastify.initialConfig.trustProxy, undefined)
})
test('Should throw if you try to modify Fastify.initialConfig', t => {
t.plan(4)
const fastify = Fastify({ ignoreTrailingSlash: true })
try {
fastify.initialConfig.ignoreTrailingSlash = false
t.assert.fail()
} catch (error) {
t.assert.ok(error instanceof TypeError)
t.assert.strictEqual(error.message, "Cannot assign to read only property 'ignoreTrailingSlash' of object '#<Object>'")
t.assert.ok(error.stack)
t.assert.ok(true)
}
})
test('We must avoid shallow freezing and ensure that the whole object is freezed', t => {
t.plan(4)
const fastify = Fastify({
https: {
allowHTTP1: true,
key: global.context.key,
cert: global.context.cert
}
})
try {
fastify.initialConfig.https.allowHTTP1 = false
t.assert.fail()
} catch (error) {
t.assert.ok(error instanceof TypeError)
t.assert.strictEqual(error.message, "Cannot assign to read only property 'allowHTTP1' of object '#<Object>'")
t.assert.ok(error.stack)
t.assert.deepStrictEqual(fastify.initialConfig.https, {
allowHTTP1: true
}, 'key cert removed')
}
})
test('https value check', t => {
t.plan(1)
const fastify = Fastify({})
t.assert.ok(!fastify.initialConfig.https)
})
test('Return an error if options do not match the validation schema', t => {
t.plan(6)
try {
Fastify({ ignoreTrailingSlash: 'string instead of boolean' })
t.assert.fail()
} catch (error) {
t.assert.ok(error instanceof Error)
t.assert.strictEqual(error.name, 'FastifyError')
t.assert.strictEqual(error.message, 'Invalid initialization options: \'["must be boolean"]\'')
t.assert.strictEqual(error.code, 'FST_ERR_INIT_OPTS_INVALID')
t.assert.ok(error.stack)
t.assert.ok(true)
}
})
test('Original options must not be frozen', t => {
t.plan(4)
const originalOptions = {
https: {
allowHTTP1: true,
key: global.context.key,
cert: global.context.cert
}
}
const fastify = Fastify(originalOptions)
t.assert.strictEqual(Object.isFrozen(originalOptions), false)
t.assert.strictEqual(Object.isFrozen(originalOptions.https), false)
t.assert.strictEqual(Object.isFrozen(fastify.initialConfig), true)
t.assert.strictEqual(Object.isFrozen(fastify.initialConfig.https), true)
})
test('Original options must not be altered (test deep cloning)', t => {
t.plan(3)
const originalOptions = {
https: {
allowHTTP1: true,
key: global.context.key,
cert: global.context.cert
}
}
const originalOptionsClone = deepClone(originalOptions)
const fastify = Fastify(originalOptions)
// initialConfig has been triggered
t.assert.strictEqual(Object.isFrozen(fastify.initialConfig), true)
// originalOptions must not have been altered
t.assert.deepStrictEqual(originalOptions.https.key, originalOptionsClone.https.key)
t.assert.deepStrictEqual(originalOptions.https.cert, originalOptionsClone.https.cert)
})
test('Should not have issues when passing stream options to Pino.js', (t, done) => {
t.plan(17)
const stream = split(JSON.parse)
const originalOptions = {
ignoreTrailingSlash: true,
logger: {
level: 'trace',
stream
}
}
let fastify
try {
fastify = Fastify(originalOptions)
fastify.setChildLoggerFactory(function (logger, bindings, opts) {
bindings.someBinding = 'value'
return logger.child(bindings, opts)
})
t.assert.ok(typeof fastify === 'object')
t.assert.deepStrictEqual(fastify.initialConfig, {
connectionTimeout: 0,
keepAliveTimeout: 72000,
maxRequestsPerSocket: 0,
requestTimeout: 0,
bodyLimit: 1024 * 1024,
caseSensitive: true,
allowUnsafeRegex: false,
disableRequestLogging: false,
ignoreTrailingSlash: true,
ignoreDuplicateSlashes: false,
maxParamLength: 100,
onProtoPoisoning: 'error',
onConstructorPoisoning: 'error',
pluginTimeout: 10000,
requestIdHeader: false,
requestIdLogLabel: 'reqId',
http2SessionTimeout: 72000,
exposeHeadRoutes: true,
useSemicolonDelimiter: false
})
} catch (error) {
t.assert.fail()
}
fastify.get('/', function (req, reply) {
t.assert.ok(req.log)
reply.send({ hello: 'world' })
})
stream.once('data', listenAtLogLine => {
t.assert.ok(listenAtLogLine, 'listen at log message is ok')
stream.once('data', line => {
const id = line.reqId
t.assert.ok(line.reqId, 'reqId is defined')
t.assert.strictEqual(line.someBinding, 'value', 'child logger binding is set')
t.assert.ok(line.req, 'req is defined')
t.assert.strictEqual(line.msg, 'incoming request', 'message is set')
t.assert.strictEqual(line.req.method, 'GET', 'method is get')
stream.once('data', line => {
t.assert.strictEqual(line.reqId, id)
t.assert.ok(line.reqId, 'reqId is defined')
t.assert.strictEqual(line.someBinding, 'value', 'child logger binding is set')
t.assert.ok(line.res, 'res is defined')
t.assert.strictEqual(line.msg, 'request completed', 'message is set')
t.assert.strictEqual(line.res.statusCode, 200, 'statusCode is 200')
t.assert.ok(line.responseTime, 'responseTime is defined')
})
})
})
fastify.listen({ port: 0, host: localhost }, err => {
t.assert.ifError(err)
t.after(() => { fastify.close() })
http.get(`http://${localhostForURL}:${fastify.server.address().port}`, () => {
done()
})
})
})
test('deepFreezeObject() should not throw on TypedArray', t => {
t.plan(5)
const object = {
buffer: Buffer.from(global.context.key),
dataView: new DataView(new ArrayBuffer(16)),
float: 1.1,
integer: 1,
object: {
nested: { string: 'string' }
},
stream: split(JSON.parse),
string: 'string'
}
try {
const frozenObject = deepFreezeObject(object)
// Buffers should not be frozen, as they are Uint8Array inherited instances
t.assert.strictEqual(Object.isFrozen(frozenObject.buffer), false)
t.assert.strictEqual(Object.isFrozen(frozenObject), true)
t.assert.strictEqual(Object.isFrozen(frozenObject.object), true)
t.assert.strictEqual(Object.isFrozen(frozenObject.object.nested), true)
t.assert.ok(true)
} catch (error) {
t.assert.fail()
}
})
test('pluginTimeout should be parsed correctly', t => {
const withDisabledTimeout = Fastify({ pluginTimeout: '0' })
t.assert.strictEqual(withDisabledTimeout.initialConfig.pluginTimeout, 0)
const withInvalidTimeout = Fastify({ pluginTimeout: undefined })
t.assert.strictEqual(withInvalidTimeout.initialConfig.pluginTimeout, 10000)
})
test('Should not mutate the options object outside Fastify', async t => {
const options = Object.freeze({})
try {
Fastify(options)
t.assert.ok(true)
} catch (error) {
t.assert.fail(error.message)
}
})