0
Fork 0
mirror of https://github.com/withastro/astro.git synced 2025-03-31 23:31:30 -05:00

chore: move tests to node (#10127)

* chore: move tests to node

* revert

* skip one test for now
This commit is contained in:
Emanuele Stoppa 2024-02-15 12:22:53 +00:00 committed by GitHub
parent e845fb2eaa
commit 9eb37a31a7
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
10 changed files with 294 additions and 263 deletions

View file

@ -107,7 +107,7 @@
"build:ci": "pnpm run prebuild && astro-scripts build \"src/**/*.{ts,js}\" && pnpm run postbuild",
"dev": "astro-scripts dev --copy-wasm --prebuild \"src/runtime/server/astro-island.ts\" --prebuild \"src/runtime/client/{idle,load,media,only,visible}.ts\" \"src/**/*.{ts,js}\"",
"postbuild": "astro-scripts copy \"src/**/*.astro\" && astro-scripts copy \"src/**/*.wasm\"",
"test": "pnpm run test:node && mocha ./test/*.test.js --exit --timeout 30000 --ignore ./test/lit-element.test.js && mocha ./test/lit-element.test.js --timeout 30000",
"test": "pnpm run test:node && mocha ./test/*.test.js --exit --timeout 30000",
"test:match": "mocha ./test/*.test.js --timeout 30000 -g",
"test:e2e": "playwright test",
"test:e2e:match": "playwright test -g",

View file

@ -2,13 +2,13 @@ import { promises as fs } from 'node:fs';
import type { APIRoute } from 'astro';
export const GET: APIRoute = async function get() {
export const GET: APIRoute = async function get(ctx) {
try {
// Image is in the public domain. Sourced from
// https://en.wikipedia.org/wiki/File:Portrait_placeholder.png
const buffer = await fs.readFile('./test/fixtures/non-html-pages/src/images/placeholder.png');
// NOTE: SSG only so not Content-Type needed
return new Response(buffer.buffer)
return new Response(buffer.buffer);
} catch (error: unknown) {
throw new Error(`Something went wrong in placeholder.png route!: ${error as string}`);
}

View file

@ -1,4 +1,5 @@
import { expect } from 'chai';
import assert from 'node:assert/strict';
import { describe, before, it } from 'node:test';
import * as cheerio from 'cheerio';
import { loadFixture } from './test-utils.js';
@ -18,6 +19,6 @@ describe('Lazily imported layouts', () => {
it('Renders styles only once', async () => {
const html = await fixture.readFile('/index.html');
const $ = cheerio.load(html);
expect($('link')).to.have.a.lengthOf(1);
assert.equal($('link').length, 1);
});
});

View file

@ -0,0 +1,154 @@
import assert from 'node:assert/strict';
import { describe, before, it } from 'node:test';
import * as cheerio from 'cheerio';
import { loadFixture } from './test-utils.js';
describe(
'LitElement test',
{ timeout: 300000, skip: 'This goes in conflict with ssr-lit test file' },
() => {
let fixture;
const NODE_VERSION = parseFloat(process.versions.node);
const stripExpressionMarkers = (html) => html.replace(/<!--\/?lit-part-->/g, '');
before(async () => {
// @lit-labs/ssr/ requires Node 13.9 or higher
if (NODE_VERSION < 13.9) {
return;
}
fixture = await loadFixture({
root: './fixtures/lit-element/',
});
await fixture.build();
});
it('Renders a custom element by Constructor', async () => {
// @lit-labs/ssr/ requires Node 13.9 or higher
if (NODE_VERSION < 13.9) {
return;
}
const html = await fixture.readFile('/index.html');
const $ = cheerio.load(html);
// test 1: attributes rendered non reactive properties
assert.equal($('#default').attr('foo'), 'bar');
// test 2: shadow rendered
assert.equal($('#default').html().includes('<div>Testing...</div>'), true);
// test 3: string reactive property set
assert.equal(
stripExpressionMarkers($('#default').html()).includes(`<div id="str">initialized</div>`),
true
);
// test 4: boolean reactive property correctly set
// <my-element bool="false"> Lit will equate to true because it uses
// this.hasAttribute to determine its value
assert.equal(
stripExpressionMarkers($('#default').html()).includes(`<div id="bool">B</div>`),
true
);
// test 5: object reactive property set
// by default objects will be stringified to [object Object]
assert.equal(
stripExpressionMarkers($('#default').html()).includes(`<div id="data">data: 1</div>`),
true
);
// test 6: reactive properties are not rendered as attributes
assert.equal($('#default').attr('obj'), undefined);
assert.equal($('#default').attr('bool'), undefined);
assert.equal($('#default').attr('str'), undefined);
// test 7: reflected reactive props are rendered as attributes
assert.equal($('#default').attr('reflectedbool'), '');
assert.equal($('#default').attr('reflected-str'), 'default reflected string');
assert.equal($('#default').attr('reflected-str-prop'), 'initialized reflected');
});
it('Sets defer-hydration on element only when necessary', async () => {
// @lit-labs/ssr/ requires Node 13.9 or higher
if (NODE_VERSION < 13.9) {
return;
}
const html = await fixture.readFile('/index.html');
const $ = cheerio.load(html);
// test 1: reflected reactive props are rendered as attributes
assert.equal($('#non-deferred').attr('count'), '10');
// test 2: non-reactive props are set as attributes
assert.equal($('#non-deferred').attr('foo'), 'bar');
// test 3: components with only reflected reactive props set are not
// deferred because their state can be completely serialized via attributes
assert.equal($('#non-deferred').attr('defer-hydration'), undefined);
// test 4: components with non-reflected reactive props set are deferred because
// their state needs to be synced with the server on the client.
assert.equal($('#default').attr('defer-hydration'), '');
});
it('Correctly passes child slots', async () => {
// @lit-labs/ssr/ requires Node 13.9 or higher
if (NODE_VERSION < 13.9) {
return;
}
const html = await fixture.readFile('/slots/index.html');
const $ = cheerio.load(html);
const $rootMyElement = $('#root');
const $slottedMyElement = $('#slotted');
const $slottedSlottedMyElement = $('#slotted-slotted');
assert.equal($('#default').length, 3);
// Root my-element
assert.equal($rootMyElement.children('.default').length, 2);
assert.equal($rootMyElement.children('.default').eq(1).text(), 'my-element default 2');
assert.equal($rootMyElement.children('[slot="named"]').length, 4);
assert.equal($rootMyElement.children('[slot="named"]').eq(1).text(), 'my-element named 2');
assert.equal($rootMyElement.children('[slot="named"]').eq(2).attr('id'), 'list');
assert.equal($rootMyElement.children('[slot="named"]').eq(3).attr('id'), 'slotted');
// Slotted my-element first level
assert.equal($slottedMyElement.children('.default').length, 1);
assert.equal(
$slottedMyElement.children('.default').eq(0).text(),
'slotted my-element default'
);
assert.equal($slottedMyElement.children('[slot="named"]').length, 3);
assert.equal(
$slottedMyElement.children('[slot="named"]').eq(1).text(),
'slotted my-element named 2'
);
assert.equal(
$slottedMyElement.children('[slot="named"]').eq(2).attr('id'),
'slotted-slotted'
);
// Slotted my-element second level
assert.equal($slottedSlottedMyElement.children('.default').length, 2);
assert.equal(
$slottedSlottedMyElement.children('.default').eq(1).text(),
'slotted slotted my-element default 2'
);
assert.equal($slottedSlottedMyElement.children('[slot="named"]').length, 2);
assert.equal(
$slottedSlottedMyElement.children('[slot="named"]').eq(1).text(),
'slotted slotted my-element named 2'
);
});
it('Is able to build when behind getStaticPaths', async () => {
const dynamicPage = await fixture.readFile('/1/index.html');
assert.equal(dynamicPage.length > 0, true);
});
}
);

View file

@ -1,139 +0,0 @@
import { expect } from 'chai';
import * as cheerio from 'cheerio';
import { loadFixture } from './test-utils.js';
describe('LitElement test', function () {
this.timeout(30000);
let fixture;
const NODE_VERSION = parseFloat(process.versions.node);
const stripExpressionMarkers = (html) => html.replace(/<!--\/?lit-part-->/g, '');
before(async () => {
// @lit-labs/ssr/ requires Node 13.9 or higher
if (NODE_VERSION < 13.9) {
return;
}
fixture = await loadFixture({
root: './fixtures/lit-element/',
});
await fixture.build();
});
it('Renders a custom element by Constructor', async () => {
// @lit-labs/ssr/ requires Node 13.9 or higher
if (NODE_VERSION < 13.9) {
return;
}
const html = await fixture.readFile('/index.html');
const $ = cheerio.load(html);
// test 1: attributes rendered non reactive properties
expect($('#default').attr('foo')).to.equal('bar');
// test 2: shadow rendered
expect($('#default').html()).to.include(`<div>Testing...</div>`);
// test 3: string reactive property set
expect(stripExpressionMarkers($('#default').html())).to.include(
`<div id="str">initialized</div>`
);
// test 4: boolean reactive property correctly set
// <my-element bool="false"> Lit will equate to true because it uses
// this.hasAttribute to determine its value
expect(stripExpressionMarkers($('#default').html())).to.include(`<div id="bool">B</div>`);
// test 5: object reactive property set
// by default objects will be stringified to [object Object]
expect(stripExpressionMarkers($('#default').html())).to.include(`<div id="data">data: 1</div>`);
// test 6: reactive properties are not rendered as attributes
expect($('#default').attr('obj')).to.equal(undefined);
expect($('#default').attr('bool')).to.equal(undefined);
expect($('#default').attr('str')).to.equal(undefined);
// test 7: reflected reactive props are rendered as attributes
expect($('#default').attr('reflectedbool')).to.equal('');
expect($('#default').attr('reflected-str')).to.equal('default reflected string');
expect($('#default').attr('reflected-str-prop')).to.equal('initialized reflected');
});
it('Sets defer-hydration on element only when necessary', async () => {
// @lit-labs/ssr/ requires Node 13.9 or higher
if (NODE_VERSION < 13.9) {
return;
}
const html = await fixture.readFile('/index.html');
const $ = cheerio.load(html);
// test 1: reflected reactive props are rendered as attributes
expect($('#non-deferred').attr('count')).to.equal('10');
// test 2: non-reactive props are set as attributes
expect($('#non-deferred').attr('foo')).to.equal('bar');
// test 3: components with only reflected reactive props set are not
// deferred because their state can be completely serialized via attributes
expect($('#non-deferred').attr('defer-hydration')).to.equal(undefined);
// test 4: components with non-reflected reactive props set are deferred because
// their state needs to be synced with the server on the client.
expect($('#default').attr('defer-hydration')).to.equal('');
});
it('Correctly passes child slots', async () => {
// @lit-labs/ssr/ requires Node 13.9 or higher
if (NODE_VERSION < 13.9) {
return;
}
const html = await fixture.readFile('/slots/index.html');
const $ = cheerio.load(html);
const $rootMyElement = $('#root');
const $slottedMyElement = $('#slotted');
const $slottedSlottedMyElement = $('#slotted-slotted');
expect($('#default').length).to.equal(3);
// Root my-element
expect($rootMyElement.children('.default').length).to.equal(2);
expect($rootMyElement.children('.default').eq(1).text()).to.equal('my-element default 2');
expect($rootMyElement.children('[slot="named"]').length).to.equal(4);
expect($rootMyElement.children('[slot="named"]').eq(1).text()).to.equal('my-element named 2');
expect($rootMyElement.children('[slot="named"]').eq(2).attr('id')).to.equal('list');
expect($rootMyElement.children('[slot="named"]').eq(3).attr('id')).to.equal('slotted');
// Slotted my-element first level
expect($slottedMyElement.children('.default').length).to.equal(1);
expect($slottedMyElement.children('.default').eq(0).text()).to.equal(
'slotted my-element default'
);
expect($slottedMyElement.children('[slot="named"]').length).to.equal(3);
expect($slottedMyElement.children('[slot="named"]').eq(1).text()).to.equal(
'slotted my-element named 2'
);
expect($slottedMyElement.children('[slot="named"]').eq(2).attr('id')).to.equal(
'slotted-slotted'
);
// Slotted my-element second level
expect($slottedSlottedMyElement.children('.default').length).to.equal(2);
expect($slottedSlottedMyElement.children('.default').eq(1).text()).to.equal(
'slotted slotted my-element default 2'
);
expect($slottedSlottedMyElement.children('[slot="named"]').length).to.equal(2);
expect($slottedSlottedMyElement.children('[slot="named"]').eq(1).text()).to.equal(
'slotted slotted my-element named 2'
);
});
it('Is able to build when behind getStaticPaths', async () => {
const dynamicPage = await fixture.readFile('/1/index.html');
expect(dynamicPage.length).to.be.greaterThan(0);
});
});

View file

@ -1,4 +1,5 @@
import { expect } from 'chai';
import assert from 'node:assert/strict';
import { describe, it, before, after } from 'node:test';
import * as cheerio from 'cheerio';
import { existsSync, readFileSync } from 'node:fs';
import { fileURLToPath } from 'node:url';
@ -24,105 +25,108 @@ describe('Middleware in DEV mode', () => {
it('should render locals data', async () => {
const html = await fixture.fetch('/').then((res) => res.text());
const $ = cheerio.load(html);
expect($('p').html()).to.equal('bar');
assert.equal($('p').html(), 'bar');
});
it('should change locals data based on URL', async () => {
let html = await fixture.fetch('/').then((res) => res.text());
let $ = cheerio.load(html);
expect($('p').html()).to.equal('bar');
assert.equal($('p').html(), 'bar');
html = await fixture.fetch('/lorem').then((res) => res.text());
$ = cheerio.load(html);
expect($('p').html()).to.equal('ipsum');
assert.equal($('p').html(), 'ipsum');
});
it('should call a second middleware', async () => {
let html = await fixture.fetch('/second').then((res) => res.text());
let $ = cheerio.load(html);
expect($('p').html()).to.equal('second');
const html = await fixture.fetch('/second').then((res) => res.text());
const $ = cheerio.load(html);
assert.equal($('p').html(), 'second');
});
it('should successfully create a new response', async () => {
let html = await fixture.fetch('/rewrite').then((res) => res.text());
let $ = cheerio.load(html);
expect($('p').html()).to.be.null;
expect($('span').html()).to.equal('New content!!');
const html = await fixture.fetch('/rewrite').then((res) => res.text());
const $ = cheerio.load(html);
assert.equal($('p').html(), null);
assert.equal($('span').html(), 'New content!!');
});
it('should return a new response that is a 500', async () => {
await fixture.fetch('/broken-500').then((res) => {
expect(res.status).to.equal(500);
assert.equal(res.status, 500);
return res.text();
});
});
it('should successfully render a page if the middleware calls only next() and returns nothing', async () => {
let html = await fixture.fetch('/not-interested').then((res) => res.text());
let $ = cheerio.load(html);
expect($('p').html()).to.equal('Not interested');
const html = await fixture.fetch('/not-interested').then((res) => res.text());
const $ = cheerio.load(html);
assert.equal($('p').html(), 'Not interested');
});
it("should throw an error when the middleware doesn't call next or doesn't return a response", async () => {
let html = await fixture.fetch('/does-nothing').then((res) => res.text());
let $ = cheerio.load(html);
expect($('title').html()).to.equal('MiddlewareNoDataOrNextCalled');
const html = await fixture.fetch('/does-nothing').then((res) => res.text());
const $ = cheerio.load(html);
assert.equal($('title').html(), 'MiddlewareNoDataOrNextCalled');
});
it('should allow setting cookies', async () => {
let res = await fixture.fetch('/');
expect(res.headers.get('set-cookie')).to.equal('foo=bar');
const res = await fixture.fetch('/');
assert.equal(res.headers.get('set-cookie'), 'foo=bar');
});
it('should be able to clone the response', async () => {
let res = await fixture.fetch('/clone');
let html = await res.text();
expect(html).to.contain('it works');
const res = await fixture.fetch('/clone');
const html = await res.text();
assert.equal(html.includes('it works'), true);
});
it('should forward cookies set in a component when the middleware returns a new response', async () => {
let res = await fixture.fetch('/return-response-cookies');
let headers = res.headers;
expect(headers.get('set-cookie')).to.not.equal(null);
const res = await fixture.fetch('/return-response-cookies');
const headers = res.headers;
assert.notEqual(headers.get('set-cookie'), null);
});
describe('Integration hooks', () => {
it('Integration middleware marked as "pre" runs', async () => {
let res = await fixture.fetch('/integration-pre');
let json = await res.json();
expect(json.pre).to.equal('works');
const res = await fixture.fetch('/integration-pre');
const json = await res.json();
assert.equal(json.pre, 'works');
});
it('Integration middleware marked as "post" runs', async () => {
let res = await fixture.fetch('/integration-post');
let json = await res.json();
expect(json.post).to.equal('works');
const res = await fixture.fetch('/integration-post');
const json = await res.json();
assert.equal(json.post, 'works');
});
});
});
describe('Integration hooks with no user middleware', () => {
before(async () => {
fixture = await loadFixture({
root: './fixtures/middleware-no-user-middleware/',
});
devServer = await fixture.startDevServer();
describe('Integration hooks with no user middleware', () => {
/** @type {import('./test-utils').Fixture} */
let fixture;
let devServer;
before(async () => {
fixture = await loadFixture({
root: './fixtures/middleware-no-user-middleware/',
});
devServer = await fixture.startDevServer();
});
after(async () => {
await devServer.stop();
});
after(async () => {
await devServer.stop();
});
it('Integration middleware marked as "pre" runs', async () => {
let res = await fixture.fetch('/pre');
let json = await res.json();
expect(json.pre).to.equal('works');
});
it('Integration middleware marked as "pre" runs', async () => {
const res = await fixture.fetch('/pre');
const json = await res.json();
assert.equal(json.pre, 'works');
});
it('Integration middleware marked as "post" runs', async () => {
let res = await fixture.fetch('/post');
let json = await res.json();
expect(json.post).to.equal('works');
});
it('Integration middleware marked as "post" runs', async () => {
const res = await fixture.fetch('/post');
const json = await res.json();
assert.equal(json.post, 'works');
});
});
@ -140,17 +144,17 @@ describe('Middleware in PROD mode, SSG', () => {
it('should render locals data', async () => {
const html = await fixture.readFile('/index.html');
const $ = cheerio.load(html);
expect($('p').html()).to.equal('bar');
assert.equal($('p').html(), 'bar');
});
it('should change locals data based on URL', async () => {
let html = await fixture.readFile('/index.html');
let $ = cheerio.load(html);
expect($('p').html()).to.equal('bar');
assert.equal($('p').html(), 'bar');
html = await fixture.readFile('/second/index.html');
$ = cheerio.load(html);
expect($('p').html()).to.equal('second');
assert.equal($('p').html(), 'second');
});
});
@ -176,32 +180,32 @@ describe('Middleware API in PROD mode, SSR', () => {
const response = await app.render(request);
const html = await response.text();
const $ = cheerio.load(html);
expect($('p').html()).to.equal('bar');
assert.equal($('p').html(), 'bar');
});
it('should change locals data based on URL', async () => {
let response = await app.render(new Request('http://example.com/'));
let html = await response.text();
let $ = cheerio.load(html);
expect($('p').html()).to.equal('bar');
assert.equal($('p').html(), 'bar');
response = await app.render(new Request('http://example.com/lorem'));
html = await response.text();
$ = cheerio.load(html);
expect($('p').html()).to.equal('ipsum');
assert.equal($('p').html(), 'ipsum');
});
it('should successfully redirect to another page', async () => {
const request = new Request('http://example.com/redirect');
const response = await app.render(request);
expect(response.status).to.equal(302);
assert.equal(response.status, 302);
});
it('should call a second middleware', async () => {
const response = await app.render(new Request('http://example.com/second'));
const html = await response.text();
const $ = cheerio.load(html);
expect($('p').html()).to.equal('second');
assert.equal($('p').html(), 'second');
});
it('should successfully create a new response', async () => {
@ -209,14 +213,14 @@ describe('Middleware API in PROD mode, SSR', () => {
const response = await app.render(request);
const html = await response.text();
const $ = cheerio.load(html);
expect($('p').html()).to.be.null;
expect($('span').html()).to.equal('New content!!');
assert.equal($('p').html(), null);
assert.equal($('span').html(), 'New content!!');
});
it('should return a new response that is a 500', async () => {
const request = new Request('http://example.com/broken-500');
const response = await app.render(request);
expect(response.status).to.equal(500);
assert.equal(response.status, 500);
});
it('should successfully render a page if the middleware calls only next() and returns nothing', async () => {
@ -224,7 +228,7 @@ describe('Middleware API in PROD mode, SSR', () => {
const response = await app.render(request);
const html = await response.text();
const $ = cheerio.load(html);
expect($('p').html()).to.equal('Not interested');
assert.equal($('p').html(), 'Not interested');
});
it("should throw an error when the middleware doesn't call next or doesn't return a response", async () => {
@ -232,21 +236,21 @@ describe('Middleware API in PROD mode, SSR', () => {
const response = await app.render(request);
const html = await response.text();
const $ = cheerio.load(html);
expect($('title').html()).to.not.equal('MiddlewareNoDataReturned');
assert.notEqual($('title').html(), 'MiddlewareNoDataReturned');
});
it('should correctly work for API endpoints that return a Response object', async () => {
const request = new Request('http://example.com/api/endpoint');
const response = await app.render(request);
expect(response.status).to.equal(200);
expect(response.headers.get('Content-Type')).to.equal('application/json');
assert.equal(response.status, 200);
assert.equal(response.headers.get('Content-Type'), 'application/json');
});
it('should correctly manipulate the response coming from API endpoints (not simple)', async () => {
const request = new Request('http://example.com/api/endpoint');
const response = await app.render(request);
const text = await response.text();
expect(text.includes('REDACTED')).to.be.true;
assert.equal(text.includes('REDACTED'), true);
});
it('should correctly call the middleware function for 404', async () => {
@ -254,8 +258,8 @@ describe('Middleware API in PROD mode, SSR', () => {
const routeData = app.match(request);
const response = await app.render(request, { routeData });
const text = await response.text();
expect(text.includes('Error')).to.be.true;
expect(text.includes('bar')).to.be.true;
assert.equal(text.includes('Error'), true);
assert.equal(text.includes('bar'), true);
});
it('should render 500.astro when the middleware throws an error', async () => {
@ -263,10 +267,10 @@ describe('Middleware API in PROD mode, SSR', () => {
const routeData = app.match(request);
const response = await app.render(request, routeData);
expect(response).to.deep.include({ status: 500 });
assert.equal(response.status, 500);
const text = await response.text();
expect(text).to.include('<h1>There was an error rendering the page.</h1>');
assert.equal(text.includes('<h1>There was an error rendering the page.</h1>'), true);
});
it('should correctly render the page even when custom headers are set in a middleware', async () => {
@ -274,13 +278,13 @@ describe('Middleware API in PROD mode, SSR', () => {
const routeData = app.match(request);
const response = await app.render(request, { routeData });
expect(response).to.deep.include({ status: 404 });
expect(response.headers.get('content-type')).equal('text/html');
assert.equal(response.status, 404);
assert.equal(response.headers.get('content-type'), 'text/html');
});
it('can set locals for prerendered pages to use', async () => {
const text = await fixture.readFile('/client/prerendered/index.html');
expect(text.includes('<p>yes they can!</p>')).to.be.true;
assert.equal(text.includes('<p>yes they can!</p>'), true);
});
// keep this last
@ -303,14 +307,14 @@ describe('Middleware API in PROD mode, SSR', () => {
}),
});
await fixture.build();
expect(middlewarePath).to.not.be.undefined;
assert.ok(middlewarePath);
try {
const path = fileURLToPath(middlewarePath);
expect(existsSync(path)).to.be.true;
assert.equal(existsSync(path), true);
const content = readFileSync(fileURLToPath(middlewarePath), 'utf-8');
expect(content.length).to.be.greaterThan(0);
assert.equal(content.length > 0, true);
} catch (e) {
throw e;
assert.fail();
}
});
});
@ -333,37 +337,43 @@ describe('Middleware with tailwind', () => {
const bundledCSS = (await fixture.readFile(bundledCSSHREF.replace(/^\/?/, '/')))
.replace(/\s/g, '')
.replace('/n', '');
expect(bundledCSS.includes('--tw-content')).to.be.true;
assert.equal(bundledCSS.includes('--tw-content'), true);
});
});
// `loadTestAdapterApp()` does not understand how to load the page with `functionPerRoute`
// since there's no `entry.mjs`. Skip for now.
describe.skip('Middleware supports functionPerRoute feature', () => {
/** @type {import('./test-utils').Fixture} */
let fixture;
describe(
'Middleware supports functionPerRoute feature',
{
skip: "`loadTestAdapterApp()` does not understand how to load the page with `functionPerRoute` since there's no `entry.mjs`",
},
() => {
/** @type {import('./test-utils').Fixture} */
let fixture;
before(async () => {
fixture = await loadFixture({
root: './fixtures/middleware space/',
output: 'server',
adapter: testAdapter({
extendAdapter: {
adapterFeatures: {
functionPerRoute: true,
before(async () => {
fixture = await loadFixture({
root: './fixtures/middleware space/',
output: 'server',
adapter: testAdapter({
extendAdapter: {
adapterFeatures: {
functionPerRoute: true,
},
},
},
}),
}),
});
await fixture.build();
});
await fixture.build();
});
it('should not render locals data because the page does not export it', async () => {
const app = await fixture.loadTestAdapterApp();
const request = new Request('http://example.com/');
const response = await app.render(request);
const html = await response.text();
const $ = cheerio.load(html);
expect($('p').html()).to.not.equal('bar');
});
});
it('should not render locals data because the page does not export it', async () => {
const app = await fixture.loadTestAdapterApp();
const request = new Request('http://example.com/');
const response = await app.render(request);
const html = await response.text();
const $ = cheerio.load(html);
assert.equal($('p').html(), 'bar');
});
}
);

View file

@ -1,4 +1,5 @@
import { expect } from 'chai';
import assert from 'node:assert/strict';
import { describe, it, before, after } from 'node:test';
import { loadFixture } from './test-utils.js';
import testAdapter from './test-adapter.js';
@ -37,10 +38,10 @@ describe('HTML minification', () => {
});
it('should emit compressed HTML in the emitted file', async () => {
let res = await fixture.fetch(`/`);
expect(res.status).to.equal(200);
let res = await fixture.fetch('/');
assert.equal(res.status, 200);
const html = await res.text();
expect(NEW_LINES.test(removeDoctypeLineInDev(html))).to.equal(false);
assert.equal(NEW_LINES.test(removeDoctypeLineInDev(html)), false);
});
});
@ -57,7 +58,7 @@ describe('HTML minification', () => {
it('should emit compressed HTML in the emitted file', async () => {
const html = await fixture.readFile('/index.html');
expect(NEW_LINES.test(html)).to.equal(false);
assert.equal(NEW_LINES.test(html), false);
});
});
@ -79,7 +80,7 @@ describe('HTML minification', () => {
const request = new Request('http://example.com/');
const response = await app.render(request);
const html = await response.text();
expect(NEW_LINES.test(removeDoctypeLine(html))).to.equal(false);
assert.equal(NEW_LINES.test(removeDoctypeLine(html)), false);
});
});
});

View file

@ -1,4 +1,5 @@
import { expect } from 'chai';
import assert from 'node:assert/strict';
import { describe, it, before } from 'node:test';
import * as cheerio from 'cheerio';
import { loadFixture } from './test-utils.js';
@ -16,6 +17,6 @@ describe('Multiple renderers', () => {
it('when the first throws but the second does not, use the second renderer', async () => {
const html = await fixture.readFile('/index.html');
const $ = cheerio.load(html);
expect($('#component')).to.have.a.lengthOf(1);
assert.equal($('#component').length, 1);
});
});

View file

@ -1,4 +1,5 @@
import { expect } from 'chai';
import assert from 'node:assert/strict';
import { describe, it, before } from 'node:test';
import * as cheerio from 'cheerio';
import { loadFixture } from './test-utils.js';
@ -15,7 +16,7 @@ describe('Non-ASCII Path Test', () => {
const html = await fixture.readFile(`/index.html`);
const $ = cheerio.load(html);
expect($('h1').text()).to.equal('测试 OK');
assert.equal($('h1').text(), '测试 OK');
});
});
});

View file

@ -1,4 +1,5 @@
import { expect } from 'chai';
import assert from 'node:assert/strict';
import { describe, it, before } from 'node:test';
import { loadFixture } from './test-utils.js';
describe('Non-HTML Pages', () => {
@ -12,8 +13,8 @@ describe('Non-HTML Pages', () => {
describe('json', () => {
it('should match contents', async () => {
const json = JSON.parse(await fixture.readFile('/about.json'));
expect(json).to.have.property('name', 'Astro');
expect(json).to.have.property('url', 'https://astro.build/');
assert.equal(json.name, 'Astro');
assert.equal(json.url, 'https://astro.build/');
});
});
@ -25,12 +26,13 @@ describe('Non-HTML Pages', () => {
const hex = Buffer.from(buffer, 'base64').toString('hex');
const firstHexByte = hex.slice(0, 2);
// If we accidentally utf8 encode the png, the first byte (in hex) will be 'c2'
expect(firstHexByte).to.not.equal('c2');
assert.notEqual(firstHexByte, 'c2');
// and if correctly encoded in binary, it should be '89'
expect(firstHexByte).to.equal('89');
assert.equal(firstHexByte, '89');
// Make sure the whole buffer (in base64) matches this snapshot
expect(buffer).to.equal(
assert.equal(
buffer,
'iVBORw0KGgoAAAANSUhEUgAAAGQAAACWCAMAAAAfZt10AAAABlBMVEXd3d3+/v7B/CFgAAAA3UlEQVR42u3ZMQ7DIBQFQeb+l06bNgUbG/5eYApLFjzWNE3TNE3TNE035av9AhAQEBBQGAQEFAaFQWFQGBQGhUGCKAwKgwQpDJ6JECgCRYIEikH8YAyCRyEGyRCDvBWRIPNNBpm/8G6kUM45EhXKlQfuFSHFpbFH+jt2j/S7xwqUYvBaCRIozZy6X2km7v1K8uwQIIWBwkBAQEBg3Tyj3z4LnzRBKgwKg8KgMEgQhaEwSBCFQWBEiMIgQQqDBCkMEqQw+APixYgcsa0TERs7D/F6xGmIAxCD/Iw4AvEB92Ec3ZAPdlMAAAAASUVORK5CYII='
);
});