stacker.news/lib/stacktrace.js

56 lines
2.0 KiB
JavaScript

import { SourceMapConsumer } from 'source-map'
// FUN@FILE:LINE:COLUMN
const STACK_TRACE_LINE_REGEX = /^([A-Za-z0-9]*)@(.*):([0-9]+):([0-9]+)/
/**
* Decode a minified stack trace using source maps
* @param {string} stack - the minified stack trace
* @param {Object<string, SourceMapConsumer>} [sourceMaps] - an object used to cache source maps
* @returns {Promise<string>} Decoded stack trace
*/
export async function decodeMinifiedStackTrace (stack, sourceMaps = {}) {
let decodedStack = ''
let decoded = false
for (const line of stack.split('\n')) {
try {
const stackLine = line.trim()
const stackLineParts = stackLine?.match(STACK_TRACE_LINE_REGEX)
if (stackLineParts) {
const [stackFile, stackLine, stackColumn] = stackLineParts.slice(2)
if (!stackFile || !stackLine || !stackColumn) throw new Error('Unsupported stack line')
if (
(
!stackFile.startsWith(process.env.NEXT_PUBLIC_ASSET_PREFIX) &&
!stackFile.startsWith(process.env.NEXT_PUBLIC_URL)
) ||
!stackFile.endsWith('.js')
) throw new Error('Unsupported file url ' + stackFile)
const sourceMapUrl = stackFile + '.map'
if (!sourceMaps[sourceMapUrl]) {
sourceMaps[sourceMapUrl] = await new SourceMapConsumer(await fetch(sourceMapUrl).then(res => res.text()))
}
const sourceMapper = sourceMaps[sourceMapUrl]
const map = sourceMapper.originalPositionFor({
line: parseInt(stackLine),
column: parseInt(stackColumn)
})
const { source, name, line, column } = map
if (!source || line === undefined) throw new Error('Unsupported stack line')
decodedStack += `${name || ''}@${source}:${line}:${column}\n`
decoded = true
continue
}
} catch (e) {
console.error('Cannot decode stack line', e)
}
decodedStack += `${line}\n`
}
if (decoded) {
decodedStack = `Decoded stacktrace:\n${decodedStack}\n\nOriginal stack trace:\n${stack}`
}
return decodedStack
}