mirror of
https://github.com/withastro/astro.git
synced 2025-01-13 22:11:20 -05:00
a23c69d0d0
* wip * done i think * Add changeset * Use hook instead * Reorder hooks [skip ci] * Update .changeset/eleven-pens-glow.md Co-authored-by: Sarah Rainsberger <sarah@rainsberger.ca> * Fix run * Fix link * Add link Co-authored-by: Sarah Rainsberger <sarah11918@users.noreply.github.com> * More accurate migration [skip ci] --------- Co-authored-by: Sarah Rainsberger <sarah@rainsberger.ca> Co-authored-by: Sarah Rainsberger <sarah11918@users.noreply.github.com>
438 lines
12 KiB
JavaScript
438 lines
12 KiB
JavaScript
import * as assert from 'node:assert/strict';
|
|
import { after, before, describe, it } from 'node:test';
|
|
import * as cheerio from 'cheerio';
|
|
import nodejs from '../dist/index.js';
|
|
import { loadFixture, waitServerListen } from './test-utils.js';
|
|
|
|
/**
|
|
* @typedef {import('../../../astro/test/test-utils').Fixture} Fixture
|
|
*/
|
|
|
|
describe('Prerendering', () => {
|
|
/** @type {import('./test-utils').Fixture} */
|
|
let fixture;
|
|
let server;
|
|
|
|
describe('With base', async () => {
|
|
before(async () => {
|
|
process.env.PRERENDER = true;
|
|
|
|
fixture = await loadFixture({
|
|
base: '/some-base',
|
|
root: './fixtures/prerender/',
|
|
output: 'server',
|
|
outDir: './dist/with-base',
|
|
build: {
|
|
client: './dist/with-base/client',
|
|
server: './dist/with-base/server',
|
|
},
|
|
adapter: nodejs({ mode: 'standalone' }),
|
|
});
|
|
await fixture.build();
|
|
const { startServer } = await fixture.loadAdapterEntryModule();
|
|
let res = startServer();
|
|
server = res.server;
|
|
await waitServerListen(server.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/one`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'One');
|
|
});
|
|
|
|
it('Can render prerendered route', async () => {
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/two`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'Two');
|
|
assert.ok(fixture.pathExists('/client/two/index.html'));
|
|
});
|
|
|
|
it('Can render prerendered route with redirect and query params', async () => {
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/two?foo=bar`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'Two');
|
|
});
|
|
|
|
it('Can render prerendered route with query params', async () => {
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/two/?foo=bar`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'Two');
|
|
});
|
|
|
|
it('Can render prerendered route without trailing slash', async () => {
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/two`, {
|
|
redirect: 'manual',
|
|
});
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'Two');
|
|
});
|
|
});
|
|
|
|
describe('Without base', async () => {
|
|
before(async () => {
|
|
process.env.PRERENDER = true;
|
|
|
|
fixture = await loadFixture({
|
|
root: './fixtures/prerender/',
|
|
output: 'server',
|
|
outDir: './dist/without-base',
|
|
build: {
|
|
client: './dist/without-base/client',
|
|
server: './dist/without-base/server',
|
|
},
|
|
adapter: nodejs({ mode: 'standalone' }),
|
|
});
|
|
await fixture.build();
|
|
const { startServer } = await fixture.loadAdapterEntryModule();
|
|
let res = startServer();
|
|
server = res.server;
|
|
await waitServerListen(server.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}/one`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'One');
|
|
});
|
|
|
|
it('Can render prerendered route', async () => {
|
|
const res = await fetch(`http://${server.host}:${server.port}/two`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'Two');
|
|
assert.ok(fixture.pathExists('/client/two/index.html'));
|
|
});
|
|
|
|
it('Can render prerendered route with redirect and query params', async () => {
|
|
const res = await fetch(`http://${server.host}:${server.port}/two?foo=bar`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'Two');
|
|
});
|
|
|
|
it('Can render prerendered route with query params', async () => {
|
|
const res = await fetch(`http://${server.host}:${server.port}/two/?foo=bar`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'Two');
|
|
});
|
|
});
|
|
|
|
describe('Via integration', () => {
|
|
before(async () => {
|
|
process.env.PRERENDER = false;
|
|
fixture = await loadFixture({
|
|
root: './fixtures/prerender/',
|
|
output: 'server',
|
|
outDir: './dist/via-integration',
|
|
build: {
|
|
client: './dist/via-integration/client',
|
|
server: './dist/via-integration/server',
|
|
},
|
|
adapter: nodejs({ mode: 'standalone' }),
|
|
integrations: [
|
|
{
|
|
name: 'test',
|
|
hooks: {
|
|
'astro:route:setup': ({ route }) => {
|
|
if (route.component.endsWith('two.astro')) {
|
|
route.prerender = true;
|
|
}
|
|
},
|
|
},
|
|
},
|
|
],
|
|
});
|
|
await fixture.build();
|
|
const { startServer } = await fixture.loadAdapterEntryModule();
|
|
let res = startServer();
|
|
server = res.server;
|
|
await waitServerListen(server.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}/one`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'One');
|
|
});
|
|
|
|
it('Can render prerendered route', async () => {
|
|
const res = await fetch(`http://${server.host}:${server.port}/two`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'Two');
|
|
assert.ok(fixture.pathExists('/client/two/index.html'));
|
|
});
|
|
});
|
|
|
|
describe('Dev', () => {
|
|
let devServer;
|
|
|
|
before(async () => {
|
|
process.env.PRERENDER = true;
|
|
|
|
fixture = await loadFixture({
|
|
root: './fixtures/prerender/',
|
|
output: 'server',
|
|
outDir: './dist/dev',
|
|
build: {
|
|
client: './dist/dev/client',
|
|
server: './dist/dev/server',
|
|
},
|
|
adapter: nodejs({ mode: 'standalone' }),
|
|
});
|
|
devServer = await fixture.startDevServer();
|
|
});
|
|
|
|
after(async () => {
|
|
await devServer.stop();
|
|
delete process.env.PRERENDER;
|
|
});
|
|
|
|
it('Can render SSR route', async () => {
|
|
const res = await fixture.fetch(`/one`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'One');
|
|
});
|
|
|
|
it('Can render prerendered route', async () => {
|
|
const res = await fixture.fetch(`/two`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'Two');
|
|
});
|
|
});
|
|
});
|
|
|
|
describe('Hybrid rendering', () => {
|
|
/** @type {import('./test-utils').Fixture} */
|
|
let fixture;
|
|
let server;
|
|
|
|
describe('With base', () => {
|
|
before(async () => {
|
|
process.env.PRERENDER = false;
|
|
fixture = await loadFixture({
|
|
base: '/some-base',
|
|
root: './fixtures/prerender/',
|
|
output: 'hybrid',
|
|
outDir: './dist/hybrid-with-base',
|
|
build: {
|
|
client: './dist/hybrid-with-base/client',
|
|
server: './dist/hybrid-with-base/server',
|
|
},
|
|
adapter: nodejs({ mode: 'standalone' }),
|
|
});
|
|
await fixture.build();
|
|
const { startServer } = await fixture.loadAdapterEntryModule();
|
|
let res = startServer();
|
|
server = res.server;
|
|
await waitServerListen(server.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/two`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'Two');
|
|
});
|
|
|
|
it('Can render prerendered route', async () => {
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'One');
|
|
assert.ok(fixture.pathExists('/client/one/index.html'));
|
|
});
|
|
|
|
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`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'One');
|
|
});
|
|
|
|
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);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'One');
|
|
});
|
|
|
|
it('Can render prerendered route without trailing slash', async () => {
|
|
const res = await fetch(`http://${server.host}:${server.port}/some-base/one`, {
|
|
redirect: 'manual',
|
|
});
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'One');
|
|
});
|
|
});
|
|
|
|
describe('Without base', () => {
|
|
before(async () => {
|
|
process.env.PRERENDER = false;
|
|
fixture = await loadFixture({
|
|
root: './fixtures/prerender/',
|
|
output: 'hybrid',
|
|
outDir: './dist/hybrid-without-base',
|
|
build: {
|
|
client: './dist/hybrid-without-base/client',
|
|
server: './dist/hybrid-without-base/server',
|
|
},
|
|
adapter: nodejs({ mode: 'standalone' }),
|
|
});
|
|
await fixture.build();
|
|
const { startServer } = await fixture.loadAdapterEntryModule();
|
|
let res = startServer();
|
|
server = res.server;
|
|
await waitServerListen(server.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}/two`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'Two');
|
|
});
|
|
|
|
it('Can render prerendered route', async () => {
|
|
const res = await fetch(`http://${server.host}:${server.port}/one`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'One');
|
|
assert.ok(fixture.pathExists('/client/one/index.html'));
|
|
});
|
|
|
|
it('Can render prerendered route with redirect 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);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'One');
|
|
});
|
|
|
|
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);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'One');
|
|
});
|
|
});
|
|
|
|
describe('Shared modules', () => {
|
|
before(async () => {
|
|
process.env.PRERENDER = false;
|
|
|
|
fixture = await loadFixture({
|
|
root: './fixtures/prerender/',
|
|
output: 'hybrid',
|
|
outDir: './dist/hybrid-shared-modules',
|
|
build: {
|
|
client: './dist/hybrid-shared-modules/client',
|
|
server: './dist/hybrid-shared-modules/server',
|
|
},
|
|
adapter: nodejs({ mode: 'standalone' }),
|
|
});
|
|
await fixture.build();
|
|
const { startServer } = await fixture.loadAdapterEntryModule();
|
|
let res = startServer();
|
|
server = res.server;
|
|
await waitServerListen(server.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}/third`);
|
|
const html = await res.text();
|
|
const $ = cheerio.load(html);
|
|
|
|
assert.equal(res.status, 200);
|
|
assert.equal($('h1').text(), 'shared');
|
|
});
|
|
});
|
|
});
|