Files
simple-mail-cleaner/backend/node_modules/fastify/test/web-api.test.js
2026-01-22 19:05:45 +01:00

469 lines
11 KiB
JavaScript

'use strict'
const { test } = require('node:test')
const Fastify = require('../fastify')
const fs = require('node:fs')
const { Readable } = require('node:stream')
const { fetch: undiciFetch } = require('undici')
const http = require('node:http')
test('should response with a ReadableStream', async (t) => {
t.plan(2)
const fastify = Fastify()
fastify.get('/', function (request, reply) {
const stream = fs.createReadStream(__filename)
reply.code(200).send(Readable.toWeb(stream))
})
const {
statusCode,
body
} = await fastify.inject({ method: 'GET', path: '/' })
const expected = await fs.promises.readFile(__filename)
t.assert.strictEqual(statusCode, 200)
t.assert.strictEqual(expected.toString(), body.toString())
})
test('should response with a Response', async (t) => {
t.plan(3)
const fastify = Fastify()
fastify.get('/', function (request, reply) {
const stream = fs.createReadStream(__filename)
reply.send(new Response(Readable.toWeb(stream), {
status: 200,
headers: {
hello: 'world'
}
}))
})
const {
statusCode,
headers,
body
} = await fastify.inject({ method: 'GET', path: '/' })
const expected = await fs.promises.readFile(__filename)
t.assert.strictEqual(statusCode, 200)
t.assert.strictEqual(expected.toString(), body.toString())
t.assert.strictEqual(headers.hello, 'world')
})
test('should response with a Response 204', async (t) => {
t.plan(3)
const fastify = Fastify()
fastify.get('/', function (request, reply) {
reply.send(new Response(null, {
status: 204,
headers: {
hello: 'world'
}
}))
})
const {
statusCode,
headers,
body
} = await fastify.inject({ method: 'GET', path: '/' })
t.assert.strictEqual(statusCode, 204)
t.assert.strictEqual(body, '')
t.assert.strictEqual(headers.hello, 'world')
})
test('should response with a Response 304', async (t) => {
t.plan(3)
const fastify = Fastify()
fastify.get('/', function (request, reply) {
reply.send(new Response(null, {
status: 304,
headers: {
hello: 'world'
}
}))
})
const {
statusCode,
headers,
body
} = await fastify.inject({ method: 'GET', path: '/' })
t.assert.strictEqual(statusCode, 304)
t.assert.strictEqual(body, '')
t.assert.strictEqual(headers.hello, 'world')
})
test('should response with a Response without body', async (t) => {
t.plan(3)
const fastify = Fastify()
fastify.get('/', function (request, reply) {
reply.send(new Response(null, {
status: 200,
headers: {
hello: 'world'
}
}))
})
const {
statusCode,
headers,
body
} = await fastify.inject({ method: 'GET', path: '/' })
t.assert.strictEqual(statusCode, 200)
t.assert.strictEqual(body, '')
t.assert.strictEqual(headers.hello, 'world')
})
test('able to use in onSend hook - ReadableStream', async (t) => {
t.plan(4)
const fastify = Fastify()
fastify.get('/', function (request, reply) {
const stream = fs.createReadStream(__filename)
reply.code(500).send(Readable.toWeb(stream))
})
fastify.addHook('onSend', (request, reply, payload, done) => {
t.assert.strictEqual(Object.prototype.toString.call(payload), '[object ReadableStream]')
done(null, new Response(payload, {
status: 200,
headers: {
hello: 'world'
}
}))
})
const {
statusCode,
headers,
body
} = await fastify.inject({ method: 'GET', path: '/' })
const expected = await fs.promises.readFile(__filename)
t.assert.strictEqual(statusCode, 200)
t.assert.strictEqual(expected.toString(), body.toString())
t.assert.strictEqual(headers.hello, 'world')
})
test('able to use in onSend hook - Response', async (t) => {
t.plan(4)
const fastify = Fastify()
fastify.get('/', function (request, reply) {
const stream = fs.createReadStream(__filename)
reply.send(new Response(Readable.toWeb(stream), {
status: 500,
headers: {
hello: 'world'
}
}))
})
fastify.addHook('onSend', (request, reply, payload, done) => {
t.assert.strictEqual(Object.prototype.toString.call(payload), '[object Response]')
done(null, new Response(payload.body, {
status: 200,
headers: payload.headers
}))
})
const {
statusCode,
headers,
body
} = await fastify.inject({ method: 'GET', path: '/' })
const expected = await fs.promises.readFile(__filename)
t.assert.strictEqual(statusCode, 200)
t.assert.strictEqual(expected.toString(), body.toString())
t.assert.strictEqual(headers.hello, 'world')
})
test('Error when Response.bodyUsed', async (t) => {
t.plan(4)
const expected = await fs.promises.readFile(__filename)
const fastify = Fastify()
fastify.get('/', async function (request, reply) {
const stream = fs.createReadStream(__filename)
const response = new Response(Readable.toWeb(stream), {
status: 200,
headers: {
hello: 'world'
}
})
const file = await response.text()
t.assert.strictEqual(expected.toString(), file)
t.assert.strictEqual(response.bodyUsed, true)
return reply.send(response)
})
const response = await fastify.inject({ method: 'GET', path: '/' })
t.assert.strictEqual(response.statusCode, 500)
const body = response.json()
t.assert.strictEqual(body.code, 'FST_ERR_REP_RESPONSE_BODY_CONSUMED')
})
test('Error when Response.body.locked', async (t) => {
t.plan(3)
const fastify = Fastify()
fastify.get('/', async function (request, reply) {
const stream = Readable.toWeb(fs.createReadStream(__filename))
const response = new Response(stream, {
status: 200,
headers: {
hello: 'world'
}
})
stream.getReader()
t.assert.strictEqual(stream.locked, true)
return reply.send(response)
})
const response = await fastify.inject({ method: 'GET', path: '/' })
t.assert.strictEqual(response.statusCode, 500)
const body = response.json()
t.assert.strictEqual(body.code, 'FST_ERR_REP_READABLE_STREAM_LOCKED')
})
test('Error when ReadableStream.locked', async (t) => {
t.plan(3)
const fastify = Fastify()
fastify.get('/', async function (request, reply) {
const stream = Readable.toWeb(fs.createReadStream(__filename))
stream.getReader()
t.assert.strictEqual(stream.locked, true)
return reply.send(stream)
})
const response = await fastify.inject({ method: 'GET', path: '/' })
t.assert.strictEqual(response.statusCode, 500)
const body = response.json()
t.assert.strictEqual(body.code, 'FST_ERR_REP_READABLE_STREAM_LOCKED')
})
test('allow to pipe with fetch', async (t) => {
t.plan(2)
const abortController = new AbortController()
const { signal } = abortController
const fastify = Fastify()
t.after(() => {
fastify.close()
abortController.abort()
})
fastify.get('/', function (request, reply) {
return fetch(`${fastify.listeningOrigin}/fetch`, {
method: 'GET',
signal
})
})
fastify.get('/fetch', function async (request, reply) {
reply.code(200).send({ ok: true })
})
await fastify.listen()
const response = await fastify.inject({ method: 'GET', path: '/' })
t.assert.strictEqual(response.statusCode, 200)
t.assert.deepStrictEqual(response.json(), { ok: true })
})
test('allow to pipe with undici.fetch', async (t) => {
t.plan(2)
const abortController = new AbortController()
const { signal } = abortController
const fastify = Fastify()
t.after(() => {
fastify.close()
abortController.abort()
})
fastify.get('/', function (request, reply) {
return undiciFetch(`${fastify.listeningOrigin}/fetch`, {
method: 'GET',
signal
})
})
fastify.get('/fetch', function (request, reply) {
reply.code(200).send({ ok: true })
})
await fastify.listen()
const response = await fastify.inject({ method: 'GET', path: '/' })
t.assert.strictEqual(response.statusCode, 200)
t.assert.deepStrictEqual(response.json(), { ok: true })
})
test('WebStream error before headers sent should trigger error handler', async (t) => {
t.plan(2)
const fastify = Fastify()
fastify.get('/', function (request, reply) {
const stream = new ReadableStream({
start (controller) {
controller.error(new Error('stream error'))
}
})
reply.send(stream)
})
const response = await fastify.inject({ method: 'GET', path: '/' })
t.assert.strictEqual(response.statusCode, 500)
t.assert.strictEqual(response.json().message, 'stream error')
})
test('WebStream error after headers sent should destroy response', (t, done) => {
t.plan(2)
const fastify = Fastify()
t.after(() => fastify.close())
fastify.get('/', function (request, reply) {
const stream = new ReadableStream({
start (controller) {
controller.enqueue('hello')
},
pull (controller) {
setTimeout(() => {
controller.error(new Error('stream error'))
}, 10)
}
})
reply.header('content-type', 'text/plain').send(stream)
})
fastify.listen({ port: 0 }, err => {
t.assert.ifError(err)
let finished = false
http.get(`http://localhost:${fastify.server.address().port}`, (res) => {
res.on('close', () => {
if (!finished) {
finished = true
t.assert.ok('response closed')
done()
}
})
res.resume()
})
})
})
test('WebStream should cancel reader when response is destroyed', (t, done) => {
t.plan(2)
const fastify = Fastify()
t.after(() => fastify.close())
let readerCancelled = false
fastify.get('/', function (request, reply) {
const stream = new ReadableStream({
start (controller) {
controller.enqueue('hello')
},
pull (controller) {
return new Promise(() => {})
},
cancel () {
readerCancelled = true
}
})
reply.header('content-type', 'text/plain').send(stream)
})
fastify.listen({ port: 0 }, err => {
t.assert.ifError(err)
const req = http.get(`http://localhost:${fastify.server.address().port}`, (res) => {
res.once('data', () => {
req.destroy()
setTimeout(() => {
t.assert.strictEqual(readerCancelled, true)
done()
}, 50)
})
})
})
})
test('WebStream should warn when headers already sent', async (t) => {
t.plan(2)
let warnCalled = false
const spyLogger = {
level: 'warn',
fatal: () => { },
error: () => { },
warn: (msg) => {
if (typeof msg === 'string' && msg.includes('use res.writeHead in stream mode')) {
warnCalled = true
}
},
info: () => { },
debug: () => { },
trace: () => { },
child: () => spyLogger
}
const fastify = Fastify({ loggerInstance: spyLogger })
t.after(() => fastify.close())
fastify.get('/', function (request, reply) {
reply.raw.writeHead(200, { 'content-type': 'text/plain' })
const stream = new ReadableStream({
start (controller) {
controller.enqueue('hello')
controller.close()
}
})
reply.send(stream)
})
await fastify.listen({ port: 0 })
const response = await fetch(`http://localhost:${fastify.server.address().port}/`)
t.assert.strictEqual(response.status, 200)
t.assert.strictEqual(warnCalled, true)
})