Refactor 404 and 500 approach (#7754)
* fix(app): refactor 404 and 500 approach * chore: refactor logic * fix: always treat error as page * test: migrate ssr-prerender-404 to node adapter * feat: merge original response metadata with error response * chore: update lockfile * chore: trigger ci * chore(lint): fix lint issue * fix: ensure merged request has proper status * fix(node): prerender test * chore: update test label * fix(node): improve 404 behavior in middleware mode * fix(vercel): improve 404 behavior * fix(netlify): improve 404 behavior * chore: update test labels * chore: force ci * chore: fix lint * fix: avoid infinite loops * test: fix failing test in Node 18 * chore: remove volta
This commit is contained in:
parent
0b8375fe82
commit
298dbb89f2
22 changed files with 384 additions and 196 deletions
5
.changeset/big-suns-wave.md
Normal file
5
.changeset/big-suns-wave.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
'astro': patch
|
||||||
|
---
|
||||||
|
|
||||||
|
Refactor `404` and `500` route handling for consistency and improved prerendering support
|
5
.changeset/heavy-cooks-laugh.md
Normal file
5
.changeset/heavy-cooks-laugh.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
'@astrojs/node': patch
|
||||||
|
---
|
||||||
|
|
||||||
|
Improve `404` behavior in middleware mode
|
6
.changeset/rich-toys-jog.md
Normal file
6
.changeset/rich-toys-jog.md
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
---
|
||||||
|
'@astrojs/netlify': patch
|
||||||
|
'@astrojs/vercel': patch
|
||||||
|
---
|
||||||
|
|
||||||
|
Improve `404` behavior for `serverless` and `edge`
|
|
@ -34,9 +34,16 @@ const clientLocalsSymbol = Symbol.for('astro.locals');
|
||||||
|
|
||||||
const responseSentSymbol = Symbol.for('astro.responseSent');
|
const responseSentSymbol = Symbol.for('astro.responseSent');
|
||||||
|
|
||||||
|
const STATUS_CODES = new Set([404, 500]);
|
||||||
|
|
||||||
export interface MatchOptions {
|
export interface MatchOptions {
|
||||||
matchNotFound?: boolean | undefined;
|
matchNotFound?: boolean | undefined;
|
||||||
}
|
}
|
||||||
|
export interface RenderErrorOptions {
|
||||||
|
routeData?: RouteData;
|
||||||
|
response?: Response;
|
||||||
|
status: 404 | 500;
|
||||||
|
}
|
||||||
|
|
||||||
export class App {
|
export class App {
|
||||||
/**
|
/**
|
||||||
|
@ -113,50 +120,29 @@ export class App {
|
||||||
}
|
}
|
||||||
return pathname;
|
return pathname;
|
||||||
}
|
}
|
||||||
match(request: Request, { matchNotFound = false }: MatchOptions = {}): RouteData | undefined {
|
// Disable no-unused-vars to avoid breaking signature change
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
||||||
|
match(request: Request, _opts: MatchOptions = {}): RouteData | undefined {
|
||||||
const url = new URL(request.url);
|
const url = new URL(request.url);
|
||||||
// ignore requests matching public assets
|
// ignore requests matching public assets
|
||||||
if (this.#manifest.assets.has(url.pathname)) {
|
if (this.#manifest.assets.has(url.pathname)) return undefined;
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
let pathname = prependForwardSlash(this.removeBase(url.pathname));
|
let pathname = prependForwardSlash(this.removeBase(url.pathname));
|
||||||
let routeData = matchRoute(pathname, this.#manifestData);
|
let routeData = matchRoute(pathname, this.#manifestData);
|
||||||
|
// missing routes fall-through, prerendered are handled by static layer
|
||||||
if (routeData) {
|
if (!routeData || routeData.prerender) return undefined;
|
||||||
if (routeData.prerender) return undefined;
|
return routeData;
|
||||||
return routeData;
|
|
||||||
} else if (matchNotFound) {
|
|
||||||
const notFoundRouteData = matchRoute('/404', this.#manifestData);
|
|
||||||
if (notFoundRouteData?.prerender) return undefined;
|
|
||||||
return notFoundRouteData;
|
|
||||||
} else {
|
|
||||||
return undefined;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
async render(request: Request, routeData?: RouteData, locals?: object): Promise<Response> {
|
async render(request: Request, routeData?: RouteData, locals?: object): Promise<Response> {
|
||||||
let defaultStatus = 200;
|
|
||||||
if (!routeData) {
|
if (!routeData) {
|
||||||
routeData = this.match(request);
|
routeData = this.match(request);
|
||||||
if (!routeData) {
|
}
|
||||||
defaultStatus = 404;
|
if (!routeData) {
|
||||||
routeData = this.match(request, { matchNotFound: true });
|
return this.#renderError(request, { routeData, status: 404 });
|
||||||
}
|
|
||||||
if (!routeData) {
|
|
||||||
return new Response(null, {
|
|
||||||
status: 404,
|
|
||||||
statusText: 'Not found',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Reflect.set(request, clientLocalsSymbol, locals ?? {});
|
Reflect.set(request, clientLocalsSymbol, locals ?? {});
|
||||||
|
const defaultStatus = this.#getDefaultStatusCode(routeData.route);
|
||||||
// Use the 404 status code for 404.astro components
|
const mod = await this.#getModuleForRoute(routeData);
|
||||||
if (routeData.route === '/404') {
|
|
||||||
defaultStatus = 404;
|
|
||||||
}
|
|
||||||
|
|
||||||
let mod = await this.#getModuleForRoute(routeData);
|
|
||||||
|
|
||||||
const pageModule = (await mod.page()) as any;
|
const pageModule = (await mod.page()) as any;
|
||||||
const url = new URL(request.url);
|
const url = new URL(request.url);
|
||||||
|
@ -179,47 +165,19 @@ export class App {
|
||||||
);
|
);
|
||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
error(this.#logging, 'ssr', err.stack || err.message || String(err));
|
error(this.#logging, 'ssr', err.stack || err.message || String(err));
|
||||||
response = new Response(null, {
|
return this.#renderError(request, { routeData, status: 500 });
|
||||||
status: 500,
|
|
||||||
statusText: 'Internal server error',
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isResponse(response, routeData.type)) {
|
if (isResponse(response, routeData.type)) {
|
||||||
// If there was a known error code, try sending the according page (e.g. 404.astro / 500.astro).
|
if (STATUS_CODES.has(response.status)) {
|
||||||
if (response.status === 500 || response.status === 404) {
|
return this.#renderError(request, { routeData, response, status: response.status as 404 | 500 } );
|
||||||
const errorRouteData = matchRoute('/' + response.status, this.#manifestData);
|
|
||||||
if (errorRouteData && errorRouteData.route !== routeData.route) {
|
|
||||||
mod = await this.#getModuleForRoute(errorRouteData);
|
|
||||||
try {
|
|
||||||
const newRenderContext = await this.#createRenderContext(
|
|
||||||
url,
|
|
||||||
request,
|
|
||||||
routeData,
|
|
||||||
mod,
|
|
||||||
response.status
|
|
||||||
);
|
|
||||||
const page = (await mod.page()) as any;
|
|
||||||
const errorResponse = await tryRenderRoute(
|
|
||||||
routeData.type,
|
|
||||||
newRenderContext,
|
|
||||||
this.#env,
|
|
||||||
page
|
|
||||||
);
|
|
||||||
return errorResponse as Response;
|
|
||||||
} catch {}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
Reflect.set(response, responseSentSymbol, true);
|
Reflect.set(response, responseSentSymbol, true);
|
||||||
return response;
|
return response;
|
||||||
} else {
|
} else {
|
||||||
if (response.type === 'response') {
|
if (response.type === 'response') {
|
||||||
if (response.response.headers.get('X-Astro-Response') === 'Not-Found') {
|
if (response.response.headers.get('X-Astro-Response') === 'Not-Found') {
|
||||||
const fourOhFourRequest = new Request(new URL('/404', request.url));
|
return this.#renderError(request, { routeData, response: response.response, status: 404 });
|
||||||
const fourOhFourRouteData = this.match(fourOhFourRequest);
|
|
||||||
if (fourOhFourRouteData) {
|
|
||||||
return this.render(fourOhFourRequest, fourOhFourRouteData);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return response.response;
|
return response.response;
|
||||||
} else {
|
} else {
|
||||||
|
@ -238,7 +196,6 @@ export class App {
|
||||||
status: 200,
|
status: 200,
|
||||||
headers,
|
headers,
|
||||||
});
|
});
|
||||||
|
|
||||||
attachToResponse(newResponse, response.cookies);
|
attachToResponse(newResponse, response.cookies);
|
||||||
return newResponse;
|
return newResponse;
|
||||||
}
|
}
|
||||||
|
@ -307,6 +264,63 @@ export class App {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If is a known error code, try sending the according page (e.g. 404.astro / 500.astro).
|
||||||
|
* This also handles pre-rendered /404 or /500 routes
|
||||||
|
*/
|
||||||
|
async #renderError(request: Request, { routeData, status, response: originalResponse }: RenderErrorOptions) {
|
||||||
|
const errorRouteData = matchRoute('/' + status, this.#manifestData);
|
||||||
|
const url = new URL(request.url);
|
||||||
|
if (errorRouteData) {
|
||||||
|
if (errorRouteData.prerender && !errorRouteData.route.endsWith(`/${status}`)) {
|
||||||
|
const statusURL = new URL(`${this.#baseWithoutTrailingSlash}/${status}`, url);
|
||||||
|
const response = await fetch(statusURL.toString());
|
||||||
|
return this.#mergeResponses(response, originalResponse);
|
||||||
|
}
|
||||||
|
const finalRouteData = routeData ?? errorRouteData;
|
||||||
|
const mod = await this.#getModuleForRoute(errorRouteData);
|
||||||
|
try {
|
||||||
|
const newRenderContext = await this.#createRenderContext(
|
||||||
|
url,
|
||||||
|
request,
|
||||||
|
finalRouteData,
|
||||||
|
mod,
|
||||||
|
status
|
||||||
|
);
|
||||||
|
const page = (await mod.page()) as any;
|
||||||
|
const response = await tryRenderRoute(
|
||||||
|
'page', // this is hardcoded to ensure proper behavior for missing endpoints
|
||||||
|
newRenderContext,
|
||||||
|
this.#env,
|
||||||
|
page
|
||||||
|
) as Response;
|
||||||
|
return this.#mergeResponses(response, originalResponse);
|
||||||
|
} catch {}
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = this.#mergeResponses(new Response(null, { status }), originalResponse);
|
||||||
|
Reflect.set(response, responseSentSymbol, true);
|
||||||
|
return response;
|
||||||
|
}
|
||||||
|
|
||||||
|
#mergeResponses(newResponse: Response, oldResponse?: Response) {
|
||||||
|
if (!oldResponse) return newResponse;
|
||||||
|
const { status, statusText, headers } = oldResponse;
|
||||||
|
|
||||||
|
return new Response(newResponse.body, {
|
||||||
|
status: status === 200 ? newResponse.status : status,
|
||||||
|
statusText,
|
||||||
|
headers: new Headers(Array.from(headers))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
#getDefaultStatusCode(route: string): number {
|
||||||
|
route = removeTrailingForwardSlash(route)
|
||||||
|
if (route.endsWith('/404')) return 404;
|
||||||
|
if (route.endsWith('/500')) return 500;
|
||||||
|
return 200;
|
||||||
|
}
|
||||||
|
|
||||||
async #getModuleForRoute(route: RouteData): Promise<SinglePageBuiltModule> {
|
async #getModuleForRoute(route: RouteData): Promise<SinglePageBuiltModule> {
|
||||||
if (route.type === 'redirect') {
|
if (route.type === 'redirect') {
|
||||||
return RedirectSinglePageBuiltModule;
|
return RedirectSinglePageBuiltModule;
|
||||||
|
|
|
@ -1,8 +0,0 @@
|
||||||
{
|
|
||||||
"name": "@test/ssr-prerender-404",
|
|
||||||
"version": "0.0.0",
|
|
||||||
"private": true,
|
|
||||||
"dependencies": {
|
|
||||||
"astro": "workspace:*"
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,5 +0,0 @@
|
||||||
---
|
|
||||||
export const prerender = true
|
|
||||||
---
|
|
||||||
|
|
||||||
Page does not exist
|
|
|
@ -1,18 +0,0 @@
|
||||||
---
|
|
||||||
export const prerender = true;
|
|
||||||
|
|
||||||
const { searchParams } = Astro.url;
|
|
||||||
---
|
|
||||||
|
|
||||||
<html>
|
|
||||||
<head>
|
|
||||||
<title>Static Page</title>
|
|
||||||
<script>
|
|
||||||
console.log('hello world');
|
|
||||||
</script>
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<h1 id="greeting">Hello world!</h1>
|
|
||||||
<div id="searchparams">{searchParams.get('q')}</div>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
|
@ -1,6 +1,7 @@
|
||||||
---
|
---
|
||||||
Astro.response.status = 404;
|
Astro.response.status = 404;
|
||||||
Astro.response.statusText = 'Oops';
|
Astro.response.statusText = 'Oops';
|
||||||
|
Astro.response.headers.set('One-Two', 'three');
|
||||||
---
|
---
|
||||||
<html>
|
<html>
|
||||||
<head>
|
<head>
|
||||||
|
|
|
@ -1,30 +0,0 @@
|
||||||
import { expect } from 'chai';
|
|
||||||
import { loadFixture } from './test-utils.js';
|
|
||||||
import testAdapter from './test-adapter.js';
|
|
||||||
|
|
||||||
describe('SSR: prerender 404', () => {
|
|
||||||
/** @type {import('./test-utils').Fixture} */
|
|
||||||
let fixture;
|
|
||||||
|
|
||||||
before(async () => {
|
|
||||||
fixture = await loadFixture({
|
|
||||||
root: './fixtures/ssr-prerender-404/',
|
|
||||||
output: 'server',
|
|
||||||
adapter: testAdapter(),
|
|
||||||
});
|
|
||||||
await fixture.build();
|
|
||||||
});
|
|
||||||
|
|
||||||
describe('Prerendering', () => {
|
|
||||||
it('Prerendered 404.astro page is not rendered', async () => {
|
|
||||||
const app = await fixture.loadTestAdapterApp();
|
|
||||||
const request = new Request('http://example.com/non-existent-page');
|
|
||||||
const response = await app.render(request);
|
|
||||||
expect(response.status).to.equal(404);
|
|
||||||
expect(response.statusText).to.equal(
|
|
||||||
'Not found',
|
|
||||||
'should be actual 404 response, not 404 page'
|
|
||||||
);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
|
@ -29,6 +29,14 @@ describe('Using Astro.response in SSR', () => {
|
||||||
expect(response.statusText).to.equal('Oops');
|
expect(response.statusText).to.equal('Oops');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('Can set headers for 404 page', async () => {
|
||||||
|
const app = await fixture.loadTestAdapterApp();
|
||||||
|
const request = new Request('http://example.com/status-code');
|
||||||
|
const response = await app.render(request);
|
||||||
|
const headers = response.headers;
|
||||||
|
expect(headers.get('one-two')).to.equal('three');
|
||||||
|
});
|
||||||
|
|
||||||
it('Can add headers', async () => {
|
it('Can add headers', async () => {
|
||||||
const app = await fixture.loadTestAdapterApp();
|
const app = await fixture.loadTestAdapterApp();
|
||||||
const request = new Request('http://example.com/some-header');
|
const request = new Request('http://example.com/some-header');
|
||||||
|
|
|
@ -15,25 +15,19 @@ export function createExports(manifest: SSRManifest) {
|
||||||
if (manifest.assets.has(url.pathname)) {
|
if (manifest.assets.has(url.pathname)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (app.match(request)) {
|
const routeData = app.match(request)
|
||||||
const ip =
|
const ip =
|
||||||
request.headers.get('x-nf-client-connection-ip') ||
|
request.headers.get('x-nf-client-connection-ip') ||
|
||||||
context?.ip ||
|
context?.ip ||
|
||||||
(context as any)?.remoteAddr?.hostname;
|
(context as any)?.remoteAddr?.hostname;
|
||||||
Reflect.set(request, clientAddressSymbol, ip);
|
Reflect.set(request, clientAddressSymbol, ip);
|
||||||
const response = await app.render(request);
|
const response = await app.render(request, routeData);
|
||||||
if (app.setCookieHeaders) {
|
if (app.setCookieHeaders) {
|
||||||
for (const setCookieHeader of app.setCookieHeaders(response)) {
|
for (const setCookieHeader of app.setCookieHeaders(response)) {
|
||||||
response.headers.append('Set-Cookie', setCookieHeader);
|
response.headers.append('Set-Cookie', setCookieHeader);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return response;
|
|
||||||
}
|
}
|
||||||
|
return response;
|
||||||
return new Response(null, {
|
|
||||||
status: 404,
|
|
||||||
statusText: 'Not found',
|
|
||||||
});
|
|
||||||
};
|
};
|
||||||
|
|
||||||
return { default: handler };
|
return { default: handler };
|
||||||
|
|
|
@ -70,15 +70,7 @@ export const createExports = (manifest: SSRManifest, args: Args) => {
|
||||||
}
|
}
|
||||||
const request = new Request(rawUrl, init);
|
const request = new Request(rawUrl, init);
|
||||||
|
|
||||||
let routeData = app.match(request, { matchNotFound: true });
|
const routeData = app.match(request);
|
||||||
|
|
||||||
if (!routeData) {
|
|
||||||
return {
|
|
||||||
statusCode: 404,
|
|
||||||
body: 'Not found',
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
const ip = headers['x-nf-client-connection-ip'];
|
const ip = headers['x-nf-client-connection-ip'];
|
||||||
Reflect.set(request, clientAddressSymbol, ip);
|
Reflect.set(request, clientAddressSymbol, ip);
|
||||||
let locals = {};
|
let locals = {};
|
||||||
|
|
|
@ -5,7 +5,9 @@ import { createOutgoingHttpHeaders } from './createOutgoingHttpHeaders';
|
||||||
import { responseIterator } from './response-iterator';
|
import { responseIterator } from './response-iterator';
|
||||||
import type { Options } from './types';
|
import type { Options } from './types';
|
||||||
|
|
||||||
export default function (app: NodeApp, mode: Options['mode']) {
|
// Disable no-unused-vars to avoid breaking signature change
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
||||||
|
export default function (app: NodeApp, _mode: Options['mode']) {
|
||||||
return async function (
|
return async function (
|
||||||
req: IncomingMessage,
|
req: IncomingMessage,
|
||||||
res: ServerResponse,
|
res: ServerResponse,
|
||||||
|
@ -13,8 +15,7 @@ export default function (app: NodeApp, mode: Options['mode']) {
|
||||||
locals?: object
|
locals?: object
|
||||||
) {
|
) {
|
||||||
try {
|
try {
|
||||||
const route =
|
const route = app.match(req);
|
||||||
mode === 'standalone' ? app.match(req, { matchNotFound: true }) : app.match(req);
|
|
||||||
if (route) {
|
if (route) {
|
||||||
try {
|
try {
|
||||||
const response = await app.render(req, route, locals);
|
const response = await app.render(req, route, locals);
|
||||||
|
@ -29,8 +30,8 @@ export default function (app: NodeApp, mode: Options['mode']) {
|
||||||
} else if (next) {
|
} else if (next) {
|
||||||
return next();
|
return next();
|
||||||
} else {
|
} else {
|
||||||
res.writeHead(404);
|
const response = await app.render(req);
|
||||||
res.end('Not found');
|
await writeWebResponse(app, res, response);
|
||||||
}
|
}
|
||||||
} catch (err: unknown) {
|
} catch (err: unknown) {
|
||||||
if (!res.headersSent) {
|
if (!res.headersSent) {
|
||||||
|
|
|
@ -9,5 +9,5 @@
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<title>404</title>
|
<title>404</title>
|
||||||
</head>
|
</head>
|
||||||
<body><h1>404!!!!!!!!!!</h1></body>
|
<body>Page does not exist</body>
|
||||||
</html>
|
</html>
|
||||||
|
|
9
packages/integrations/node/test/fixtures/prerender-404/package.json
vendored
Normal file
9
packages/integrations/node/test/fixtures/prerender-404/package.json
vendored
Normal file
|
@ -0,0 +1,9 @@
|
||||||
|
{
|
||||||
|
"name": "@test/nodejs-prerender-404",
|
||||||
|
"version": "0.0.0",
|
||||||
|
"private": true,
|
||||||
|
"dependencies": {
|
||||||
|
"astro": "workspace:*",
|
||||||
|
"@astrojs/node": "workspace:*"
|
||||||
|
}
|
||||||
|
}
|
5
packages/integrations/node/test/fixtures/prerender-404/src/pages/404.astro
vendored
Normal file
5
packages/integrations/node/test/fixtures/prerender-404/src/pages/404.astro
vendored
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
export const prerender = true;
|
||||||
|
---
|
||||||
|
|
||||||
|
Page does not exist
|
12
packages/integrations/node/test/fixtures/prerender-404/src/pages/static.astro
vendored
Normal file
12
packages/integrations/node/test/fixtures/prerender-404/src/pages/static.astro
vendored
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
---
|
||||||
|
export const prerender = true;
|
||||||
|
---
|
||||||
|
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<title>Static Page</title>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>Hello world!</h1>
|
||||||
|
</body>
|
||||||
|
</html>
|
|
@ -3,35 +3,51 @@ import { loadFixture } from './test-utils.js';
|
||||||
import { expect } from 'chai';
|
import { expect } from 'chai';
|
||||||
import * as cheerio from 'cheerio';
|
import * as cheerio from 'cheerio';
|
||||||
|
|
||||||
describe('test 404 cant load', () => {
|
/**
|
||||||
|
* @typedef {import('../../../astro/test/test-utils').Fixture} Fixture
|
||||||
|
*/
|
||||||
|
|
||||||
|
async function load() {
|
||||||
|
const mod = await import(`./fixtures/node-middleware/dist/server/entry.mjs?dropcache=${Date.now()}`);
|
||||||
|
return mod;
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('behavior from middleware', () => {
|
||||||
|
/** @type {import('./test-utils').Fixture} */
|
||||||
let fixture;
|
let fixture;
|
||||||
|
let server;
|
||||||
|
|
||||||
before(async () => {
|
before(async () => {
|
||||||
|
process.env.ASTRO_NODE_AUTOSTART = 'disabled';
|
||||||
|
process.env.PRERENDER = false;
|
||||||
fixture = await loadFixture({
|
fixture = await loadFixture({
|
||||||
root: './fixtures/node-middleware/',
|
root: './fixtures/node-middleware/',
|
||||||
output: 'server',
|
output: 'server',
|
||||||
adapter: nodejs({ mode: 'standalone' }),
|
adapter: nodejs({ mode: 'standalone' }),
|
||||||
});
|
});
|
||||||
await fixture.build();
|
await fixture.build();
|
||||||
|
const { startServer } = await load();
|
||||||
|
let res = startServer();
|
||||||
|
server = res.server;
|
||||||
});
|
});
|
||||||
describe('test 404', async () => {
|
|
||||||
let devPreview;
|
|
||||||
|
|
||||||
before(async () => {
|
after(async () => {
|
||||||
devPreview = await fixture.preview();
|
await server.stop();
|
||||||
});
|
await fixture.clean();
|
||||||
after(async () => {
|
delete process.env.PRERENDER;
|
||||||
await devPreview.stop();
|
})
|
||||||
});
|
|
||||||
|
describe('404', async () => {
|
||||||
it('when mode is standalone', async () => {
|
it('when mode is standalone', async () => {
|
||||||
const res = await fixture.fetch('/error-page');
|
const res = await fetch(`http://${server.host}:${server.port}/error-page`);
|
||||||
|
|
||||||
expect(res.status).to.equal(404);
|
expect(res.status).to.equal(404);
|
||||||
|
|
||||||
const html = await res.text();
|
const html = await res.text();
|
||||||
const $ = cheerio.load(html);
|
const $ = cheerio.load(html);
|
||||||
|
|
||||||
const h1 = $('h1');
|
const body = $('body');
|
||||||
expect(h1.text()).to.equal('404!!!!!!!!!!');
|
expect(body.text()).to.equal('Page does not exist');
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
189
packages/integrations/node/test/prerender-404.test.js
Normal file
189
packages/integrations/node/test/prerender-404.test.js
Normal file
|
@ -0,0 +1,189 @@
|
||||||
|
import nodejs from '../dist/index.js';
|
||||||
|
import { loadFixture } from './test-utils.js';
|
||||||
|
import { expect } from 'chai';
|
||||||
|
import * as cheerio from 'cheerio';
|
||||||
|
import { fetch } from 'undici';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @typedef {import('../../../astro/test/test-utils').Fixture} Fixture
|
||||||
|
*/
|
||||||
|
|
||||||
|
async function load() {
|
||||||
|
const mod = await import(`./fixtures/prerender-404/dist/server/entry.mjs?dropcache=${Date.now()}`);
|
||||||
|
return mod;
|
||||||
|
}
|
||||||
|
describe('Prerender 404', () => {
|
||||||
|
/** @type {import('./test-utils').Fixture} */
|
||||||
|
let fixture;
|
||||||
|
let server;
|
||||||
|
|
||||||
|
describe('With base', async () => {
|
||||||
|
before(async () => {
|
||||||
|
process.env.ASTRO_NODE_AUTOSTART = 'disabled';
|
||||||
|
process.env.PRERENDER = true;
|
||||||
|
|
||||||
|
fixture = await loadFixture({
|
||||||
|
base: '/some-base',
|
||||||
|
root: './fixtures/prerender-404/',
|
||||||
|
output: 'server',
|
||||||
|
adapter: nodejs({ mode: 'standalone' }),
|
||||||
|
});
|
||||||
|
await fixture.build();
|
||||||
|
const { startServer } = await load();
|
||||||
|
let res = startServer();
|
||||||
|
server = res.server;
|
||||||
|
});
|
||||||
|
|
||||||
|
after(async () => {
|
||||||
|
await server.stop();
|
||||||
|
await fixture.clean();
|
||||||
|
delete process.env.PRERENDER;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Can render SSR route', async () => {
|
||||||
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/static`);
|
||||||
|
const html = await res.text();
|
||||||
|
const $ = cheerio.load(html);
|
||||||
|
|
||||||
|
expect(res.status).to.equal(200);
|
||||||
|
expect($('h1').text()).to.equal('Hello world!');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Can handle prerendered 404', async () => {
|
||||||
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/missing`);
|
||||||
|
const html = await res.text();
|
||||||
|
const $ = cheerio.load(html);
|
||||||
|
|
||||||
|
expect(res.status).to.equal(404);
|
||||||
|
expect($('body').text()).to.equal('Page does not exist');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('Without base', async () => {
|
||||||
|
before(async () => {
|
||||||
|
process.env.ASTRO_NODE_AUTOSTART = 'disabled';
|
||||||
|
process.env.PRERENDER = true;
|
||||||
|
|
||||||
|
fixture = await loadFixture({
|
||||||
|
root: './fixtures/prerender-404/',
|
||||||
|
output: 'server',
|
||||||
|
adapter: nodejs({ mode: 'standalone' }),
|
||||||
|
});
|
||||||
|
await fixture.build();
|
||||||
|
const { startServer } = await await load();
|
||||||
|
let res = startServer();
|
||||||
|
server = res.server;
|
||||||
|
});
|
||||||
|
|
||||||
|
after(async () => {
|
||||||
|
await server.stop();
|
||||||
|
await fixture.clean();
|
||||||
|
delete process.env.PRERENDER;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Can render SSR route', async () => {
|
||||||
|
const res = await fetch(`http://${server.host}:${server.port}/static`);
|
||||||
|
const html = await res.text();
|
||||||
|
const $ = cheerio.load(html);
|
||||||
|
|
||||||
|
expect(res.status).to.equal(200);
|
||||||
|
expect($('h1').text()).to.equal('Hello world!');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Can handle prerendered 404', async () => {
|
||||||
|
const res = await fetch(`http://${server.host}:${server.port}/missing`);
|
||||||
|
const html = await res.text();
|
||||||
|
const $ = cheerio.load(html);
|
||||||
|
|
||||||
|
expect(res.status).to.equal(404);
|
||||||
|
expect($('body').text()).to.equal('Page does not exist');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('Hybrid 404', () => {
|
||||||
|
/** @type {import('./test-utils').Fixture} */
|
||||||
|
let fixture;
|
||||||
|
let server;
|
||||||
|
|
||||||
|
describe('With base', async () => {
|
||||||
|
before(async () => {
|
||||||
|
process.env.ASTRO_NODE_AUTOSTART = 'disabled';
|
||||||
|
process.env.PRERENDER = false;
|
||||||
|
fixture = await loadFixture({
|
||||||
|
base: '/some-base',
|
||||||
|
root: './fixtures/prerender-404/',
|
||||||
|
output: 'hybrid',
|
||||||
|
adapter: nodejs({ mode: 'standalone' }),
|
||||||
|
});
|
||||||
|
await fixture.build();
|
||||||
|
const { startServer } = await await load();
|
||||||
|
let res = startServer();
|
||||||
|
server = res.server;
|
||||||
|
});
|
||||||
|
|
||||||
|
after(async () => {
|
||||||
|
await server.stop();
|
||||||
|
await fixture.clean();
|
||||||
|
delete process.env.PRERENDER;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Can render SSR route', async () => {
|
||||||
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/static`);
|
||||||
|
const html = await res.text();
|
||||||
|
const $ = cheerio.load(html);
|
||||||
|
|
||||||
|
expect(res.status).to.equal(200);
|
||||||
|
expect($('h1').text()).to.equal('Hello world!');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Can handle prerendered 404', async () => {
|
||||||
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/missing`);
|
||||||
|
const html = await res.text();
|
||||||
|
const $ = cheerio.load(html);
|
||||||
|
|
||||||
|
expect(res.status).to.equal(404);
|
||||||
|
expect($('body').text()).to.equal('Page does not exist');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('Without base', async () => {
|
||||||
|
before(async () => {
|
||||||
|
process.env.ASTRO_NODE_AUTOSTART = 'disabled';
|
||||||
|
process.env.PRERENDER = false;
|
||||||
|
fixture = await loadFixture({
|
||||||
|
root: './fixtures/prerender-404/',
|
||||||
|
output: 'hybrid',
|
||||||
|
adapter: nodejs({ mode: 'standalone' }),
|
||||||
|
});
|
||||||
|
await fixture.build();
|
||||||
|
const { startServer } = await await load();
|
||||||
|
let res = startServer();
|
||||||
|
server = res.server;
|
||||||
|
});
|
||||||
|
|
||||||
|
after(async () => {
|
||||||
|
await server.stop();
|
||||||
|
await fixture.clean();
|
||||||
|
delete process.env.PRERENDER;
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Can render SSR route', async () => {
|
||||||
|
const res = await fetch(`http://${server.host}:${server.port}/static`);
|
||||||
|
const html = await res.text();
|
||||||
|
const $ = cheerio.load(html);
|
||||||
|
|
||||||
|
expect(res.status).to.equal(200);
|
||||||
|
expect($('h1').text()).to.equal('Hello world!');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('Can handle prerendered 404', async () => {
|
||||||
|
const res = await fetch(`http://${server.host}:${server.port}/missing`);
|
||||||
|
const html = await res.text();
|
||||||
|
const $ = cheerio.load(html);
|
||||||
|
|
||||||
|
expect(res.status).to.equal(404);
|
||||||
|
expect($('body').text()).to.equal('Page does not exist');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -13,21 +13,15 @@ export function createExports(manifest: SSRManifest) {
|
||||||
const app = new App(manifest);
|
const app = new App(manifest);
|
||||||
|
|
||||||
const handler = async (request: Request): Promise<Response> => {
|
const handler = async (request: Request): Promise<Response> => {
|
||||||
if (app.match(request)) {
|
const routeData = app.match(request);
|
||||||
Reflect.set(request, clientAddressSymbol, request.headers.get('x-forwarded-for'));
|
Reflect.set(request, clientAddressSymbol, request.headers.get('x-forwarded-for'));
|
||||||
const response = await app.render(request);
|
const response = await app.render(request, routeData);
|
||||||
if (app.setCookieHeaders) {
|
if (app.setCookieHeaders) {
|
||||||
for (const setCookieHeader of app.setCookieHeaders(response)) {
|
for (const setCookieHeader of app.setCookieHeaders(response)) {
|
||||||
response.headers.append('Set-Cookie', setCookieHeader);
|
response.headers.append('Set-Cookie', setCookieHeader);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return response;
|
|
||||||
}
|
}
|
||||||
|
return response;
|
||||||
return new Response(null, {
|
|
||||||
status: 404,
|
|
||||||
statusText: 'Not found',
|
|
||||||
});
|
|
||||||
};
|
};
|
||||||
|
|
||||||
return { default: handler };
|
return { default: handler };
|
||||||
|
|
|
@ -23,12 +23,7 @@ export const createExports = (manifest: SSRManifest) => {
|
||||||
return res.end(err.reason || 'Invalid request body');
|
return res.end(err.reason || 'Invalid request body');
|
||||||
}
|
}
|
||||||
|
|
||||||
let routeData = app.match(request, { matchNotFound: true });
|
let routeData = app.match(request);
|
||||||
if (!routeData) {
|
|
||||||
res.statusCode = 404;
|
|
||||||
return res.end('Not found');
|
|
||||||
}
|
|
||||||
|
|
||||||
let locals = {};
|
let locals = {};
|
||||||
if (request.headers.has(ASTRO_LOCALS_HEADER)) {
|
if (request.headers.has(ASTRO_LOCALS_HEADER)) {
|
||||||
let localsAsString = request.headers.get(ASTRO_LOCALS_HEADER);
|
let localsAsString = request.headers.get(ASTRO_LOCALS_HEADER);
|
||||||
|
|
|
@ -3316,12 +3316,6 @@ importers:
|
||||||
specifier: workspace:*
|
specifier: workspace:*
|
||||||
version: link:../../..
|
version: link:../../..
|
||||||
|
|
||||||
packages/astro/test/fixtures/ssr-prerender-404:
|
|
||||||
dependencies:
|
|
||||||
astro:
|
|
||||||
specifier: workspace:*
|
|
||||||
version: link:../../..
|
|
||||||
|
|
||||||
packages/astro/test/fixtures/ssr-prerender-get-static-paths:
|
packages/astro/test/fixtures/ssr-prerender-get-static-paths:
|
||||||
dependencies:
|
dependencies:
|
||||||
astro:
|
astro:
|
||||||
|
@ -4656,6 +4650,15 @@ importers:
|
||||||
specifier: workspace:*
|
specifier: workspace:*
|
||||||
version: link:../../../../../astro
|
version: link:../../../../../astro
|
||||||
|
|
||||||
|
packages/integrations/node/test/fixtures/prerender-404:
|
||||||
|
dependencies:
|
||||||
|
'@astrojs/node':
|
||||||
|
specifier: workspace:*
|
||||||
|
version: link:../../..
|
||||||
|
astro:
|
||||||
|
specifier: workspace:*
|
||||||
|
version: link:../../../../../astro
|
||||||
|
|
||||||
packages/integrations/node/test/fixtures/url-protocol:
|
packages/integrations/node/test/fixtures/url-protocol:
|
||||||
dependencies:
|
dependencies:
|
||||||
'@astrojs/node':
|
'@astrojs/node':
|
||||||
|
|
Loading…
Reference in a new issue