2021-05-24 17:18:56 -05:00
|
|
|
import fs from 'fs';
|
|
|
|
import path from 'path';
|
2021-06-10 11:30:48 -05:00
|
|
|
import http from 'http';
|
2021-06-11 16:07:30 -05:00
|
|
|
import { green, red } from 'kleur/colors';
|
2021-12-13 15:59:08 -05:00
|
|
|
import { execa } from 'execa';
|
2021-06-10 11:30:48 -05:00
|
|
|
import glob from 'tiny-glob';
|
2021-06-08 10:12:07 -05:00
|
|
|
import { TEMPLATES } from '../dist/templates.js';
|
2021-07-06 14:14:22 -05:00
|
|
|
import { GITHUB_SHA, FIXTURES_DIR } from './helpers.js';
|
2021-05-24 17:18:56 -05:00
|
|
|
|
2021-06-11 16:07:30 -05:00
|
|
|
// helpers
|
2021-06-10 11:30:48 -05:00
|
|
|
async function fetch(url) {
|
2021-12-22 16:11:05 -05:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
http
|
|
|
|
.get(url, (res) => {
|
|
|
|
// not OK
|
|
|
|
if (res.statusCode !== 200) {
|
|
|
|
reject(res.statusCode);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// OK
|
|
|
|
let body = '';
|
|
|
|
res.on('data', (chunk) => {
|
|
|
|
body += chunk;
|
|
|
|
});
|
|
|
|
res.on('end', () => resolve({ statusCode: res.statusCode, body }));
|
|
|
|
})
|
|
|
|
.on('error', (err) => {
|
|
|
|
// other error
|
|
|
|
reject(err);
|
|
|
|
});
|
|
|
|
});
|
2021-06-10 11:30:48 -05:00
|
|
|
}
|
|
|
|
|
2021-06-11 16:07:30 -05:00
|
|
|
function assert(a, b, message) {
|
2021-12-22 16:11:05 -05:00
|
|
|
if (a !== b) throw new Error(red(`✘ ${message}`));
|
2021-06-11 16:07:30 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
async function testTemplate(template) {
|
2021-12-22 16:11:05 -05:00
|
|
|
const templateDir = path.join(FIXTURES_DIR, template);
|
|
|
|
|
|
|
|
// test 1: install
|
|
|
|
const DOES_HAVE = ['.gitignore', 'package.json', 'public', 'src'];
|
|
|
|
const DOES_NOT_HAVE = ['.git', 'meta.json'];
|
|
|
|
|
|
|
|
// test 1a: expect template contains essential files & folders
|
|
|
|
for (const file of DOES_HAVE) {
|
|
|
|
assert(fs.existsSync(path.join(templateDir, file)), true, `[${template}] has ${file}`);
|
|
|
|
}
|
|
|
|
// test 1b: expect template DOES NOT contain files supposed to be stripped away
|
|
|
|
for (const file of DOES_NOT_HAVE) {
|
|
|
|
assert(fs.existsSync(path.join(templateDir, file)), false, `[${template}] cleaned up ${file}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// test 2: build
|
|
|
|
const MUST_HAVE_FILES = ['index.html', '_astro'];
|
|
|
|
await execa('npm', ['run', 'build'], { cwd: templateDir });
|
|
|
|
const builtFiles = await glob('**/*', { cwd: path.join(templateDir, 'dist') });
|
|
|
|
// test 2a: expect all files built successfully
|
|
|
|
for (const file of MUST_HAVE_FILES) {
|
|
|
|
assert(builtFiles.includes(file), true, `[${template}] built ${file}`);
|
|
|
|
}
|
|
|
|
|
|
|
|
// test 3: dev server (should happen after build so dependency install can be reused)
|
|
|
|
|
|
|
|
// TODO: fix dev server test in CI
|
|
|
|
if (process.env.CI === true) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// start dev server in background & wait until ready
|
|
|
|
const templateIndex = TEMPLATES.findIndex(({ value }) => value === template);
|
|
|
|
const port = 3000 + templateIndex; // use different port per-template
|
|
|
|
const devServer = execa('npm', ['run', 'start', '--', '--port', port], { cwd: templateDir });
|
|
|
|
let sigkill = setTimeout(() => {
|
|
|
|
throw new Error(`Dev server failed to start`); // if 10s has gone by with no update, kill process
|
|
|
|
}, 10000);
|
|
|
|
|
|
|
|
// read stdout until "Server started" appears
|
|
|
|
await new Promise((resolve, reject) => {
|
|
|
|
devServer.stdout.on('data', (data) => {
|
|
|
|
clearTimeout(sigkill);
|
|
|
|
sigkill = setTimeout(() => {
|
|
|
|
reject(`Dev server failed to start`);
|
|
|
|
}, 10000);
|
|
|
|
if (data.toString('utf8').includes('Server started')) resolve();
|
|
|
|
});
|
|
|
|
devServer.stderr.on('data', (data) => {
|
|
|
|
reject(data.toString('utf8'));
|
|
|
|
});
|
|
|
|
});
|
|
|
|
clearTimeout(sigkill); // done!
|
|
|
|
|
|
|
|
// send request to dev server that should be ready
|
|
|
|
const { statusCode, body } = (await fetch(`http://localhost:${port}`)) || {};
|
|
|
|
|
|
|
|
// test 3a: expect 200 status code
|
|
|
|
assert(statusCode, 200, `[${template}] 200 response`);
|
|
|
|
// test 3b: expect non-empty response
|
|
|
|
assert(body.length > 0, true, `[${template}] non-empty response`);
|
|
|
|
|
|
|
|
// clean up
|
|
|
|
devServer.kill();
|
2021-06-11 16:07:30 -05:00
|
|
|
}
|
2021-06-10 11:30:48 -05:00
|
|
|
|
2021-06-11 16:07:30 -05:00
|
|
|
async function testAll() {
|
2021-12-22 16:11:05 -05:00
|
|
|
// setup
|
|
|
|
await Promise.all(
|
|
|
|
TEMPLATES.map(async ({ value: template }) => {
|
|
|
|
// setup: `npm init astro`
|
|
|
|
await execa('../../create-astro.mjs', [template, '--template', template, '--commit', GITHUB_SHA, '--force-overwrite'], {
|
|
|
|
cwd: FIXTURES_DIR,
|
|
|
|
});
|
2022-03-08 16:46:11 -05:00
|
|
|
// setup: `pnpm install` (note: running multiple `pnpm`s in parallel in CI will conflict)
|
|
|
|
await execa('pnpm', ['install', '--no-package-lock', '--silent'], { cwd: path.join(FIXTURES_DIR, template) });
|
2021-12-22 16:11:05 -05:00
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
// test (note: not parallelized because Snowpack HMR reuses same port in dev)
|
|
|
|
for (let n = 0; n < TEMPLATES.length; n += 1) {
|
|
|
|
const template = TEMPLATES[n].value;
|
|
|
|
|
|
|
|
try {
|
|
|
|
await testTemplate(template);
|
|
|
|
} catch (err) {
|
|
|
|
console.error(red(`✘ [${template}]`));
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
|
|
|
|
console.info(green(`✔ [${template}] All tests passed (${n + 1}/${TEMPLATES.length})`));
|
|
|
|
}
|
2021-05-24 17:18:56 -05:00
|
|
|
}
|
2021-06-11 16:07:30 -05:00
|
|
|
testAll();
|