2024-03-19 10:43:08 -05:00
|
|
|
import * as assert from 'node:assert/strict';
|
2024-03-19 10:44:07 -05:00
|
|
|
import { after, before, describe, it } from 'node:test';
|
2024-01-25 05:23:27 -05:00
|
|
|
import * as cheerio from 'cheerio';
|
2024-02-22 08:58:10 -05:00
|
|
|
import nodejs from '../dist/index.js';
|
|
|
|
import { loadFixture } from './test-utils.js';
|
2024-01-25 05:23:27 -05:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @typedef {import('../../../astro/test/test-utils').Fixture} Fixture
|
|
|
|
*/
|
|
|
|
|
|
|
|
async function load() {
|
2024-01-25 05:24:46 -05:00
|
|
|
const mod = await import(
|
|
|
|
`./fixtures/trailing-slash/dist/server/entry.mjs?dropcache=${Date.now()}`
|
|
|
|
);
|
2024-01-25 05:23:27 -05:00
|
|
|
return mod;
|
|
|
|
}
|
|
|
|
|
|
|
|
describe('Trailing slash', () => {
|
|
|
|
/** @type {import('./test-utils').Fixture} */
|
|
|
|
let fixture;
|
|
|
|
let server;
|
|
|
|
describe('Always', async () => {
|
|
|
|
describe('With base', async () => {
|
|
|
|
before(async () => {
|
|
|
|
process.env.ASTRO_NODE_AUTOSTART = 'disabled';
|
|
|
|
process.env.PRERENDER = true;
|
|
|
|
|
|
|
|
fixture = await loadFixture({
|
|
|
|
root: './fixtures/trailing-slash/',
|
|
|
|
base: '/some-base',
|
|
|
|
output: 'hybrid',
|
|
|
|
trailingSlash: 'always',
|
|
|
|
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 prerendered base route', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'Index');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with redirect', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 301);
|
|
|
|
assert.equal(res.headers.get('location'), '/some-base/one/');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with redirect and query params', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one?foo=bar`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 301);
|
|
|
|
assert.equal(res.headers.get('location'), '/some-base/one/?foo=bar');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with query params', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one/?foo=bar`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'One');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
2024-03-19 10:43:08 -05:00
|
|
|
|
|
|
|
it('Does not add trailing slash to subresource urls', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one.css`);
|
|
|
|
const css = await res.text();
|
|
|
|
|
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal(css, 'h1 { color: red; }\n');
|
2024-03-19 10:44:07 -05:00
|
|
|
});
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
describe('Without base', async () => {
|
|
|
|
before(async () => {
|
|
|
|
process.env.ASTRO_NODE_AUTOSTART = 'disabled';
|
|
|
|
process.env.PRERENDER = true;
|
|
|
|
|
|
|
|
fixture = await loadFixture({
|
|
|
|
root: './fixtures/trailing-slash/',
|
|
|
|
output: 'hybrid',
|
|
|
|
trailingSlash: 'always',
|
|
|
|
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 prerendered base route', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'Index');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with redirect', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/one`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 301);
|
|
|
|
assert.equal(res.headers.get('location'), '/one/');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with redirect and query params', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/one?foo=bar`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 301);
|
|
|
|
assert.equal(res.headers.get('location'), '/one/?foo=bar');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with query params', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/one/?foo=bar`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'One');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
2024-03-19 10:43:08 -05:00
|
|
|
|
|
|
|
it('Does not add trailing slash to subresource urls', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/one.css`);
|
|
|
|
const css = await res.text();
|
|
|
|
|
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal(css, 'h1 { color: red; }\n');
|
2024-03-19 10:44:07 -05:00
|
|
|
});
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
describe('Never', async () => {
|
|
|
|
describe('With base', async () => {
|
|
|
|
before(async () => {
|
|
|
|
process.env.ASTRO_NODE_AUTOSTART = 'disabled';
|
|
|
|
process.env.PRERENDER = true;
|
|
|
|
|
|
|
|
fixture = await loadFixture({
|
|
|
|
root: './fixtures/trailing-slash/',
|
|
|
|
base: '/some-base',
|
|
|
|
output: 'hybrid',
|
|
|
|
trailingSlash: 'never',
|
|
|
|
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 prerendered base route', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'Index');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with redirect', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one/`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 301);
|
|
|
|
assert.equal(res.headers.get('location'), '/some-base/one');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with redirect and query params', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one/?foo=bar`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 301);
|
|
|
|
assert.equal(res.headers.get('location'), '/some-base/one?foo=bar');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with query params', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one?foo=bar`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'One');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
describe('Without base', async () => {
|
|
|
|
before(async () => {
|
|
|
|
process.env.ASTRO_NODE_AUTOSTART = 'disabled';
|
|
|
|
process.env.PRERENDER = true;
|
|
|
|
|
|
|
|
fixture = await loadFixture({
|
|
|
|
root: './fixtures/trailing-slash/',
|
|
|
|
output: 'hybrid',
|
|
|
|
trailingSlash: 'never',
|
|
|
|
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 prerendered base route', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'Index');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with redirect', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/one/`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 301);
|
|
|
|
assert.equal(res.headers.get('location'), '/one');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with redirect and query params', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/one/?foo=bar`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 301);
|
|
|
|
assert.equal(res.headers.get('location'), '/one?foo=bar');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route and query params', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/one?foo=bar`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'One');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
describe('Ignore', async () => {
|
|
|
|
describe('With base', async () => {
|
|
|
|
before(async () => {
|
|
|
|
process.env.ASTRO_NODE_AUTOSTART = 'disabled';
|
|
|
|
process.env.PRERENDER = true;
|
|
|
|
|
|
|
|
fixture = await loadFixture({
|
|
|
|
root: './fixtures/trailing-slash/',
|
|
|
|
base: '/some-base',
|
|
|
|
output: 'hybrid',
|
|
|
|
trailingSlash: 'ignore',
|
|
|
|
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 prerendered base route', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'Index');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with slash', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one/`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'One');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route without slash', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'One');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with slash and query params', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one/?foo=bar`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'One');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route without slash and with query params', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one?foo=bar`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'One');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
describe('Without base', async () => {
|
|
|
|
before(async () => {
|
|
|
|
process.env.ASTRO_NODE_AUTOSTART = 'disabled';
|
|
|
|
process.env.PRERENDER = true;
|
|
|
|
|
|
|
|
fixture = await loadFixture({
|
|
|
|
root: './fixtures/trailing-slash/',
|
|
|
|
output: 'hybrid',
|
|
|
|
trailingSlash: 'ignore',
|
|
|
|
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 prerendered base route', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'Index');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with slash', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/one/`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'One');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route without slash', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/one`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'One');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route with slash and query params', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/one/?foo=bar`, {
|
2024-01-25 05:24:46 -05:00
|
|
|
redirect: 'manual',
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'One');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
|
|
|
|
it('Can render prerendered route without slash and with query params', async () => {
|
|
|
|
const res = await fetch(`http://${server.host}:${server.port}/one?foo=bar`);
|
|
|
|
const html = await res.text();
|
|
|
|
const $ = cheerio.load(html);
|
|
|
|
|
2024-03-19 10:43:08 -05:00
|
|
|
assert.equal(res.status, 200);
|
|
|
|
assert.equal($('h1').text(), 'One');
|
2024-01-25 05:23:27 -05:00
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|