api-route.test.js 4.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. import nodejs from '../dist/index.js';
  2. import { loadFixture, createRequestAndResponse } from './test-utils.js';
  3. import crypto from 'node:crypto';
  4. import { describe, it, before, after } from 'node:test';
  5. import * as assert from 'node:assert/strict';
  6. describe('API routes', () => {
  7. /** @type {import('./test-utils').Fixture} */
  8. let fixture;
  9. /** @type {import('astro/src/@types/astro.js').PreviewServer} */
  10. let previewServer;
  11. /** @type {URL} */
  12. let baseUri;
  13. before(async () => {
  14. fixture = await loadFixture({
  15. root: './fixtures/api-route/',
  16. output: 'server',
  17. adapter: nodejs({ mode: 'middleware' }),
  18. });
  19. await fixture.build();
  20. previewServer = await fixture.preview();
  21. baseUri = new URL(`http://${previewServer.host ?? 'localhost'}:${previewServer.port}/`);
  22. });
  23. after(() => previewServer.stop());
  24. it('Can get the request body', async () => {
  25. const { handler } = await import('./fixtures/api-route/dist/server/entry.mjs');
  26. let { req, res, done } = createRequestAndResponse({
  27. method: 'POST',
  28. url: '/recipes',
  29. });
  30. req.once('async_iterator', () => {
  31. req.send(JSON.stringify({ id: 2 }));
  32. });
  33. handler(req, res);
  34. let [buffer] = await done;
  35. let json = JSON.parse(buffer.toString('utf-8'));
  36. assert.equal(json.length, 1);
  37. assert.equal(json[0].name, 'Broccoli Soup');
  38. });
  39. it('Can get binary data', async () => {
  40. const { handler } = await import('./fixtures/api-route/dist/server/entry.mjs');
  41. let { req, res, done } = createRequestAndResponse({
  42. method: 'POST',
  43. url: '/binary',
  44. });
  45. req.once('async_iterator', () => {
  46. req.send(Buffer.from(new Uint8Array([1, 2, 3, 4, 5])));
  47. });
  48. handler(req, res);
  49. let [out] = await done;
  50. let arr = Array.from(new Uint8Array(out.buffer));
  51. assert.deepEqual(arr, [5, 4, 3, 2, 1]);
  52. });
  53. it('Can post large binary data', async () => {
  54. const { handler } = await import('./fixtures/api-route/dist/server/entry.mjs');
  55. let { req, res, done } = createRequestAndResponse({
  56. method: 'POST',
  57. url: '/hash',
  58. });
  59. handler(req, res);
  60. let expectedDigest = null;
  61. req.once('async_iterator', () => {
  62. // Send 256MB of garbage data in 256KB chunks. This should be fast (< 1sec).
  63. let remainingBytes = 256 * 1024 * 1024;
  64. const chunkSize = 256 * 1024;
  65. const hash = crypto.createHash('sha256');
  66. while (remainingBytes > 0) {
  67. const size = Math.min(remainingBytes, chunkSize);
  68. const chunk = Buffer.alloc(size, Math.floor(Math.random() * 256));
  69. hash.update(chunk);
  70. req.emit('data', chunk);
  71. remainingBytes -= size;
  72. }
  73. req.emit('end');
  74. expectedDigest = hash.digest();
  75. });
  76. let [out] = await done;
  77. assert.deepEqual(new Uint8Array(out.buffer), new Uint8Array(expectedDigest));
  78. });
  79. it('Can bail on streaming', async () => {
  80. const { handler } = await import('./fixtures/api-route/dist/server/entry.mjs');
  81. let { req, res, done } = createRequestAndResponse({
  82. url: '/streaming',
  83. });
  84. let locals = { cancelledByTheServer: false };
  85. handler(req, res, () => {}, locals);
  86. req.send();
  87. await new Promise((resolve) => setTimeout(resolve, 500));
  88. res.emit('close');
  89. await done;
  90. assert.deepEqual(locals, { cancelledByTheServer: true });
  91. });
  92. it('Can respond with SSR redirect', async () => {
  93. const controller = new AbortController();
  94. setTimeout(() => controller.abort(), 1000);
  95. const response = await fetch(new URL('/redirect', baseUri), {
  96. redirect: 'manual',
  97. signal: controller.signal,
  98. });
  99. assert.equal(response.status, 302);
  100. assert.equal(response.headers.get('location'), '/destination');
  101. });
  102. it('Can respond with Astro.redirect', async () => {
  103. const controller = new AbortController();
  104. setTimeout(() => controller.abort(), 1000);
  105. const response = await fetch(new URL('/astro-redirect', baseUri), {
  106. redirect: 'manual',
  107. signal: controller.signal,
  108. });
  109. assert.equal(response.status, 303);
  110. assert.equal(response.headers.get('location'), '/destination');
  111. });
  112. it('Can respond with Response.redirect', async () => {
  113. const controller = new AbortController();
  114. setTimeout(() => controller.abort(), 1000);
  115. const response = await fetch(new URL('/response-redirect', baseUri), {
  116. redirect: 'manual',
  117. signal: controller.signal,
  118. });
  119. assert.equal(response.status, 307);
  120. assert.equal(response.headers.get('location'), String(new URL('/destination', baseUri)));
  121. });
  122. });