You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
svelte/test/sourcemaps/sourcemaps.test.js

111 lines
3.6 KiB

import * as fs from 'node:fs';
import * as path from 'node:path';
import * as svelte from '../../src/compiler/index.js';
import { try_load_config } from '../helpers.js';
import { describe, assert, it } from 'vitest';
// keep source-map at version 0.7.x
// https://github.com/mozilla/source-map/issues/400
import { getLocator } from 'locate-character';
import { SourceMapConsumer } from 'source-map';
describe('sourcemaps', async () => {
await Promise.all(fs.readdirSync(`${__dirname}/samples`).map((dir) => run(dir)));
async function run(dir) {
if (dir[0] === '.') return;
const config = await try_load_config(`${__dirname}/samples/${dir}/_config.js`);
// add .solo to a sample directory name to only run that test
const solo = config.solo || /\.solo/.test(dir);
const skip = config.skip || /\.skip/.test(dir);
const it_fn = solo ? it.only : skip ? it.skip : it;
it_fn(dir, async () => {
const { test } = await import(`./samples/${dir}/test.js`);
const inputFile = path.resolve(`${__dirname}/samples/${dir}/input.svelte`);
const outputName = '_actual';
const outputBase = path.resolve(`${__dirname}/samples/${dir}/${outputName}`);
const inputCode = fs.readFileSync(inputFile, 'utf-8');
const input = {
code: inputCode,
locate: getLocator(inputCode),
locate_1: getLocator(inputCode, { offsetLine: 1 })
};
const preprocessed = await svelte.preprocess(
input.code,
config.preprocess || {},
config.options || {
filename: 'input.svelte'
}
);
const { js, css } = svelte.compile(preprocessed.code, {
filename: 'input.svelte',
// filenames for sourcemaps
sourcemap: preprocessed.map,
outputFilename: `${outputName}.js`,
cssOutputFilename: `${outputName}.css`,
...(config.compile_options || {})
});
js.code = js.code.replace(/generated by Svelte v\d+\.\d+\.\d+/, (match) =>
match.replace(/\d/g, 'x')
);
fs.writeFileSync(`${outputBase}.svelte`, preprocessed.code);
if (preprocessed.map) {
fs.writeFileSync(
`${outputBase}.svelte.map`,
// TODO encode mappings for output - svelte.preprocess returns decoded mappings
JSON.stringify(preprocessed.map, null, 2)
);
}
fs.writeFileSync(`${outputBase}.js`, `${js.code}\n//# sourceMappingURL=${outputName}.js.map`);
fs.writeFileSync(`${outputBase}.js.map`, JSON.stringify(js.map, null, 2));
if (css.code) {
fs.writeFileSync(
`${outputBase}.css`,
`${css.code}\n/*# sourceMappingURL=${outputName}.css.map */`
);
fs.writeFileSync(`${outputBase}.css.map`, JSON.stringify(css.map, null, ' '));
}
if (js.map) {
assert.deepEqual(
js.map.sources.slice().sort(),
(config.js_map_sources || ['input.svelte']).sort(),
'js.map.sources is wrong'
);
}
if (css.map) {
assert.deepEqual(
css.map.sources.slice().sort(),
(config.css_map_sources || ['input.svelte']).sort(),
'css.map.sources is wrong'
);
}
// use locate_1 with mapConsumer:
// lines are one-based, columns are zero-based
preprocessed.mapConsumer =
preprocessed.map && (await new SourceMapConsumer(preprocessed.map));
preprocessed.locate = getLocator(preprocessed.code);
preprocessed.locate_1 = getLocator(preprocessed.code, { offsetLine: 1 });
js.mapConsumer = js.map && (await new SourceMapConsumer(js.map));
js.locate = getLocator(js.code);
js.locate_1 = getLocator(js.code, { offsetLine: 1 });
css.mapConsumer = css.map && (await new SourceMapConsumer(css.map));
css.locate = getLocator(css.code || '');
css.locate_1 = getLocator(css.code || '', { offsetLine: 1 });
await test({ assert, input, preprocessed, js, css });
});
}
});