Prerendering corner cases (#8070)
Co-authored-by: Nate Moore <natemoo-re@users.noreply.github.com>
This commit is contained in:
parent
5b4b782451
commit
097a8e4e91
16 changed files with 221 additions and 48 deletions
5
.changeset/many-actors-flash.md
Normal file
5
.changeset/many-actors-flash.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
"astro": patch
|
||||||
|
---
|
||||||
|
|
||||||
|
Fix a handful of edge cases with prerendered 404/500 pages
|
|
@ -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, {
|
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)
|
// 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
|
// Otherwise, the user set a specific status while rendering and we should respect that one
|
||||||
status: status === 200 ? newResponse.status : status,
|
const status =
|
||||||
|
override?.status ? override.status :
|
||||||
|
oldResponse.status === 200 ? newResponse.status :
|
||||||
|
oldResponse.status
|
||||||
|
|
||||||
|
return new Response(newResponse.body, {
|
||||||
|
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)),
|
||||||
});
|
});
|
||||||
|
|
|
@ -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' });
|
||||||
})
|
})
|
||||||
|
|
|
@ -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;
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
|
@ -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:*"
|
3
packages/integrations/node/test/fixtures/prerender-404-500/src/external-stylesheet.css
vendored
Normal file
3
packages/integrations/node/test/fixtures/prerender-404-500/src/external-stylesheet.css
vendored
Normal file
|
@ -0,0 +1,3 @@
|
||||||
|
body {
|
||||||
|
background-color: ivory;
|
||||||
|
}
|
17
packages/integrations/node/test/fixtures/prerender-404-500/src/nondeterminism-404.ts
vendored
Normal file
17
packages/integrations/node/test/fixtures/prerender-404-500/src/nondeterminism-404.ts
vendored
Normal 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
|
||||||
|
}
|
17
packages/integrations/node/test/fixtures/prerender-404-500/src/nondeterminism-500.ts
vendored
Normal file
17
packages/integrations/node/test/fixtures/prerender-404-500/src/nondeterminism-500.ts
vendored
Normal 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
|
||||||
|
}
|
5
packages/integrations/node/test/fixtures/prerender-404-500/src/pages/404.astro
vendored
Normal file
5
packages/integrations/node/test/fixtures/prerender-404-500/src/pages/404.astro
vendored
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
---
|
||||||
|
import message from "../nondeterminism-404"
|
||||||
|
export const prerender = true;
|
||||||
|
---
|
||||||
|
{message()}
|
6
packages/integrations/node/test/fixtures/prerender-404-500/src/pages/500.astro
vendored
Normal file
6
packages/integrations/node/test/fixtures/prerender-404-500/src/pages/500.astro
vendored
Normal file
|
@ -0,0 +1,6 @@
|
||||||
|
---
|
||||||
|
import "../external-stylesheet.css"
|
||||||
|
import message from "../nondeterminism-500"
|
||||||
|
export const prerender = true
|
||||||
|
---
|
||||||
|
<h1>{message()}</h1>
|
4
packages/integrations/node/test/fixtures/prerender-404-500/src/pages/fivehundred.astro
vendored
Normal file
4
packages/integrations/node/test/fixtures/prerender-404-500/src/pages/fivehundred.astro
vendored
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
---
|
||||||
|
return new Response(null, { status: 500 })
|
||||||
|
---
|
||||||
|
<p>This html will not be served</p>
|
|
@ -1,5 +0,0 @@
|
||||||
---
|
|
||||||
export const prerender = true;
|
|
||||||
---
|
|
||||||
|
|
||||||
Page does not exist
|
|
|
@ -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);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -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');
|
||||||
});
|
});
|
||||||
});
|
});
|
|
@ -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:*
|
||||||
|
|
Loading…
Reference in a new issue