2024-01-25 16:17:31 +00:00
|
|
|
import * as assert from 'node:assert/strict';
|
2024-02-21 14:08:19 +00:00
|
|
|
import crypto from 'node:crypto';
|
|
|
|
import { after, before, describe, it } from 'node:test';
|
|
|
|
import nodejs from '../dist/index.js';
|
|
|
|
import { createRequestAndResponse, loadFixture } from './test-utils.js';
|
2022-07-22 15:22:31 -04:00
|
|
|
|
|
|
|
describe('API routes', () => {
|
|
|
|
/** @type {import('./test-utils').Fixture} */
|
|
|
|
let fixture;
|
2024-02-01 21:24:21 +00:00
|
|
|
/** @type {import('astro/src/@types/astro.js').PreviewServer} */
|
|
|
|
let previewServer;
|
|
|
|
/** @type {URL} */
|
|
|
|
let baseUri;
|
2022-07-22 15:22:31 -04:00
|
|
|
|
|
|
|
before(async () => {
|
|
|
|
fixture = await loadFixture({
|
|
|
|
root: './fixtures/api-route/',
|
2022-07-25 00:18:02 -04:00
|
|
|
output: 'server',
|
2022-10-12 17:25:51 -04:00
|
|
|
adapter: nodejs({ mode: 'middleware' }),
|
2022-07-22 15:22:31 -04:00
|
|
|
});
|
|
|
|
await fixture.build();
|
2024-02-01 21:24:21 +00:00
|
|
|
previewServer = await fixture.preview();
|
|
|
|
baseUri = new URL(`http://${previewServer.host ?? 'localhost'}:${previewServer.port}/`);
|
2022-07-22 15:22:31 -04:00
|
|
|
});
|
|
|
|
|
2024-02-01 21:24:21 +00:00
|
|
|
after(() => previewServer.stop());
|
|
|
|
|
2022-07-22 15:22:31 -04:00
|
|
|
it('Can get the request body', async () => {
|
|
|
|
const { handler } = await import('./fixtures/api-route/dist/server/entry.mjs');
|
|
|
|
let { req, res, done } = createRequestAndResponse({
|
|
|
|
method: 'POST',
|
2022-07-22 19:24:58 +00:00
|
|
|
url: '/recipes',
|
2022-07-22 15:22:31 -04:00
|
|
|
});
|
|
|
|
|
2023-08-15 16:26:18 +02:00
|
|
|
req.once('async_iterator', () => {
|
|
|
|
req.send(JSON.stringify({ id: 2 }));
|
|
|
|
});
|
2022-07-22 15:22:31 -04:00
|
|
|
|
2023-08-15 16:26:18 +02:00
|
|
|
handler(req, res);
|
2023-01-10 17:01:27 +00:00
|
|
|
|
2022-07-22 19:24:58 +00:00
|
|
|
let [buffer] = await done;
|
2023-01-11 00:59:20 +08:00
|
|
|
|
2022-07-22 15:22:31 -04:00
|
|
|
let json = JSON.parse(buffer.toString('utf-8'));
|
2023-01-11 00:59:20 +08:00
|
|
|
|
2024-01-25 16:17:31 +00:00
|
|
|
assert.equal(json.length, 1);
|
2023-01-10 17:01:27 +00:00
|
|
|
|
2024-01-25 16:17:31 +00:00
|
|
|
assert.equal(json[0].name, 'Broccoli Soup');
|
2022-07-22 15:22:31 -04:00
|
|
|
});
|
2022-07-26 10:31:54 -04:00
|
|
|
|
|
|
|
it('Can get binary data', async () => {
|
|
|
|
const { handler } = await import('./fixtures/api-route/dist/server/entry.mjs');
|
|
|
|
|
|
|
|
let { req, res, done } = createRequestAndResponse({
|
|
|
|
method: 'POST',
|
|
|
|
url: '/binary',
|
|
|
|
});
|
|
|
|
|
2023-08-15 16:26:18 +02:00
|
|
|
req.once('async_iterator', () => {
|
|
|
|
req.send(Buffer.from(new Uint8Array([1, 2, 3, 4, 5])));
|
|
|
|
});
|
|
|
|
|
2022-07-26 10:31:54 -04:00
|
|
|
handler(req, res);
|
|
|
|
|
|
|
|
let [out] = await done;
|
|
|
|
let arr = Array.from(new Uint8Array(out.buffer));
|
2024-01-25 16:17:31 +00:00
|
|
|
assert.deepEqual(arr, [5, 4, 3, 2, 1]);
|
2022-07-26 10:31:54 -04:00
|
|
|
});
|
2023-08-15 16:26:18 +02:00
|
|
|
|
|
|
|
it('Can post large binary data', async () => {
|
|
|
|
const { handler } = await import('./fixtures/api-route/dist/server/entry.mjs');
|
|
|
|
|
|
|
|
let { req, res, done } = createRequestAndResponse({
|
|
|
|
method: 'POST',
|
|
|
|
url: '/hash',
|
|
|
|
});
|
|
|
|
|
|
|
|
handler(req, res);
|
|
|
|
|
|
|
|
let expectedDigest = null;
|
|
|
|
req.once('async_iterator', () => {
|
|
|
|
// Send 256MB of garbage data in 256KB chunks. This should be fast (< 1sec).
|
|
|
|
let remainingBytes = 256 * 1024 * 1024;
|
|
|
|
const chunkSize = 256 * 1024;
|
|
|
|
|
|
|
|
const hash = crypto.createHash('sha256');
|
|
|
|
while (remainingBytes > 0) {
|
|
|
|
const size = Math.min(remainingBytes, chunkSize);
|
|
|
|
const chunk = Buffer.alloc(size, Math.floor(Math.random() * 256));
|
|
|
|
hash.update(chunk);
|
|
|
|
req.emit('data', chunk);
|
|
|
|
remainingBytes -= size;
|
|
|
|
}
|
|
|
|
|
|
|
|
req.emit('end');
|
|
|
|
expectedDigest = hash.digest();
|
|
|
|
});
|
|
|
|
|
|
|
|
let [out] = await done;
|
2024-01-25 16:17:31 +00:00
|
|
|
assert.deepEqual(new Uint8Array(out.buffer), new Uint8Array(expectedDigest));
|
2023-08-15 16:26:18 +02:00
|
|
|
});
|
2023-11-16 08:39:41 +09:00
|
|
|
|
|
|
|
it('Can bail on streaming', async () => {
|
|
|
|
const { handler } = await import('./fixtures/api-route/dist/server/entry.mjs');
|
|
|
|
let { req, res, done } = createRequestAndResponse({
|
|
|
|
url: '/streaming',
|
|
|
|
});
|
|
|
|
|
|
|
|
let locals = { cancelledByTheServer: false };
|
|
|
|
|
|
|
|
handler(req, res, () => {}, locals);
|
|
|
|
req.send();
|
|
|
|
|
|
|
|
await new Promise((resolve) => setTimeout(resolve, 500));
|
2023-11-15 23:41:21 +00:00
|
|
|
res.emit('close');
|
2023-11-16 08:39:41 +09:00
|
|
|
|
|
|
|
await done;
|
|
|
|
|
2024-01-25 16:17:31 +00:00
|
|
|
assert.deepEqual(locals, { cancelledByTheServer: true });
|
2023-11-16 08:39:41 +09:00
|
|
|
});
|
2024-02-01 21:24:21 +00:00
|
|
|
|
|
|
|
it('Can respond with SSR redirect', async () => {
|
|
|
|
const controller = new AbortController();
|
|
|
|
setTimeout(() => controller.abort(), 1000);
|
|
|
|
const response = await fetch(new URL('/redirect', baseUri), {
|
|
|
|
redirect: 'manual',
|
|
|
|
signal: controller.signal,
|
|
|
|
});
|
|
|
|
assert.equal(response.status, 302);
|
|
|
|
assert.equal(response.headers.get('location'), '/destination');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('Can respond with Astro.redirect', async () => {
|
|
|
|
const controller = new AbortController();
|
|
|
|
setTimeout(() => controller.abort(), 1000);
|
|
|
|
const response = await fetch(new URL('/astro-redirect', baseUri), {
|
|
|
|
redirect: 'manual',
|
|
|
|
signal: controller.signal,
|
|
|
|
});
|
|
|
|
assert.equal(response.status, 303);
|
|
|
|
assert.equal(response.headers.get('location'), '/destination');
|
|
|
|
});
|
|
|
|
|
|
|
|
it('Can respond with Response.redirect', async () => {
|
|
|
|
const controller = new AbortController();
|
|
|
|
setTimeout(() => controller.abort(), 1000);
|
|
|
|
const response = await fetch(new URL('/response-redirect', baseUri), {
|
|
|
|
redirect: 'manual',
|
|
|
|
signal: controller.signal,
|
|
|
|
});
|
|
|
|
assert.equal(response.status, 307);
|
|
|
|
assert.equal(response.headers.get('location'), String(new URL('/destination', baseUri)));
|
|
|
|
});
|
2022-07-22 15:22:31 -04:00
|
|
|
});
|