From c3180654fbad1636dc40a1fa0d8cca42d5311cdb Mon Sep 17 00:00:00 2001 From: fi3ework Date: Tue, 18 Apr 2023 03:51:06 +0800 Subject: [PATCH] fix(ssr): stacktrace uses abs path with or without sourcemap --- .../vite/src/node/server/transformRequest.ts | 8 ++++ packages/vite/src/node/ssr/ssrTransform.ts | 7 ++- .../__tests__/css-sourcemap.spec.ts | 1 + .../__tests__/js-sourcemap.spec.ts | 1 + .../ssr-html/__tests__/ssr-html.spec.ts | 45 ++++++++++++------- playground/ssr-html/src/error.ts | 3 ++ playground/ssr-html/test-stacktrace.js | 29 +++++++----- playground/test-utils.ts | 3 ++ 8 files changed, 69 insertions(+), 28 deletions(-) create mode 100644 playground/ssr-html/src/error.ts diff --git a/packages/vite/src/node/server/transformRequest.ts b/packages/vite/src/node/server/transformRequest.ts index 2b74b25b526f06..0bab9703c075c7 100644 --- a/packages/vite/src/node/server/transformRequest.ts +++ b/packages/vite/src/node/server/transformRequest.ts @@ -295,6 +295,7 @@ async function loadAndTransform( // to resolve and display them in a meaningful way (rather than // with absolute paths). if (path.isAbsolute(sourcePath)) { + map.sourceRoot = path.dirname(mod.file) + path.sep map.sources[sourcesIndex] = path.relative( path.dirname(mod.file), sourcePath, @@ -305,6 +306,13 @@ async function loadAndTransform( } } + // no sourcemap for raw js source file + if (!map && mod.file) { + map = { + sources: [mod.file], + } as SourceMap + } + const result = ssr && !server.config.experimental.skipSsrTransform ? await server.ssrTransform(code, map as SourceMap, url, originalCode) diff --git a/packages/vite/src/node/ssr/ssrTransform.ts b/packages/vite/src/node/ssr/ssrTransform.ts index 96ee465e2a0ebe..0c787f51381de0 100644 --- a/packages/vite/src/node/ssr/ssrTransform.ts +++ b/packages/vite/src/node/ssr/ssrTransform.ts @@ -274,13 +274,18 @@ async function ssrTransformScript( ...map, sources: inMap.sources, sourcesContent: inMap.sourcesContent, + // sourceRoot: inMap.sourceRoot, } as RawSourceMap, inMap as RawSourceMap, ], false, ) as SourceMap } else { - map.sources = [url] + if (inMap?.sources) { + map.sources = inMap.sources + } else { + map.sources = [url] + } // needs to use originalCode instead of code // because code might be already transformed even if map is null map.sourcesContent = [originalCode] diff --git a/playground/css-sourcemap/__tests__/css-sourcemap.spec.ts b/playground/css-sourcemap/__tests__/css-sourcemap.spec.ts index 696864b12ffd64..5be3c0a23b3e24 100644 --- a/playground/css-sourcemap/__tests__/css-sourcemap.spec.ts +++ b/playground/css-sourcemap/__tests__/css-sourcemap.spec.ts @@ -69,6 +69,7 @@ describe.runIf(isServe)('serve', () => { expect(formatSourcemapForSnapshot(map)).toMatchInlineSnapshot(` { "mappings": "AAAA;EACE,UAAU;AACZ;;ACAA;EACE,UAAU;AACZ", + "sourceRoot": "/root/", "sources": [ "be-imported.css", "linked-with-import.css", diff --git a/playground/js-sourcemap/__tests__/js-sourcemap.spec.ts b/playground/js-sourcemap/__tests__/js-sourcemap.spec.ts index e04a7c36126d2f..b86af9df81aeca 100644 --- a/playground/js-sourcemap/__tests__/js-sourcemap.spec.ts +++ b/playground/js-sourcemap/__tests__/js-sourcemap.spec.ts @@ -24,6 +24,7 @@ if (!isBuild) { expect(formatSourcemapForSnapshot(map)).toMatchInlineSnapshot(` { "mappings": "AAAO,aAAM,MAAM;", + "sourceRoot": "/root/", "sources": [ "bar.ts", ], diff --git a/playground/ssr-html/__tests__/ssr-html.spec.ts b/playground/ssr-html/__tests__/ssr-html.spec.ts index 2b9da5cadd01c0..2bb082aa7407c2 100644 --- a/playground/ssr-html/__tests__/ssr-html.spec.ts +++ b/playground/ssr-html/__tests__/ssr-html.spec.ts @@ -62,23 +62,34 @@ describe.runIf(isServe)('hmr', () => { describe.runIf(isServe)('stacktrace', () => { const execFileAsync = promisify(execFile) - for (const sourcemapsEnabled of [false, true]) { - test(`stacktrace is correct when sourcemaps is${ - sourcemapsEnabled ? '' : ' not' - } enabled in Node.js`, async () => { - const testStacktraceFile = path.resolve( - __dirname, - '../test-stacktrace.js', - ) + for (const ext of ['js', 'ts']) { + for (const sourcemapsEnabled of [false, true]) { + test(`stacktrace of ${ext} is correct when sourcemaps is${ + sourcemapsEnabled ? '' : ' not' + } enabled in Node.js`, async () => { + const testStacktraceFile = path.resolve( + __dirname, + '../test-stacktrace.js', + ) - const p = await execFileAsync('node', [ - testStacktraceFile, - '' + sourcemapsEnabled, - ]) - const line = p.stdout - .split('\n') - .find((line) => line.includes('Module.error')) - expect(line.trim()).toMatch(/[\\/]src[\\/]error\.js:2:9/) - }) + const p = await execFileAsync('node', [ + testStacktraceFile, + '' + sourcemapsEnabled, + ext, + ]) + const lines = p.stdout + .split('\n') + .filter((line) => line.includes('Module.error')) + + const reg = new RegExp( + // TODO: ts without sourcemaps will resolve column to 8 which should be 9 + path.resolve(__dirname, '../src') + '/error\\.' + ext + ':2:[89]', + ) + + lines.forEach((line) => { + expect(line.trim()).toMatch(reg) + }) + }) + } } }) diff --git a/playground/ssr-html/src/error.ts b/playground/ssr-html/src/error.ts new file mode 100644 index 00000000000000..fe8eeb20af8f8a --- /dev/null +++ b/playground/ssr-html/src/error.ts @@ -0,0 +1,3 @@ +export function error() { + throw new Error('e') +} diff --git a/playground/ssr-html/test-stacktrace.js b/playground/ssr-html/test-stacktrace.js index c3ce5e56736799..327a8b4c423dae 100644 --- a/playground/ssr-html/test-stacktrace.js +++ b/playground/ssr-html/test-stacktrace.js @@ -3,8 +3,10 @@ import { fileURLToPath } from 'node:url' import { createServer } from 'vite' const isSourceMapEnabled = process.argv[2] === 'true' +const ext = process.argv[3] process.setSourceMapsEnabled(isSourceMapEnabled) console.log('# sourcemaps enabled:', isSourceMapEnabled) +console.log('# source file extension:', ext) const version = (() => { const m = process.version.match(/^v(\d+)\.(\d+)\.\d+$/) @@ -31,17 +33,24 @@ const vite = await createServer({ appType: 'custom', }) -const mod = await vite.ssrLoadModule('/src/error.js') -try { - mod.error() -} catch (e) { - // this should not be called - // when sourcemap support for `new Function` is supported and sourcemap is enabled - // because the stacktrace is already rewritten by Node.js - if (!(isSourceMapEnabled && isFunctionSourceMapSupported)) { - vite.ssrFixStacktrace(e) +const dir = path.dirname(fileURLToPath(import.meta.url)) + +const abs1 = await vite.ssrLoadModule(`/src/error.${ext}`) +const abs2 = await vite.ssrLoadModule(path.resolve(dir, `./src/error.${ext}`)) +const relative = await vite.ssrLoadModule(`./src/error.${ext}`) + +for (const mod of [abs1, abs2, relative]) { + try { + mod.error() + } catch (e) { + // this should not be called + // when sourcemap support for `new Function` is supported and sourcemap is enabled + // because the stacktrace is already rewritten by Node.js + if (!(isSourceMapEnabled && isFunctionSourceMapSupported)) { + vite.ssrFixStacktrace(e) + } + console.log(e) } - console.log(e) } await vite.close() diff --git a/playground/test-utils.ts b/playground/test-utils.ts index 6d3489ad66dd03..9bfbb2d550851b 100644 --- a/playground/test-utils.ts +++ b/playground/test-utils.ts @@ -307,6 +307,9 @@ export const formatSourcemapForSnapshot = (map: any): any => { delete m.file delete m.names m.sources = m.sources.map((source) => source.replace(root, '/root')) + if (m.sourceRoot) { + m.sourceRoot = m.sourceRoot.replace(root + path.sep, '/root/') + } return m }