|
|
|
// @vitest-environment happy-dom
|
|
|
|
|
|
|
|
import * as fs from 'node:fs';
|
|
|
|
import { assert, describe, it } from 'vitest';
|
|
|
|
import * as svelte from 'svelte/compiler';
|
|
|
|
import { create_loader, should_update_expected, try_load_config } from '../helpers.js';
|
|
|
|
import { assert_html_equal } from '../html_equal.js';
|
|
|
|
|
|
|
|
function normalize_warning(warning) {
|
|
|
|
warning.frame = warning.frame.replace(/^\n/, '').replace(/^\t+/gm, '').replace(/\s+$/gm, '');
|
|
|
|
delete warning.filename;
|
|
|
|
delete warning.toString;
|
|
|
|
return warning;
|
|
|
|
}
|
|
|
|
|
|
|
|
describe('css', () => {
|
|
|
|
fs.readdirSync(`${__dirname}/samples`).forEach((dir) => {
|
|
|
|
if (dir[0] === '.') return;
|
|
|
|
|
|
|
|
// add .solo to a sample directory name to only run that test
|
|
|
|
const solo = /\.solo/.test(dir);
|
|
|
|
const skip = /\.skip/.test(dir);
|
|
|
|
|
|
|
|
const it_fn = solo ? it.only : skip ? it.skip : it;
|
|
|
|
|
|
|
|
it_fn(dir, async () => {
|
|
|
|
const cwd = `${__dirname}/samples/${dir}`;
|
|
|
|
const config = await try_load_config(`${cwd}/_config.js`);
|
|
|
|
|
|
|
|
const input = fs
|
|
|
|
.readFileSync(`${cwd}/input.svelte`, 'utf-8')
|
|
|
|
.replace(/\s+$/, '')
|
|
|
|
.replace(/\r/g, '');
|
|
|
|
|
|
|
|
const expected_warnings = (config.warnings || []).map(normalize_warning);
|
|
|
|
|
|
|
|
const dom = svelte.compile(input, Object.assign({}, config.compileOptions || {}));
|
|
|
|
|
|
|
|
const ssr = svelte.compile(input, Object.assign({}, config.compileOptions || {}));
|
|
|
|
|
|
|
|
assert.equal(dom.css.code, ssr.css.code);
|
|
|
|
|
|
|
|
const dom_warnings = dom.warnings.map(normalize_warning);
|
|
|
|
const ssr_warnings = ssr.warnings.map(normalize_warning);
|
|
|
|
|
|
|
|
assert.deepEqual(dom_warnings, ssr_warnings);
|
|
|
|
assert.deepEqual(dom_warnings.map(normalize_warning), expected_warnings);
|
|
|
|
|
|
|
|
fs.writeFileSync(`${cwd}/_actual.css`, dom.css.code);
|
|
|
|
const expected = {
|
|
|
|
html: read(`${cwd}/expected.html`),
|
|
|
|
css: read(`${cwd}/expected.css`)
|
|
|
|
};
|
|
|
|
|
|
|
|
const actual_css = replace_css_hash(dom.css.code);
|
|
|
|
try {
|
|
|
|
assert.equal(actual_css, expected.css);
|
|
|
|
} catch (error) {
|
|
|
|
if (should_update_expected()) {
|
|
|
|
fs.writeFileSync(`${cwd}/expected.css`, actual_css);
|
|
|
|
console.log(`Updated ${dir}/expected.css.`);
|
|
|
|
} else {
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let ClientComponent;
|
|
|
|
let ServerComponent;
|
|
|
|
|
|
|
|
// we do this here, rather than in the expected.html !== null
|
|
|
|
// block, to verify that valid code was generated
|
|
|
|
const load = create_loader({ ...(config.compileOptions || {}) }, cwd);
|
|
|
|
try {
|
|
|
|
ClientComponent = (await load('input.svelte')).default;
|
|
|
|
} catch (err) {
|
|
|
|
console.log(dom.js.code);
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
|
|
|
|
const load_ssr = create_loader({ ...(config.compileOptions || {}), generate: 'ssr' }, cwd);
|
|
|
|
try {
|
|
|
|
ServerComponent = (await load_ssr('input.svelte')).default;
|
|
|
|
} catch (err) {
|
|
|
|
console.log(dom.js.code);
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
|
|
|
|
// verify that the right elements have scoping selectors
|
|
|
|
if (expected.html !== null) {
|
|
|
|
const target = window.document.createElement('main');
|
|
|
|
|
|
|
|
new ClientComponent({ target, props: config.props });
|
|
|
|
const html = target.innerHTML;
|
|
|
|
|
|
|
|
fs.writeFileSync(`${cwd}/_actual.html`, html);
|
|
|
|
|
|
|
|
const actual_html = replace_css_hash(html);
|
|
|
|
assert_html_equal(actual_html, expected.html);
|
|
|
|
|
|
|
|
window.document.head.innerHTML = ''; // remove added styles
|
|
|
|
|
|
|
|
const actual_ssr = replace_css_hash(ServerComponent.render(config.props).html);
|
|
|
|
assert_html_equal(actual_ssr, expected.html);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
function replace_css_hash(str) {
|
|
|
|
return str.replace(/svelte-[a-z0-9]+/g, 'svelte-xyz');
|
|
|
|
}
|
|
|
|
|
|
|
|
function read(file) {
|
|
|
|
try {
|
|
|
|
return fs.readFileSync(file, 'utf-8');
|
|
|
|
} catch (err) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
}
|