|
| 1 | +// type-only imports get erased at runtime |
| 2 | +import type FS from 'fs' |
| 3 | +import type {Chalk} from 'chalk' |
| 4 | +import type {codeFrameColumns as CodeFrameColumnsFn} from '@babel/code-frame' |
| 5 | + |
| 6 | +// frame has the form "at myMethod (location/to/my/file.js:10:2)" |
| 7 | +function getCodeFrame(frame: string) { |
| 8 | + const locationStart = frame.indexOf('(') + 1 |
| 9 | + const locationEnd = frame.indexOf(')') |
| 10 | + const frameLocation = frame.slice(locationStart, locationEnd) |
| 11 | + |
| 12 | + const frameLocationElements = frameLocation.split(':') |
| 13 | + const [filename, line, column] = [ |
| 14 | + frameLocationElements[0], |
| 15 | + parseInt(frameLocationElements[1], 10), |
| 16 | + parseInt(frameLocationElements[2], 10), |
| 17 | + ] |
| 18 | + |
| 19 | + // use require() instead of a top-level import for Node dependencies |
| 20 | + // so that errors thrown when called in a browser environment can be caught and handled. |
| 21 | + // expect to throw errors here and catch them in `getUserCodeFrame`. |
| 22 | + |
| 23 | + const {readFileSync} = module.require('fs') as typeof FS |
| 24 | + const rawFileContents = readFileSync(filename, 'utf-8') |
| 25 | + |
| 26 | + const {codeFrameColumns} = module.require('@babel/code-frame') as { |
| 27 | + codeFrameColumns: typeof CodeFrameColumnsFn |
| 28 | + } |
| 29 | + const codeFrame = codeFrameColumns( |
| 30 | + rawFileContents, |
| 31 | + { |
| 32 | + start: {line, column}, |
| 33 | + }, |
| 34 | + { |
| 35 | + highlightCode: true, |
| 36 | + linesBelow: 0, |
| 37 | + }, |
| 38 | + ) |
| 39 | + |
| 40 | + const chalk = module.require('chalk') as Chalk |
| 41 | + |
| 42 | + return `${chalk.dim(frameLocation)}\n${codeFrame}\n` |
| 43 | +} |
| 44 | + |
| 45 | +function getUserCodeFrame(): string { |
| 46 | + try { |
| 47 | + const err = new Error() |
| 48 | + const firstClientCodeFrame = err.stack |
| 49 | + ?.split('\n') |
| 50 | + .slice(1) // Remove first line which has the form "Error: TypeError" |
| 51 | + .find(frame => !frame.includes('node_modules/')) // Ignore frames from 3rd party libraries |
| 52 | + |
| 53 | + /* istanbul ignore next */ |
| 54 | + if (!firstClientCodeFrame) return '' |
| 55 | + return getCodeFrame(firstClientCodeFrame) |
| 56 | + } catch { |
| 57 | + // Expect to throw and catch errors when in the browser |
| 58 | + // If we couldn't load dependencies, we can't generate the user trace |
| 59 | + return '' |
| 60 | + } |
| 61 | +} |
| 62 | + |
| 63 | +export {getUserCodeFrame} |
0 commit comments