Skip to content

Commit a199cdf

Browse files
committed
added typescript types to get-user-code-frame file
1 parent 56a4c75 commit a199cdf

File tree

3 files changed

+64
-65
lines changed

3 files changed

+64
-65
lines changed

package.json

+1
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,7 @@
5050
},
5151
"devDependencies": {
5252
"@testing-library/jest-dom": "^5.11.6",
53+
"@types/babel__code-frame": "^7.0.2",
5354
"jest-in-case": "^1.0.2",
5455
"jest-serializer-ansi": "^1.0.3",
5556
"jest-watch-select-projects": "^2.0.0",

src/get-user-code-frame.js

-65
This file was deleted.

src/get-user-code-frame.ts

+63
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,63 @@
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

Comments
 (0)