Prerendering corner cases (#8070)

Co-authored-by: Nate Moore <natemoo-re@users.noreply.github.com>
This commit is contained in:
Arsh 2023-08-17 14:04:31 +05:30 committed by GitHub
parent 5b4b782451
commit 097a8e4e91
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
16 changed files with 221 additions and 48 deletions

View file

@ -0,0 +1,5 @@
---
"astro": patch
---
Fix a handful of edge cases with prerendered 404/500 pages

View file

@ -286,10 +286,16 @@ export class App {
const errorRouteData = matchRoute('/' + status, this.#manifestData); const errorRouteData = matchRoute('/' + status, this.#manifestData);
const url = new URL(request.url); const url = new URL(request.url);
if (errorRouteData) { if (errorRouteData) {
if (errorRouteData.prerender && !errorRouteData.route.endsWith(`/${status}`)) { if (errorRouteData.prerender){
const statusURL = new URL(`${this.#baseWithoutTrailingSlash}/${status}`, url); const maybeDotHtml = errorRouteData.route.endsWith(`/${status}`) ? '.html' : ''
const statusURL = new URL(`${this.#baseWithoutTrailingSlash}/${status}${maybeDotHtml}`, url);
const response = await fetch(statusURL.toString()); const response = await fetch(statusURL.toString());
return this.#mergeResponses(response, originalResponse);
// response for /404.html and 500.html is 200, which is not meaningful
// so we create an override
const override = { status }
return this.#mergeResponses(response, originalResponse, override);
} }
const mod = await this.#getModuleForRoute(errorRouteData); const mod = await this.#getModuleForRoute(errorRouteData);
try { try {
@ -316,14 +322,30 @@ export class App {
return response; return response;
} }
#mergeResponses(newResponse: Response, oldResponse?: Response) { #mergeResponses(newResponse: Response, oldResponse?: Response, override?: { status : 404 | 500 }) {
if (!oldResponse) return newResponse; if (!oldResponse) {
const { status, statusText, headers } = oldResponse; if (override !== undefined) {
return new Response(newResponse.body, {
status: override.status,
statusText: newResponse.statusText,
headers: newResponse.headers
})
}
return newResponse;
}
const { statusText, headers } = oldResponse;
// If the the new response did not have a meaningful status, an override may have been provided
// If the original status was 200 (default), override it with the new status (probably 404 or 500)
// Otherwise, the user set a specific status while rendering and we should respect that one
const status =
override?.status ? override.status :
oldResponse.status === 200 ? newResponse.status :
oldResponse.status
return new Response(newResponse.body, { return new Response(newResponse.body, {
// If the original status was 200 (default), override it with the new status (probably 404 or 500) status,
// Otherwise, the user set a specific status while rendering and we should respect that one
status: status === 200 ? newResponse.status : status,
statusText: status === 200 ? newResponse.statusText : statusText, statusText: status === 200 ? newResponse.statusText : statusText,
headers: new Headers(Array.from(headers)), headers: new Headers(Array.from(headers)),
}); });

View file

@ -308,8 +308,12 @@ async function runPostBuildHooks(
async function cleanStaticOutput(opts: StaticBuildOptions, internals: BuildInternals) { async function cleanStaticOutput(opts: StaticBuildOptions, internals: BuildInternals) {
const allStaticFiles = new Set(); const allStaticFiles = new Set();
for (const pageData of eachPageData(internals)) { for (const pageData of eachPageData(internals)) {
if (pageData.route.prerender) if (pageData.route.prerender) {
allStaticFiles.add(internals.pageToBundleMap.get(pageData.moduleSpecifier)); const { moduleSpecifier } = pageData;
const pageBundleId = internals.pageToBundleMap.get(moduleSpecifier);
const entryBundleId = internals.entrySpecifierToBundleMap.get(moduleSpecifier);
allStaticFiles.add(pageBundleId ?? entryBundleId);
}
} }
const ssr = isServerLikeOutput(opts.settings.config); const ssr = isServerLikeOutput(opts.settings.config);
const out = ssr const out = ssr
@ -337,7 +341,8 @@ async function cleanStaticOutput(opts: StaticBuildOptions, internals: BuildInter
// Replace exports (only prerendered pages) with a noop // Replace exports (only prerendered pages) with a noop
let value = 'const noop = () => {};'; let value = 'const noop = () => {};';
for (const e of exports) { for (const e of exports) {
value += `\nexport const ${e.n} = noop;`; if (e.n === 'default') value += `\n export default noop;`
else value += `\nexport const ${e.n} = noop;`;
} }
await fs.promises.writeFile(url, value, { encoding: 'utf8' }); await fs.promises.writeFile(url, value, { encoding: 'utf8' });
}) })

View file

@ -30,8 +30,7 @@ describe('SSR: prerender', () => {
const app = await fixture.loadTestAdapterApp(); const app = await fixture.loadTestAdapterApp();
/** @type {Set<string>} */ /** @type {Set<string>} */
const assets = app.manifest.assets; const assets = app.manifest.assets;
expect(assets.size).to.equal(1); expect(assets).to.contain('/static/index.html');
expect(Array.from(assets)[0].endsWith('static/index.html')).to.be.true;
}); });
}); });

View file

@ -1,7 +1,8 @@
{ {
"name": "@test/nodejs-prerender-404", "name": "@test/nodejs-prerender-404-500",
"version": "0.0.0", "version": "0.0.0",
"private": true, "private": true,
"type": "module",
"dependencies": { "dependencies": {
"astro": "workspace:*", "astro": "workspace:*",
"@astrojs/node": "workspace:*" "@astrojs/node": "workspace:*"

View file

@ -0,0 +1,3 @@
body {
background-color: ivory;
}

View file

@ -0,0 +1,17 @@
// This module is only used by the prerendered 404.astro.
// It exhibits different behavior if it's called more than once,
// which is detected by a test and interpreted as a failure.
let usedOnce = false
let dynamicMessage = "Page was not prerendered"
export default function () {
if (usedOnce === false) {
usedOnce = true
return "Page does not exist"
}
dynamicMessage += "+"
return dynamicMessage
}

View file

@ -0,0 +1,17 @@
// This module is only used by the prerendered 500.astro.
// It exhibits different behavior if it's called more than once,
// which is detected by a test and interpreted as a failure.
let usedOnce = false
let dynamicMessage = "Page was not prerendered"
export default function () {
if (usedOnce === false) {
usedOnce = true
return "Something went wrong"
}
dynamicMessage += "+"
return dynamicMessage
}

View file

@ -0,0 +1,5 @@
---
import message from "../nondeterminism-404"
export const prerender = true;
---
{message()}

View file

@ -0,0 +1,6 @@
---
import "../external-stylesheet.css"
import message from "../nondeterminism-500"
export const prerender = true
---
<h1>{message()}</h1>

View file

@ -0,0 +1,4 @@
---
return new Response(null, { status: 500 })
---
<p>This html will not be served</p>

View file

@ -1,5 +0,0 @@
---
export const prerender = true;
---
Page does not exist

View file

@ -32,9 +32,6 @@ describe('Image endpoint', () => {
'/_image?href=/_astro/some_penguin.97ef5f92.png&w=50&f=webp' '/_image?href=/_astro/some_penguin.97ef5f92.png&w=50&f=webp'
); );
console.log(resImage);
const content = resImage.text();
console.log(content);
expect(resImage.status).to.equal(200); expect(resImage.status).to.equal(200);
}); });
}); });

View file

@ -10,10 +10,11 @@ import { fetch } from 'undici';
async function load() { async function load() {
const mod = await import( const mod = await import(
`./fixtures/prerender-404/dist/server/entry.mjs?dropcache=${Date.now()}` `./fixtures/prerender-404-500/dist/server/entry.mjs?dropcache=${Date.now()}`
); );
return mod; return mod;
} }
describe('Prerender 404', () => { describe('Prerender 404', () => {
/** @type {import('./test-utils').Fixture} */ /** @type {import('./test-utils').Fixture} */
let fixture; let fixture;
@ -25,8 +26,12 @@ describe('Prerender 404', () => {
process.env.PRERENDER = true; process.env.PRERENDER = true;
fixture = await loadFixture({ fixture = await loadFixture({
// inconsequential config that differs between tests
// to bust cache and prevent modules and their state
// from being reused
site: 'https://test.dev/',
base: '/some-base', base: '/some-base',
root: './fixtures/prerender-404/', root: './fixtures/prerender-404-500/',
output: 'server', output: 'server',
adapter: nodejs({ mode: 'standalone' }), adapter: nodejs({ mode: 'standalone' }),
}); });
@ -52,13 +57,54 @@ describe('Prerender 404', () => {
}); });
it('Can handle prerendered 404', async () => { it('Can handle prerendered 404', async () => {
const res = await fetch(`http://${server.host}:${server.port}/some-base/missing`); const url = `http://${server.host}:${server.port}/some-base/missing`;
const html = await res.text(); const res1 = await fetch(url);
const $ = cheerio.load(html); const res2 = await fetch(url);
const res3 = await fetch(url);
expect(res1.status).to.equal(404);
expect(res2.status).to.equal(404);
expect(res3.status).to.equal(404);
const html1 = await res1.text();
const html2 = await res2.text();
const html3 = await res3.text();
expect(html1).to.equal(html2);
expect(html2).to.equal(html3);
const $ = cheerio.load(html1);
expect(res.status).to.equal(404);
expect($('body').text()).to.equal('Page does not exist'); expect($('body').text()).to.equal('Page does not exist');
}); });
it(' Can handle prerendered 500 called indirectly', async () => {
const url = `http://${server.host}:${server.port}/some-base/fivehundred`;
const response1 = await fetch(url);
const response2 = await fetch(url);
const response3 = await fetch(url);
expect(response1.status).to.equal(500);
const html1 = await response1.text();
const html2 = await response2.text();
const html3 = await response3.text();
expect(html1).to.contain("Something went wrong");
expect(html1).to.equal(html2);
expect(html2).to.equal(html3);
});
it('prerendered 500 page includes expected styles', async () => {
const response = await fetch(`http://${server.host}:${server.port}/some-base/fivehundred`);
const html = await response.text();
const $ = cheerio.load(html);
// length will be 0 if the stylesheet does not get included
expect($('link[rel=stylesheet]')).to.have.a.lengthOf(1);
});
}); });
describe('Without base', async () => { describe('Without base', async () => {
@ -67,12 +113,16 @@ describe('Prerender 404', () => {
process.env.PRERENDER = true; process.env.PRERENDER = true;
fixture = await loadFixture({ fixture = await loadFixture({
root: './fixtures/prerender-404/', // inconsequential config that differs between tests
// to bust cache and prevent modules and their state
// from being reused
site: 'https://test.info/',
root: './fixtures/prerender-404-500/',
output: 'server', output: 'server',
adapter: nodejs({ mode: 'standalone' }), adapter: nodejs({ mode: 'standalone' }),
}); });
await fixture.build(); await fixture.build();
const { startServer } = await await load(); const { startServer } = await load();
let res = startServer(); let res = startServer();
server = res.server; server = res.server;
}); });
@ -93,11 +143,24 @@ describe('Prerender 404', () => {
}); });
it('Can handle prerendered 404', async () => { it('Can handle prerendered 404', async () => {
const res = await fetch(`http://${server.host}:${server.port}/missing`); const url = `http://${server.host}:${server.port}/some-base/missing`
const html = await res.text(); const res1 = await fetch(url);
const $ = cheerio.load(html); const res2 = await fetch(url);
const res3 = await fetch(url);
expect(res1.status).to.equal(404);
expect(res2.status).to.equal(404);
expect(res3.status).to.equal(404);
const html1 = await res1.text();
const html2 = await res2.text();
const html3 = await res3.text();
expect(html1).to.equal(html2);
expect(html2).to.equal(html3);
const $ = cheerio.load(html1);
expect(res.status).to.equal(404);
expect($('body').text()).to.equal('Page does not exist'); expect($('body').text()).to.equal('Page does not exist');
}); });
}); });
@ -113,13 +176,17 @@ describe('Hybrid 404', () => {
process.env.ASTRO_NODE_AUTOSTART = 'disabled'; process.env.ASTRO_NODE_AUTOSTART = 'disabled';
process.env.PRERENDER = false; process.env.PRERENDER = false;
fixture = await loadFixture({ fixture = await loadFixture({
// inconsequential config that differs between tests
// to bust cache and prevent modules and their state
// from being reused
site: 'https://test.com/',
base: '/some-base', base: '/some-base',
root: './fixtures/prerender-404/', root: './fixtures/prerender-404-500/',
output: 'hybrid', output: 'hybrid',
adapter: nodejs({ mode: 'standalone' }), adapter: nodejs({ mode: 'standalone' }),
}); });
await fixture.build(); await fixture.build();
const { startServer } = await await load(); const { startServer } = await load();
let res = startServer(); let res = startServer();
server = res.server; server = res.server;
}); });
@ -140,11 +207,24 @@ describe('Hybrid 404', () => {
}); });
it('Can handle prerendered 404', async () => { it('Can handle prerendered 404', async () => {
const res = await fetch(`http://${server.host}:${server.port}/some-base/missing`); const url = `http://${server.host}:${server.port}/some-base/missing`
const html = await res.text(); const res1 = await fetch(url);
const $ = cheerio.load(html); const res2 = await fetch(url);
const res3 = await fetch(url);
expect(res1.status).to.equal(404);
expect(res2.status).to.equal(404);
expect(res3.status).to.equal(404);
const html1 = await res1.text();
const html2 = await res2.text();
const html3 = await res3.text();
expect(html1).to.equal(html2);
expect(html2).to.equal(html3);
const $ = cheerio.load(html1);
expect(res.status).to.equal(404);
expect($('body').text()).to.equal('Page does not exist'); expect($('body').text()).to.equal('Page does not exist');
}); });
}); });
@ -154,12 +234,16 @@ describe('Hybrid 404', () => {
process.env.ASTRO_NODE_AUTOSTART = 'disabled'; process.env.ASTRO_NODE_AUTOSTART = 'disabled';
process.env.PRERENDER = false; process.env.PRERENDER = false;
fixture = await loadFixture({ fixture = await loadFixture({
root: './fixtures/prerender-404/', // inconsequential config that differs between tests
// to bust cache and prevent modules and their state
// from being reused
site: 'https://test.net/',
root: './fixtures/prerender-404-500/',
output: 'hybrid', output: 'hybrid',
adapter: nodejs({ mode: 'standalone' }), adapter: nodejs({ mode: 'standalone' }),
}); });
await fixture.build(); await fixture.build();
const { startServer } = await await load(); const { startServer } = await load();
let res = startServer(); let res = startServer();
server = res.server; server = res.server;
}); });
@ -180,11 +264,24 @@ describe('Hybrid 404', () => {
}); });
it('Can handle prerendered 404', async () => { it('Can handle prerendered 404', async () => {
const res = await fetch(`http://${server.host}:${server.port}/missing`); const url = `http://${server.host}:${server.port}/missing`
const html = await res.text(); const res1 = await fetch(url);
const $ = cheerio.load(html); const res2 = await fetch(url);
const res3 = await fetch(url);
expect(res1.status).to.equal(404);
expect(res2.status).to.equal(404);
expect(res3.status).to.equal(404);
const html1 = await res1.text();
const html2 = await res2.text();
const html3 = await res3.text();
expect(html1).to.equal(html2);
expect(html2).to.equal(html3);
const $ = cheerio.load(html1);
expect(res.status).to.equal(404);
expect($('body').text()).to.equal('Page does not exist'); expect($('body').text()).to.equal('Page does not exist');
}); });
}); });

View file

@ -4641,7 +4641,7 @@ importers:
specifier: workspace:* specifier: workspace:*
version: link:../../../../../astro version: link:../../../../../astro
packages/integrations/node/test/fixtures/prerender-404: packages/integrations/node/test/fixtures/prerender-404-500:
dependencies: dependencies:
'@astrojs/node': '@astrojs/node':
specifier: workspace:* specifier: workspace:*