0
Fork 0
mirror of https://github.com/withastro/astro.git synced 2025-03-31 23:31:30 -05:00

fix(astro): don't run middlewarein dev for prerendered 404 (#11273)

This commit is contained in:
Matt Kane 2024-06-18 16:04:59 +01:00 committed by GitHub
parent 080156fc65
commit cb4d07819f
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
7 changed files with 97 additions and 107 deletions

View file

@ -0,0 +1,7 @@
---
'astro': patch
---
Corrects an inconsistency in dev where middleware would run for prerendered 404 routes.
Middleware is not run for prerendered 404 routes in production, so this was incorrect.

View file

@ -162,11 +162,10 @@ export async function handleRoute({
}: HandleRoute): Promise<void> {
const timeStart = performance.now();
const { config, loader, logger } = pipeline;
if (!matchedRoute && !config.i18n) {
if (isLoggedRequest(pathname)) {
logger.info(null, req({ url: pathname, method: incomingRequest.method, statusCode: 404 }));
}
return handle404Response(origin, incomingRequest, incomingResponse);
if (!matchedRoute) {
// This should never happen, because ensure404Route will add a 404 route if none exists.
throw new Error('No route matched, and default 404 route was not found.');
}
let request: Request;
@ -177,107 +176,48 @@ export async function handleRoute({
const middleware = (await loadMiddleware(loader)).onRequest;
const locals = Reflect.get(incomingRequest, clientLocalsSymbol);
if (!matchedRoute) {
if (config.i18n) {
const locales = config.i18n.locales;
const pathNameHasLocale = pathname
.split('/')
.filter(Boolean)
.some((segment) => {
let found = false;
for (const locale of locales) {
if (typeof locale === 'string') {
if (normalizeTheLocale(locale) === normalizeTheLocale(segment)) {
found = true;
break;
}
} else {
if (locale.path === segment) {
found = true;
break;
}
}
}
return found;
});
// Even when we have `config.base`, the pathname is still `/` because it gets stripped before
if (!pathNameHasLocale && pathname !== '/') {
return handle404Response(origin, incomingRequest, incomingResponse);
}
}
request = createRequest({
base: config.base,
url,
headers: incomingRequest.headers,
logger,
// no route found, so we assume the default for rendering the 404 page
staticLike: config.output === 'static' || config.output === 'hybrid',
});
route = {
component: '',
generate(_data: any): string {
return '';
},
params: [],
// Disable eslint as we only want to generate an empty RegExp
// eslint-disable-next-line prefer-regex-literals
pattern: new RegExp(''),
prerender: false,
segments: [],
type: 'fallback',
route: '',
fallbackRoutes: [],
isIndex: false,
};
const filePath: URL | undefined = matchedRoute.filePath;
const { preloadedComponent } = matchedRoute;
route = matchedRoute.route;
// Allows adapters to pass in locals in dev mode.
request = createRequest({
base: config.base,
url,
headers: incomingRequest.headers,
method: incomingRequest.method,
body,
logger,
clientAddress: incomingRequest.socket.remoteAddress,
staticLike: config.output === 'static' || route.prerender,
});
renderContext = RenderContext.create({
pipeline: pipeline,
pathname,
middleware,
request,
routeData: route,
});
} else {
const filePath: URL | undefined = matchedRoute.filePath;
const { preloadedComponent } = matchedRoute;
route = matchedRoute.route;
// Allows adapters to pass in locals in dev mode.
request = createRequest({
base: config.base,
url,
headers: incomingRequest.headers,
method: incomingRequest.method,
body,
logger,
clientAddress: incomingRequest.socket.remoteAddress,
staticLike: config.output === 'static' || route.prerender,
});
// Set user specified headers to response object.
for (const [name, value] of Object.entries(config.server.headers ?? {})) {
if (value) incomingResponse.setHeader(name, value);
}
options = {
pipeline,
filePath,
preload: preloadedComponent,
pathname,
request,
route,
};
mod = preloadedComponent;
renderContext = RenderContext.create({
locals,
pipeline,
pathname,
middleware,
request,
routeData: route,
});
// Set user specified headers to response object.
for (const [name, value] of Object.entries(config.server.headers ?? {})) {
if (value) incomingResponse.setHeader(name, value);
}
options = {
pipeline,
filePath,
preload: preloadedComponent,
pathname,
request,
route,
};
mod = preloadedComponent;
const isPrerendered404 = matchedRoute.route.route === '/404' && matchedRoute.route.prerender;
renderContext = RenderContext.create({
locals,
pipeline,
pathname,
middleware: isPrerendered404 ? undefined : middleware,
request,
routeData: route,
});
let response;
try {
response = await renderContext.render(mod);
@ -288,9 +228,9 @@ export async function handleRoute({
}
// Log useful information that the custom 500 page may not display unlike the default error overlay
logger.error('router', err.stack || err.message);
const filePath = new URL(`./${custom500.component}`, config.root);
const preloadedComponent = await pipeline.preload(custom500, filePath);
response = await renderContext.render(preloadedComponent);
const filePath500 = new URL(`./${custom500.component}`, config.root);
const preloaded500Component = await pipeline.preload(custom500, filePath500);
response = await renderContext.render(preloaded500Component);
status = 500;
}

View file

@ -11,7 +11,7 @@ export const onRequest = defineMiddleware(async (context, next) => {
return await next();
}
if (context.url.pathname === '/') {
if (context.url.pathname === '/' || context.url.pathname === '/redirect-me') {
return redirectToDefaultLocale(context);
}
return new Response(null, {

View file

@ -0,0 +1,12 @@
---
export const prerender = true
---
<html>
<head>
<title>Astro</title>
</head>
<body>
404
</body>
</html>

View file

@ -0,0 +1,13 @@
---
const data = Astro.locals;
---
<html>
<head>
<title>Testing</title>
</head>
<body>
<p>{data?.name}</p>
</body>
</html>

View file

@ -0,0 +1,13 @@
---
const data = Astro.locals;
---
<html>
<head>
<title>Testing</title>
</head>
<body>
<p>{data?.name}</p>
</body>
</html>

View file

@ -52,6 +52,11 @@ describe('Dev server manual routing', () => {
text = await response.text();
assert.equal(text.includes('Hola.'), true);
});
it('should not redirect prerendered 404 routes in dev', async () => {
const response = await fixture.fetch('/redirect-me');
assert.equal(response.status, 404);
});
});
// SSG