import { expect } from 'chai'; import * as cheerio from 'cheerio'; import nodejs from '../dist/index.js'; import { loadFixture } from './test-utils.js'; /** * @typedef {import('../../../astro/test/test-utils').Fixture} Fixture */ async function load() { const mod = await import( `./fixtures/prerender-404-500/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({ // 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', root: './fixtures/prerender-404-500/', 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 url = `http://${server.host}:${server.port}/some-base/missing`; const res1 = await fetch(url); 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($('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($('style')).to.have.a.lengthOf(1); }); }); describe('Without base', async () => { before(async () => { process.env.ASTRO_NODE_AUTOSTART = 'disabled'; process.env.PRERENDER = true; fixture = await loadFixture({ // 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', 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}/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 url = `http://${server.host}:${server.port}/some-base/missing`; const res1 = await fetch(url); 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($('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({ // 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', root: './fixtures/prerender-404-500/', output: 'hybrid', 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 url = `http://${server.host}:${server.port}/some-base/missing`; const res1 = await fetch(url); 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($('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({ // 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', 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}/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 url = `http://${server.host}:${server.port}/missing`; const res1 = await fetch(url); 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($('body').text()).to.equal('Page does not exist'); }); }); });