|
|
|
import * as fs from 'fs';
|
|
|
|
import * as path from 'path';
|
|
|
|
import glob from 'tiny-glob/sync.js';
|
|
|
|
|
|
|
|
import {
|
|
|
|
assert,
|
|
|
|
showOutput,
|
|
|
|
loadConfig,
|
|
|
|
loadSvelte,
|
|
|
|
setupHtmlEqual,
|
|
|
|
tryToLoadJson,
|
|
|
|
cleanRequireCache,
|
|
|
|
shouldUpdateExpected,
|
|
|
|
mkdirp
|
|
|
|
} from '../helpers';
|
|
|
|
import { set_current_component } from '../../internal';
|
|
|
|
|
|
|
|
function tryToReadFile(file) {
|
|
|
|
try {
|
|
|
|
return fs.readFileSync(file, 'utf-8');
|
|
|
|
} catch (err) {
|
|
|
|
if (err.code !== 'ENOENT') throw err;
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const sveltePath = process.cwd().split('\\').join('/');
|
|
|
|
let compile = null;
|
|
|
|
|
|
|
|
describe('ssr', () => {
|
|
|
|
before(() => {
|
|
|
|
compile = loadSvelte(true).compile;
|
|
|
|
|
|
|
|
return setupHtmlEqual();
|
|
|
|
});
|
|
|
|
|
|
|
|
let saved_window;
|
|
|
|
before(() => saved_window = global.window);
|
|
|
|
after(() => global.window = saved_window);
|
|
|
|
|
|
|
|
fs.readdirSync(`${__dirname}/samples`).forEach(dir => {
|
|
|
|
if (dir[0] === '.') return;
|
|
|
|
|
|
|
|
const config = loadConfig(`${__dirname}/samples/${dir}/_config.js`);
|
|
|
|
|
|
|
|
// add .solo to a sample directory name to only run that test, or
|
|
|
|
// .show to always show the output. or both
|
|
|
|
const solo = config.solo || /\.solo/.test(dir);
|
|
|
|
const show = /\.show/.test(dir);
|
|
|
|
|
|
|
|
if (solo && process.env.CI) {
|
|
|
|
throw new Error('Forgot to remove `solo: true` from test');
|
|
|
|
}
|
|
|
|
|
|
|
|
(solo ? it.only : it)(dir, (done) => {
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
|
|
|
dir = path.resolve(`${__dirname}/samples`, dir);
|
|
|
|
|
|
|
|
cleanRequireCache();
|
|
|
|
|
|
|
|
const compileOptions = {
|
|
|
|
sveltePath,
|
|
|
|
...config.compileOptions,
|
|
|
|
generate: 'ssr',
|
|
|
|
format: 'cjs'
|
|
|
|
};
|
|
|
|
|
|
|
|
require('../../register')(compileOptions);
|
|
|
|
|
|
|
|
const Component = require(`${dir}/main.svelte`).default;
|
|
|
|
|
|
|
|
const expectedHtml = tryToReadFile(`${dir}/_expected.html`);
|
|
|
|
const expectedCss = tryToReadFile(`${dir}/_expected.css`) || '';
|
|
|
|
|
|
|
|
const props = tryToLoadJson(`${dir}/data.json`) || undefined;
|
|
|
|
|
|
|
|
const rendered = Component.render(props);
|
|
|
|
const { html, css, head } = rendered;
|
|
|
|
|
|
|
|
fs.writeFileSync(`${dir}/_actual.html`, html);
|
|
|
|
if (css.code) fs.writeFileSync(`${dir}/_actual.css`, css.code);
|
|
|
|
|
|
|
|
try {
|
|
|
|
if (config.withoutNormalizeHtml) {
|
|
|
|
assert.strictEqual(html.trim().replace(/\r\n/g, '\n'), expectedHtml.trim().replace(/\r\n/g, '\n'));
|
|
|
|
} else {
|
|
|
|
(compileOptions.preserveComments
|
|
|
|
? assert.htmlEqualWithComments
|
|
|
|
: assert.htmlEqual)(html, expectedHtml);
|
|
|
|
}
|
|
|
|
} catch (error) {
|
|
|
|
if (shouldUpdateExpected()) {
|
|
|
|
fs.writeFileSync(`${dir}/_expected.html`, html);
|
|
|
|
console.log(`Updated ${dir}/_expected.html.`);
|
|
|
|
} else {
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
assert.equal(
|
|
|
|
css.code.replace(/^\s+/gm, '').replace(/[\r\n]/g, ''),
|
|
|
|
expectedCss.replace(/^\s+/gm, '').replace(/[\r\n]/g, '')
|
|
|
|
);
|
|
|
|
} catch (error) {
|
|
|
|
if (shouldUpdateExpected()) {
|
|
|
|
fs.writeFileSync(`${dir}/_expected.css`, css.code);
|
|
|
|
console.log(`Updated ${dir}/_expected.css.`);
|
|
|
|
} else {
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fs.existsSync(`${dir}/_expected-head.html`)) {
|
|
|
|
fs.writeFileSync(`${dir}/_actual-head.html`, head);
|
|
|
|
|
|
|
|
try {
|
|
|
|
assert.htmlEqual(
|
|
|
|
head,
|
|
|
|
fs.readFileSync(`${dir}/_expected-head.html`, 'utf-8')
|
|
|
|
);
|
|
|
|
} catch (error) {
|
|
|
|
if (shouldUpdateExpected()) {
|
|
|
|
fs.writeFileSync(`${dir}/_expected-head.html`, head);
|
|
|
|
console.log(`Updated ${dir}/_expected-head.html.`);
|
|
|
|
} else {
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (show) showOutput(dir, { generate: 'ssr', format: 'cjs' });
|
|
|
|
done();
|
|
|
|
} catch (err) {
|
|
|
|
showOutput(dir, { generate: 'ssr', format: 'cjs' });
|
|
|
|
err.stack += `\n\ncmd-click: ${path.relative(process.cwd(), dir)}/main.svelte`;
|
|
|
|
done(err);
|
|
|
|
} finally {
|
|
|
|
set_current_component(null);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// duplicate client-side tests, as far as possible
|
|
|
|
runRuntimeSamples('runtime');
|
|
|
|
runRuntimeSamples('runtime-puppeteer');
|
|
|
|
|
|
|
|
function runRuntimeSamples(suite) {
|
|
|
|
fs.readdirSync(`test/${suite}/samples`).forEach(dir => {
|
|
|
|
if (dir[0] === '.') return;
|
|
|
|
|
|
|
|
const config = loadConfig(`./${suite}/samples/${dir}/_config.js`);
|
|
|
|
const solo = config.solo || /\.solo/.test(dir);
|
|
|
|
|
|
|
|
if (solo && process.env.CI) {
|
|
|
|
throw new Error('Forgot to remove `solo: true` from test');
|
|
|
|
}
|
|
|
|
|
|
|
|
if (config.skip_if_ssr) return;
|
|
|
|
|
|
|
|
(config.skip ? it.skip : solo ? it.only : it)(dir, () => {
|
|
|
|
const cwd = path.resolve(`test/${suite}/samples`, dir);
|
|
|
|
|
|
|
|
cleanRequireCache();
|
|
|
|
|
|
|
|
delete global.window;
|
|
|
|
|
|
|
|
const compileOptions = {
|
|
|
|
sveltePath,
|
|
|
|
...config.compileOptions,
|
|
|
|
generate: 'ssr',
|
|
|
|
format: 'cjs'
|
|
|
|
};
|
|
|
|
|
|
|
|
require('../../register')(compileOptions);
|
|
|
|
|
|
|
|
glob('**/*.svelte', { cwd }).forEach(file => {
|
|
|
|
if (file[0] === '_') return;
|
|
|
|
|
|
|
|
const dir = `${cwd}/_output/ssr`;
|
|
|
|
const out = `${dir}/${file.replace(/\.svelte$/, '.js')}`;
|
|
|
|
|
|
|
|
if (fs.existsSync(out)) {
|
|
|
|
fs.unlinkSync(out);
|
|
|
|
}
|
|
|
|
|
|
|
|
mkdirp(dir);
|
|
|
|
|
|
|
|
try {
|
|
|
|
const { js } = compile(
|
|
|
|
fs.readFileSync(`${cwd}/${file}`, 'utf-8'),
|
|
|
|
{
|
|
|
|
...compileOptions,
|
|
|
|
filename: file
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
fs.writeFileSync(out, js.code);
|
|
|
|
} catch (err) {
|
|
|
|
// do nothing
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
try {
|
|
|
|
if (config.before_test) config.before_test();
|
|
|
|
|
|
|
|
const Component = require(`../${suite}/samples/${dir}/main.svelte`).default;
|
|
|
|
const { html } = Component.render(config.props, {
|
|
|
|
store: (config.store !== true) && config.store
|
|
|
|
});
|
|
|
|
|
|
|
|
if (config.ssrHtml) {
|
|
|
|
assert.htmlEqual(html, config.ssrHtml);
|
|
|
|
} else if (config.html) {
|
|
|
|
assert.htmlEqual(html, config.html);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (config.test_ssr) {
|
|
|
|
config.test_ssr({ assert });
|
|
|
|
}
|
|
|
|
|
|
|
|
if (config.after_test) config.after_test();
|
|
|
|
|
|
|
|
if (config.show) {
|
|
|
|
showOutput(cwd, compileOptions);
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
err.stack += `\n\ncmd-click: ${path.relative(process.cwd(), cwd)}/main.svelte`;
|
|
|
|
|
|
|
|
if (config.error) {
|
|
|
|
if (typeof config.error === 'function') {
|
|
|
|
config.error(assert, err);
|
|
|
|
} else {
|
|
|
|
assert.equal(err.message, config.error);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
showOutput(cwd, compileOptions);
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
} finally {
|
|
|
|
set_current_component(null);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|