469 lines
11 KiB
JavaScript
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)
|
|
})
|