chrome-devtools-frontend
Version:
Chrome DevTools UI
873 lines (767 loc) • 33.4 kB
text/typescript
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import * as TextUtils from '../../models/text_utils/text_utils.js';
import * as ScopesCodec from '../../third_party/source-map-scopes-codec/source-map-scopes-codec.js';
import * as Common from '../common/common.js';
import * as Platform from '../platform/platform.js';
import * as Root from '../root/root.js';
import type {CallFrame, ScopeChainEntry} from './DebuggerModel.js';
import {scopeTreeForScript} from './ScopeTreeCache.js';
import type {Script} from './Script.js';
import {buildOriginalScopes, decodePastaRanges, type NamedFunctionRange} from './SourceMapFunctionRanges.js';
import {SourceMapScopesInfo, type TranslatedFrame} from './SourceMapScopesInfo.js';
/**
* Type of the base source map JSON object, which contains the sources and the mappings at the very least, plus
* some additional fields.
*
* @see {@link SourceMapV3}
* @see {@link https://docs.google.com/document/d/1U1RGAehQwRypUTovF1KRlpiOFze0b-_2gc6fAH0KY0k Source Map Revision 3 Proposal}
*/
export interface SourceMapV3Object {
/* eslint-disable @typescript-eslint/naming-convention */
version: number;
sources: string[];
mappings: string;
file?: string;
sourceRoot?: string;
sourcesContent?: Array<string|null>;
names?: string[];
ignoreList?: number[];
scopes?: string;
debugId?: string;
x_google_linecount?: number;
x_google_ignoreList?: number[];
x_com_bloomberg_sourcesFunctionMappings?: string[];
/* eslint-enable @typescript-eslint/naming-convention */
}
/**
* Type of JSON objects that classify as valid sourcemaps per version 3 of the specification.
*
* We support both possible formats, the traditional source map object (represented by the {@link SourceMapV3Object} type),
* as well as the index map format, which consists of a sequence of sections that each hold source maps objects themselves
* or URLs to external source map files.
*
* @see {@link SourceMapV3Object}
* @see {@link https://docs.google.com/document/d/1U1RGAehQwRypUTovF1KRlpiOFze0b-_2gc6fAH0KY0k Source Map Revision 3 Proposal}
*/
export type SourceMapV3 = SourceMapV3Object|{
// clang-format off
version: number,
sections: Array<{
offset: {line: number, column: number},
map: SourceMapV3Object,
} | {
offset: {line: number, column: number},
url: string,
}>,
file?: string,
// clang-format on
};
/**
* Parses the {@link content} as JSON, ignoring BOM markers in the beginning, and
* also handling the CORB bypass prefix correctly.
*
* @param content the string representation of a sourcemap.
* @returns the {@link SourceMapV3} representation of the {@link content}.
*/
export function parseSourceMap(content: string): SourceMapV3 {
if (content.startsWith(')]}')) {
content = content.substring(content.indexOf('\n'));
}
if (content.charCodeAt(0) === 0xFEFF) {
// Strip BOM at the beginning before parsing the JSON.
content = content.slice(1);
}
return JSON.parse(content) as SourceMapV3;
}
export type DebugId = Platform.Brand.Brand<string, 'DebugId'>;
export class SourceMapEntry {
readonly lineNumber: number;
readonly columnNumber: number;
readonly sourceIndex?: number;
readonly sourceURL: Platform.DevToolsPath.UrlString|undefined;
readonly sourceLineNumber: number;
readonly sourceColumnNumber: number;
readonly name: string|undefined;
constructor(
lineNumber: number, columnNumber: number, sourceIndex?: number, sourceURL?: Platform.DevToolsPath.UrlString,
sourceLineNumber?: number, sourceColumnNumber?: number, name?: string) {
this.lineNumber = lineNumber;
this.columnNumber = columnNumber;
this.sourceIndex = sourceIndex;
this.sourceURL = sourceURL;
this.sourceLineNumber = (sourceLineNumber as number);
this.sourceColumnNumber = (sourceColumnNumber as number);
this.name = name;
}
static compare(entry1: SourceMapEntry, entry2: SourceMapEntry): number {
if (entry1.lineNumber !== entry2.lineNumber) {
return entry1.lineNumber - entry2.lineNumber;
}
return entry1.columnNumber - entry2.columnNumber;
}
}
interface SourceInfo {
sourceURL: Platform.DevToolsPath.UrlString;
content: string|null;
ignoreListHint: boolean;
reverseMappings: number[]|null;
}
export class SourceMap {
static retainRawSourceMaps = false;
#json: SourceMapV3|null;
readonly #compiledURL: Platform.DevToolsPath.UrlString;
readonly #sourceMappingURL: Platform.DevToolsPath.UrlString;
readonly #baseURL: Platform.DevToolsPath.UrlString;
#mappings: SourceMapEntry[]|null = null;
readonly #sourceInfos: SourceInfo[] = [];
readonly #sourceInfoByURL = new Map<Platform.DevToolsPath.UrlString, SourceInfo>();
readonly #script?: Script;
#scopesInfo: SourceMapScopesInfo|null = null;
readonly #debugId?: DebugId;
scopesFallbackPromiseForTest?: Promise<unknown>;
/**
* Implements Source Map V3 model. See https://github.com/google/closure-compiler/wiki/Source-Maps
* for format description.
*/
constructor(
compiledURL: Platform.DevToolsPath.UrlString, sourceMappingURL: Platform.DevToolsPath.UrlString,
payload: SourceMapV3, script?: Script) {
this.#json = payload;
this.#script = script;
this.#compiledURL = compiledURL;
this.#sourceMappingURL = sourceMappingURL;
this.#baseURL = (Common.ParsedURL.schemeIs(sourceMappingURL, 'data:')) ? compiledURL : sourceMappingURL;
this.#debugId = 'debugId' in payload ? (payload.debugId as DebugId | undefined) : undefined;
if ('sections' in this.#json) {
if (this.#json.sections.find(section => 'url' in section)) {
Common.Console.Console.instance().warn(
`SourceMap "${sourceMappingURL}" contains unsupported "URL" field in one of its sections.`);
}
}
this.eachSection(this.parseSources.bind(this));
}
json(): SourceMapV3|null {
return this.#json;
}
augmentWithScopes(scriptUrl: Platform.DevToolsPath.UrlString, ranges: NamedFunctionRange[]): void {
this.#ensureSourceMapProcessed();
if (this.#json && this.#json.version > 3) {
throw new Error('Only support augmenting source maps up to version 3.');
}
// Ensure scriptUrl is associated with sourceMap sources
const sourceIdx = this.#sourceIndex(scriptUrl);
if (sourceIdx >= 0) {
if (!this.#scopesInfo) {
// First time seeing this sourcemap, create an new empty scopesInfo object
this.#scopesInfo = new SourceMapScopesInfo(this, {scopes: [], ranges: []});
}
if (!this.#scopesInfo.hasOriginalScopes(sourceIdx)) {
const originalScopes = buildOriginalScopes(ranges);
this.#scopesInfo.addOriginalScopesAtIndex(sourceIdx, originalScopes);
}
} else {
throw new Error(`Could not find sourceURL ${scriptUrl} in sourceMap`);
}
}
#sourceIndex(sourceURL: Platform.DevToolsPath.UrlString): number {
return this.#sourceInfos.findIndex(info => info.sourceURL === sourceURL);
}
compiledURL(): Platform.DevToolsPath.UrlString {
return this.#compiledURL;
}
url(): Platform.DevToolsPath.UrlString {
return this.#sourceMappingURL;
}
debugId(): DebugId|null {
return this.#debugId ?? null;
}
sourceURLForSourceIndex(index: number): Platform.DevToolsPath.UrlString|undefined {
return this.#sourceInfos[index]?.sourceURL;
}
sourceURLs(): Platform.DevToolsPath.UrlString[] {
return [...this.#sourceInfoByURL.keys()];
}
embeddedContentByURL(sourceURL: Platform.DevToolsPath.UrlString): string|null {
const entry = this.#sourceInfoByURL.get(sourceURL);
if (!entry) {
return null;
}
return entry.content;
}
hasScopeInfo(): boolean {
this.#ensureSourceMapProcessed();
return this.#scopesInfo !== null && !this.#scopesInfo.isEmpty();
}
findEntry(lineNumber: number, columnNumber: number, inlineFrameIndex?: number): SourceMapEntry|null {
this.#ensureSourceMapProcessed();
if (inlineFrameIndex && this.#scopesInfo !== null) {
// For inlineFrameIndex != 0 we use the callsite info for the corresponding inlining site.
// Note that the callsite for "inlineFrameIndex" is actually in the previous frame.
const {inlinedFunctions} = this.#scopesInfo.findInlinedFunctions(lineNumber, columnNumber);
const {callsite} = inlinedFunctions[inlineFrameIndex - 1];
if (!callsite) {
console.error('Malformed source map. Expected to have a callsite info for index', inlineFrameIndex);
return null;
}
return {
lineNumber,
columnNumber,
sourceIndex: callsite.sourceIndex,
sourceURL: this.sourceURLs()[callsite.sourceIndex],
sourceLineNumber: callsite.line,
sourceColumnNumber: callsite.column,
name: undefined,
};
}
const mappings = this.mappings();
const index = Platform.ArrayUtilities.upperBound(
mappings, undefined, (_, entry) => lineNumber - entry.lineNumber || columnNumber - entry.columnNumber);
return index ? mappings[index - 1] : null;
}
findEntryRanges(lineNumber: number, columnNumber: number): {
range: TextUtils.TextRange.TextRange,
sourceRange: TextUtils.TextRange.TextRange,
sourceURL: Platform.DevToolsPath.UrlString,
}|null {
const mappings = this.mappings();
const endIndex = Platform.ArrayUtilities.upperBound(
mappings, undefined, (_, entry) => lineNumber - entry.lineNumber || columnNumber - entry.columnNumber);
if (!endIndex) {
// If the line and column are preceding all the entries, then there is nothing to map.
return null;
}
// startIndex must be within mappings range because endIndex must be not falsy
const startIndex = endIndex - 1;
const sourceURL = mappings[startIndex].sourceURL;
if (!sourceURL) {
return null;
}
// Let us compute the range that contains the source position in the compiled code.
const endLine = endIndex < mappings.length ? mappings[endIndex].lineNumber : 2 ** 31 - 1;
const endColumn = endIndex < mappings.length ? mappings[endIndex].columnNumber : 2 ** 31 - 1;
const range = new TextUtils.TextRange.TextRange(
mappings[startIndex].lineNumber, mappings[startIndex].columnNumber, endLine, endColumn);
// Now try to find the corresponding token in the original code.
const reverseMappings = this.reversedMappings(sourceURL);
const startSourceLine = mappings[startIndex].sourceLineNumber;
const startSourceColumn = mappings[startIndex].sourceColumnNumber;
const endReverseIndex = Platform.ArrayUtilities.upperBound(
reverseMappings, undefined,
(_, i) => startSourceLine - mappings[i].sourceLineNumber || startSourceColumn - mappings[i].sourceColumnNumber);
if (!endReverseIndex) {
return null;
}
const endSourceLine = endReverseIndex < reverseMappings.length ?
mappings[reverseMappings[endReverseIndex]].sourceLineNumber :
2 ** 31 - 1;
const endSourceColumn = endReverseIndex < reverseMappings.length ?
mappings[reverseMappings[endReverseIndex]].sourceColumnNumber :
2 ** 31 - 1;
const sourceRange =
new TextUtils.TextRange.TextRange(startSourceLine, startSourceColumn, endSourceLine, endSourceColumn);
return {range, sourceRange, sourceURL};
}
sourceLineMapping(sourceURL: Platform.DevToolsPath.UrlString, lineNumber: number, columnNumber: number):
SourceMapEntry|null {
const mappings = this.mappings();
const reverseMappings = this.reversedMappings(sourceURL);
const first = Platform.ArrayUtilities.lowerBound(reverseMappings, lineNumber, lineComparator);
const last = Platform.ArrayUtilities.upperBound(reverseMappings, lineNumber, lineComparator);
if (first >= reverseMappings.length || mappings[reverseMappings[first]].sourceLineNumber !== lineNumber) {
return null;
}
const columnMappings = reverseMappings.slice(first, last);
if (!columnMappings.length) {
return null;
}
const index = Platform.ArrayUtilities.lowerBound(
columnMappings, columnNumber, (columnNumber, i) => columnNumber - mappings[i].sourceColumnNumber);
return index >= columnMappings.length ? mappings[columnMappings[columnMappings.length - 1]] :
mappings[columnMappings[index]];
function lineComparator(lineNumber: number, i: number): number {
return lineNumber - mappings[i].sourceLineNumber;
}
}
private findReverseIndices(sourceURL: Platform.DevToolsPath.UrlString, lineNumber: number, columnNumber: number):
number[] {
const mappings = this.mappings();
const reverseMappings = this.reversedMappings(sourceURL);
const endIndex = Platform.ArrayUtilities.upperBound(
reverseMappings, undefined,
(_, i) => lineNumber - mappings[i].sourceLineNumber || columnNumber - mappings[i].sourceColumnNumber);
let startIndex = endIndex;
while (startIndex > 0 &&
mappings[reverseMappings[startIndex - 1]].sourceLineNumber ===
mappings[reverseMappings[endIndex - 1]].sourceLineNumber &&
mappings[reverseMappings[startIndex - 1]].sourceColumnNumber ===
mappings[reverseMappings[endIndex - 1]].sourceColumnNumber) {
--startIndex;
}
return reverseMappings.slice(startIndex, endIndex);
}
findReverseEntries(sourceURL: Platform.DevToolsPath.UrlString, lineNumber: number, columnNumber: number):
SourceMapEntry[] {
const mappings = this.mappings();
return this.findReverseIndices(sourceURL, lineNumber, columnNumber).map(i => mappings[i]);
}
findReverseRanges(sourceURL: Platform.DevToolsPath.UrlString, lineNumber: number, columnNumber: number):
TextUtils.TextRange.TextRange[] {
const mappings = this.mappings();
const indices = this.findReverseIndices(sourceURL, lineNumber, columnNumber);
const ranges: TextUtils.TextRange.TextRange[] = [];
for (let i = 0; i < indices.length; ++i) {
const startIndex = indices[i];
// Merge adjacent ranges.
let endIndex = startIndex + 1;
while (i + 1 < indices.length && endIndex === indices[i + 1]) {
++endIndex;
++i;
}
// Source maps don't contain end positions for entries, but each entry is assumed to
// span until the following entry. This doesn't work however in case of the last
// entry, where there's no following entry. We also don't know the number of lines
// and columns in the original source code (which might not be available at all), so
// for that case we store the maximum signed 32-bit integer, which is definitely going
// to be larger than any script we can process and can safely be serialized as part of
// the skip list we send to V8 with `Debugger.stepOver` (http://crbug.com/1305956).
const startLine = mappings[startIndex].lineNumber;
const startColumn = mappings[startIndex].columnNumber;
const endLine = endIndex < mappings.length ? mappings[endIndex].lineNumber : 2 ** 31 - 1;
const endColumn = endIndex < mappings.length ? mappings[endIndex].columnNumber : 2 ** 31 - 1;
ranges.push(new TextUtils.TextRange.TextRange(startLine, startColumn, endLine, endColumn));
}
return ranges;
}
mappings(): SourceMapEntry[] {
this.#ensureSourceMapProcessed();
return this.#mappings ?? [];
}
/**
* If the source map does not contain scope information by itself (e.g. "scopes proposal"
* or "pasta" scopes), then we'll use this getter to calculate basic function name information from
* the AST and mappings.
*/
async #buildScopesFallback(): Promise<SourceMapScopesInfo|null> {
const scopeTreeAndText = this.#script ? await scopeTreeForScript(this.#script) : null;
if (!scopeTreeAndText) {
return null;
}
const {scopeTree, text} = scopeTreeAndText;
return SourceMapScopesInfo.createFromAst(this, scopeTree, text);
}
private reversedMappings(sourceURL: Platform.DevToolsPath.UrlString): number[] {
this.#ensureSourceMapProcessed();
return this.#sourceInfoByURL.get(sourceURL)?.reverseMappings ?? [];
}
#ensureSourceMapProcessed(): void {
if (this.#mappings === null) {
this.#mappings = [];
try {
this.eachSection(this.parseMap.bind(this));
if (!this.hasScopeInfo()) {
this.scopesFallbackPromiseForTest = this.#buildScopesFallback().then(info => {
this.#scopesInfo = info;
});
}
} catch (e) {
console.error('Failed to parse source map', e);
this.#mappings = [];
}
// As per spec, mappings are not necessarily sorted.
this.mappings().sort(SourceMapEntry.compare);
this.#computeReverseMappings(this.#mappings);
}
if (!SourceMap.retainRawSourceMaps) {
this.#json = null;
}
}
#computeReverseMappings(mappings: SourceMapEntry[]): void {
const reverseMappingsPerUrl = new Map<Platform.DevToolsPath.UrlString, number[]>();
for (let i = 0; i < mappings.length; i++) {
const entryUrl = mappings[i]?.sourceURL;
if (!entryUrl) {
continue;
}
let reverseMap = reverseMappingsPerUrl.get(entryUrl);
if (!reverseMap) {
reverseMap = [];
reverseMappingsPerUrl.set(entryUrl, reverseMap);
}
reverseMap.push(i);
}
for (const [url, reverseMap] of reverseMappingsPerUrl.entries()) {
const info = this.#sourceInfoByURL.get(url);
if (!info) {
continue;
}
reverseMap.sort(sourceMappingComparator);
info.reverseMappings = reverseMap;
}
function sourceMappingComparator(indexA: number, indexB: number): number {
const a = mappings[indexA];
const b = mappings[indexB];
return a.sourceLineNumber - b.sourceLineNumber || a.sourceColumnNumber - b.sourceColumnNumber ||
a.lineNumber - b.lineNumber || a.columnNumber - b.columnNumber;
}
}
private eachSection(callback: (arg0: SourceMapV3Object, arg1: number, arg2: number, arg3: number) => void): void {
if (!this.#json) {
return;
}
if ('sections' in this.#json) {
let sourcesIndex = 0;
for (const section of this.#json.sections) {
if ('map' in section) {
callback(section.map, sourcesIndex, section.offset.line, section.offset.column);
sourcesIndex += section.map.sources.length;
}
}
} else {
callback(this.#json, 0, 0, 0);
}
}
private parseSources(sourceMap: SourceMapV3Object): void {
const sourceRoot = sourceMap.sourceRoot ?? '';
const ignoreList = new Set(sourceMap.ignoreList ?? sourceMap.x_google_ignoreList);
for (let i = 0; i < sourceMap.sources.length; ++i) {
let href = sourceMap.sources[i];
// The source map v3 proposal says to prepend the sourceRoot to the source URL
// and if the resulting URL is not absolute, then resolve the source URL against
// the source map URL. Prepending the sourceRoot (if one exists) is not likely to
// be meaningful or useful if the source URL is already absolute though. In this
// case, use the source URL as is without prepending the sourceRoot.
if (Common.ParsedURL.ParsedURL.isRelativeURL(href)) {
if (sourceRoot && !sourceRoot.endsWith('/') && href && !href.startsWith('/')) {
href = sourceRoot.concat('/', href);
} else {
href = sourceRoot.concat(href);
}
}
const url =
Common.ParsedURL.ParsedURL.completeURL(this.#baseURL, href) || (href as Platform.DevToolsPath.UrlString);
const source = sourceMap.sourcesContent?.[i];
const sourceInfo: SourceInfo = {
sourceURL: url,
content: source ?? null,
ignoreListHint: ignoreList.has(i),
reverseMappings: null,
};
this.#sourceInfos.push(sourceInfo);
if (!this.#sourceInfoByURL.has(url)) {
this.#sourceInfoByURL.set(url, sourceInfo);
}
}
}
private parseMap(map: SourceMapV3Object, baseSourceIndex: number, baseLineNumber: number, baseColumnNumber: number):
void {
let sourceIndex = baseSourceIndex;
let lineNumber = baseLineNumber;
let columnNumber = baseColumnNumber;
let sourceLineNumber = 0;
let sourceColumnNumber = 0;
let nameIndex = 0;
const names = map.names ?? [];
const tokenIter = new TokenIterator(map.mappings);
let sourceURL: Platform.DevToolsPath.UrlString|undefined = this.#sourceInfos[sourceIndex]?.sourceURL;
while (true) {
if (tokenIter.peek() === ',') {
tokenIter.next();
} else {
while (tokenIter.peek() === ';') {
lineNumber += 1;
columnNumber = 0;
tokenIter.next();
}
if (!tokenIter.hasNext()) {
break;
}
}
columnNumber += tokenIter.nextVLQ();
if (!tokenIter.hasNext() || this.isSeparator(tokenIter.peek())) {
this.mappings().push(new SourceMapEntry(lineNumber, columnNumber));
continue;
}
const sourceIndexDelta = tokenIter.nextVLQ();
if (sourceIndexDelta) {
sourceIndex += sourceIndexDelta;
sourceURL = this.#sourceInfos[sourceIndex]?.sourceURL;
}
sourceLineNumber += tokenIter.nextVLQ();
sourceColumnNumber += tokenIter.nextVLQ();
if (!tokenIter.hasNext() || this.isSeparator(tokenIter.peek())) {
this.mappings().push(
new SourceMapEntry(lineNumber, columnNumber, sourceIndex, sourceURL, sourceLineNumber, sourceColumnNumber));
continue;
}
nameIndex += tokenIter.nextVLQ();
this.mappings().push(new SourceMapEntry(
lineNumber, columnNumber, sourceIndex, sourceURL, sourceLineNumber, sourceColumnNumber, names[nameIndex]));
}
if (Root.Runtime.experiments.isEnabled(Root.Runtime.ExperimentName.USE_SOURCE_MAP_SCOPES)) {
if (!this.#scopesInfo) {
this.#scopesInfo = new SourceMapScopesInfo(this, {scopes: [], ranges: []});
}
if (map.scopes) {
const {scopes, ranges} = ScopesCodec.decode(
map as ScopesCodec.SourceMapJson,
{mode: ScopesCodec.DecodeMode.LAX, generatedOffset: {line: baseLineNumber, column: baseColumnNumber}});
this.#scopesInfo.addOriginalScopes(scopes);
this.#scopesInfo.addGeneratedRanges(ranges);
} else if (map.x_com_bloomberg_sourcesFunctionMappings) {
const originalScopes = this.parseBloombergScopes(map);
this.#scopesInfo.addOriginalScopes(originalScopes);
} else {
// Keep the OriginalScope[] tree array consistent with sources.
this.#scopesInfo.addOriginalScopes(new Array(map.sources.length).fill(null));
}
}
}
private parseBloombergScopes(map: SourceMapV3Object): Array<ScopesCodec.OriginalScope|null> {
const scopeList = map.x_com_bloomberg_sourcesFunctionMappings;
if (!scopeList) {
throw new Error('Cant decode pasta scopes without x_com_bloomberg_sourcesFunctionMappings field');
} else if (scopeList.length !== map.sources.length) {
throw new Error(`x_com_bloomberg_sourcesFunctionMappings must have ${map.sources.length} scope trees`);
}
const names = map.names ?? [];
return scopeList.map(rawScopes => {
if (!rawScopes) {
return null;
}
const ranges = decodePastaRanges(rawScopes, names);
return buildOriginalScopes(ranges);
});
}
private isSeparator(char: string): boolean {
return char === ',' || char === ';';
}
/**
* Finds all the reverse mappings that intersect with the given `textRange` within the
* source entity identified by the `url`. If the `url` does not have any reverse mappings
* within this source map, an empty array is returned.
*
* @param url the URL of the source entity to query.
* @param textRange the range of text within the entity to check, considered `[start,end[`.
* @returns the list of ranges in the generated file that map to locations overlapping the
* {@link textRange} in the source file identified by the {@link url}, or `[]`
* if the {@link url} does not identify an entity in this source map.
*/
reverseMapTextRanges(url: Platform.DevToolsPath.UrlString, textRange: TextUtils.TextRange.TextRange):
TextUtils.TextRange.TextRange[] {
const reverseMappings = this.reversedMappings(url);
const mappings = this.mappings();
if (reverseMappings.length === 0) {
return [];
}
// Determine the first reverse mapping that contains the starting point of the `textRange`.
let startReverseIndex =
Platform.ArrayUtilities.lowerBound(reverseMappings, textRange, ({startLine, startColumn}, index) => {
const {sourceLineNumber, sourceColumnNumber} = mappings[index];
return startLine - sourceLineNumber || startColumn - sourceColumnNumber;
});
// Check if the current mapping does not start on the exact start of the `textRange`, and if
// so we know that a previous mapping entry (if any) would also overlap. If we reach the end
// of the reverse mappings table, we just take the last entry and report that.
while (startReverseIndex === reverseMappings.length ||
startReverseIndex > 0 &&
(mappings[reverseMappings[startReverseIndex]].sourceLineNumber > textRange.startLine ||
mappings[reverseMappings[startReverseIndex]].sourceColumnNumber > textRange.startColumn)) {
startReverseIndex--;
}
// Determine the last reverse mapping that contains the end point of the `textRange`.
let endReverseIndex = startReverseIndex + 1;
for (; endReverseIndex < reverseMappings.length; ++endReverseIndex) {
const {sourceLineNumber, sourceColumnNumber} = mappings[reverseMappings[endReverseIndex]];
if (sourceLineNumber < textRange.endLine ||
(sourceLineNumber === textRange.endLine && sourceColumnNumber < textRange.endColumn)) {
continue;
}
break;
}
// Create the ranges...
const ranges = [];
for (let reverseIndex = startReverseIndex; reverseIndex < endReverseIndex; ++reverseIndex) {
const startIndex = reverseMappings[reverseIndex], endIndex = startIndex + 1;
const range = TextUtils.TextRange.TextRange.createUnboundedFromLocation(
mappings[startIndex].lineNumber, mappings[startIndex].columnNumber);
if (endIndex < mappings.length) {
range.endLine = mappings[endIndex].lineNumber;
range.endColumn = mappings[endIndex].columnNumber;
}
ranges.push(range);
}
// ...sort them...
ranges.sort(TextUtils.TextRange.TextRange.comparator);
// ...and ensure they are maximally merged.
let j = 0;
for (let i = 1; i < ranges.length; ++i) {
if (ranges[j].immediatelyPrecedes(ranges[i])) {
ranges[j].endLine = ranges[i].endLine;
ranges[j].endColumn = ranges[i].endColumn;
} else {
ranges[++j] = ranges[i];
}
}
ranges.length = j + 1;
return ranges;
}
mapsOrigin(): boolean {
const mappings = this.mappings();
if (mappings.length > 0) {
const firstEntry = mappings[0];
return firstEntry?.lineNumber === 0 || firstEntry.columnNumber === 0;
}
return false;
}
hasIgnoreListHint(sourceURL: Platform.DevToolsPath.UrlString): boolean {
return this.#sourceInfoByURL.get(sourceURL)?.ignoreListHint ?? false;
}
/**
* Returns a list of ranges in the generated script for original sources that
* match a predicate. Each range is a [begin, end) pair, meaning that code at
* the beginning location, up to but not including the end location, matches
* the predicate.
*/
findRanges(predicate: (sourceURL: Platform.DevToolsPath.UrlString) => boolean, options?: {isStartMatching: boolean}):
TextUtils.TextRange.TextRange[] {
const mappings = this.mappings();
const ranges = [];
if (!mappings.length) {
return [];
}
let current: TextUtils.TextRange.TextRange|null = null;
// If the first mapping isn't at the beginning of the original source, it's
// up to the caller to decide if it should be considered matching the
// predicate or not. By default, it's not.
if ((mappings[0].lineNumber !== 0 || mappings[0].columnNumber !== 0) && options?.isStartMatching) {
current = TextUtils.TextRange.TextRange.createUnboundedFromLocation(0, 0);
ranges.push(current);
}
for (const {sourceURL, lineNumber, columnNumber} of mappings) {
const ignoreListHint = sourceURL && predicate(sourceURL);
if (!current && ignoreListHint) {
current = TextUtils.TextRange.TextRange.createUnboundedFromLocation(lineNumber, columnNumber);
ranges.push(current);
continue;
}
if (current && !ignoreListHint) {
current.endLine = lineNumber;
current.endColumn = columnNumber;
current = null;
}
}
return ranges;
}
/**
* Determines whether this and the {@link other} `SourceMap` agree on content and ignore-list hint
* with respect to the {@link sourceURL}.
*
* @param sourceURL the URL to test for (might not be provided by either of the sourcemaps).
* @param other the other `SourceMap` to check.
* @returns `true` if both this and the {@link other} `SourceMap` either both have the ignore-list
* hint for {@link sourceURL} or neither, and if both of them either provide the same
* content for the {@link sourceURL} inline or both provide no `sourcesContent` entry
* for it.
*/
compatibleForURL(sourceURL: Platform.DevToolsPath.UrlString, other: SourceMap): boolean {
return this.embeddedContentByURL(sourceURL) === other.embeddedContentByURL(sourceURL) &&
this.hasIgnoreListHint(sourceURL) === other.hasIgnoreListHint(sourceURL);
}
expandCallFrame(frame: CallFrame): CallFrame[] {
this.#ensureSourceMapProcessed();
if (this.#scopesInfo === null) {
return [frame];
}
return this.#scopesInfo.expandCallFrame(frame);
}
resolveScopeChain(frame: CallFrame): ScopeChainEntry[]|null {
this.#ensureSourceMapProcessed();
if (this.#scopesInfo === null) {
return null;
}
return this.#scopesInfo.resolveMappedScopeChain(frame);
}
findOriginalFunctionName(position: ScopesCodec.Position): string|null {
this.#ensureSourceMapProcessed();
return this.#scopesInfo?.findOriginalFunctionName(position) ?? null;
}
findOriginalFunctionScope(position: ScopesCodec.Position):
{scope: ScopesCodec.OriginalScope, url?: Platform.DevToolsPath.UrlString}|null {
this.#ensureSourceMapProcessed();
return this.#scopesInfo?.findOriginalFunctionScope(position) ?? null;
}
isOutlinedFrame(generatedLine: number, generatedColumn: number): boolean {
this.#ensureSourceMapProcessed();
return this.#scopesInfo?.isOutlinedFrame(generatedLine, generatedColumn) ?? false;
}
hasInlinedFrames(generatedLine: number, generatedColumn: number): boolean {
this.#ensureSourceMapProcessed();
return this.#scopesInfo?.hasInlinedFrames(generatedLine, generatedColumn) ?? false;
}
translateCallSite(generatedLine: number, generatedColumn: number): TranslatedFrame[] {
this.#ensureSourceMapProcessed();
return this.#scopesInfo?.translateCallSite(generatedLine, generatedColumn) ?? [];
}
}
const VLQ_BASE_SHIFT = 5;
const VLQ_BASE_MASK = (1 << 5) - 1;
const VLQ_CONTINUATION_MASK = 1 << 5;
export class TokenIterator {
readonly #string: string;
#position: number;
constructor(string: string) {
this.#string = string;
this.#position = 0;
}
next(): string {
return this.#string.charAt(this.#position++);
}
/** Returns the unicode value of the next character and advances the iterator */
nextCharCode(): number {
return this.#string.charCodeAt(this.#position++);
}
peek(): string {
return this.#string.charAt(this.#position);
}
hasNext(): boolean {
return this.#position < this.#string.length;
}
nextVLQ(): number {
// Read unsigned value.
let result = 0;
let shift = 0;
let digit: number = VLQ_CONTINUATION_MASK;
while (digit & VLQ_CONTINUATION_MASK) {
if (!this.hasNext()) {
throw new Error('Unexpected end of input while decodling VLQ number!');
}
const charCode = this.nextCharCode();
digit = Common.Base64.BASE64_CODES[charCode];
if (charCode !== 65 /* 'A' */ && digit === 0) {
throw new Error(`Unexpected char '${String.fromCharCode(charCode)}' encountered while decoding`);
}
result += (digit & VLQ_BASE_MASK) << shift;
shift += VLQ_BASE_SHIFT;
}
// Fix the sign.
const negative = result & 1;
result >>= 1;
return negative ? -result : result;
}
/**
* @returns the next VLQ number without iterating further. Or returns null if
* the iterator is at the end or it's not a valid number.
*/
peekVLQ(): null|number {
const pos = this.#position;
try {
return this.nextVLQ();
} catch {
return null;
} finally {
this.#position = pos; // Reset the iterator.
}
}
}