UNPKG

chrome-devtools-frontend

Version:
684 lines (603 loc) • 27 kB
// Copyright 2021 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. /* * Copyright (C) 2012 Google Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are * met: * * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above * copyright notice, this list of conditions and the following disclaimer * in the documentation and/or other materials provided with the * distribution. * * Neither the #name of Google Inc. nor the names of its * contributors may be used to endorse or promote products derived from * this software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ import * as TextUtils from '../../models/text_utils/text_utils.js'; import * as Common from '../common/common.js'; import * as i18n from '../i18n/i18n.js'; import * as Platform from '../platform/platform.js'; import {CompilerSourceMappingContentProvider} from './CompilerSourceMappingContentProvider.js'; import {PageResourceLoader, type PageResourceLoadInitiator} from './PageResourceLoader.js'; const UIStrings = { /** *@description Error message when failing to load a source map text via the network *@example {https://example.com/sourcemap.map} PH1 *@example {A certificate error occurred} PH2 */ couldNotLoadContentForSS: 'Could not load content for {PH1}: {PH2}', /** *@description Error message when failing to load a script source text via the network *@example {https://example.com} PH1 *@example {Unexpected token} PH2 */ couldNotParseContentForSS: 'Could not parse content for {PH1}: {PH2}', }; const str_ = i18n.i18n.registerUIStrings('core/sdk/SourceMap.ts', UIStrings); const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_); export interface SourceMap { compiledURL(): Platform.DevToolsPath.UrlString; url(): Platform.DevToolsPath.UrlString; sourceURLs(): Platform.DevToolsPath.UrlString[]; sourceContentProvider(sourceURL: Platform.DevToolsPath.UrlString, contentType: Common.ResourceType.ResourceType): TextUtils.ContentProvider.ContentProvider; embeddedContentByURL(sourceURL: Platform.DevToolsPath.UrlString): string|null; findEntry(lineNumber: number, columnNumber: number): SourceMapEntry|null; findEntryRanges(lineNumber: number, columnNumber: number): { range: TextUtils.TextRange.TextRange, sourceRange: TextUtils.TextRange.TextRange, sourceURL: Platform.DevToolsPath.UrlString, }|null; findReverseRanges(sourceURL: Platform.DevToolsPath.UrlString, lineNumber: number, columnNumber: number): TextUtils.TextRange.TextRange[]; sourceLineMapping(sourceURL: Platform.DevToolsPath.UrlString, lineNumber: number, columnNumber: number): SourceMapEntry|null; mappings(): SourceMapEntry[]; mapsOrigin(): boolean; hasIgnoreListHint(sourceURL: Platform.DevToolsPath.UrlString): boolean; findRanges(predicate: (sourceURL: Platform.DevToolsPath.UrlString) => boolean, options: {isStartMatching: boolean}): TextUtils.TextRange.TextRange[]; } export class SourceMapV3 { version!: number; file!: string|undefined; sources!: Platform.DevToolsPath.UrlString[]; sections!: Section[]|undefined; mappings!: string; sourceRoot!: Platform.DevToolsPath.UrlString|undefined; names!: string[]|undefined; sourcesContent!: string|undefined; // eslint-disable-next-line @typescript-eslint/naming-convention x_google_ignoreList!: number[]|undefined; constructor() { } } export class Section { map!: SourceMapV3; offset!: Offset; url!: Platform.DevToolsPath.UrlString|undefined; constructor() { } } export class Offset { line!: number; column!: number; constructor() { } } export class SourceMapEntry { lineNumber: number; columnNumber: number; sourceURL: Platform.DevToolsPath.UrlString|undefined; sourceLineNumber: number; sourceColumnNumber: number; name: string|undefined; constructor( lineNumber: number, columnNumber: number, sourceURL?: Platform.DevToolsPath.UrlString, sourceLineNumber?: number, sourceColumnNumber?: number, name?: string) { this.lineNumber = lineNumber; this.columnNumber = columnNumber; 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; } } const base64Digits = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/'; const base64Map = new Map<string, number>(); for (let i = 0; i < base64Digits.length; ++i) { base64Map.set(base64Digits.charAt(i), i); } const sourceMapToSourceList = new WeakMap<SourceMapV3, Platform.DevToolsPath.UrlString[]>(); export class TextSourceMap implements SourceMap { readonly #initiator: PageResourceLoadInitiator; #json: SourceMapV3|null; readonly #compiledURLInternal: Platform.DevToolsPath.UrlString; readonly #sourceMappingURL: Platform.DevToolsPath.UrlString; readonly #baseURL: Platform.DevToolsPath.UrlString; #mappingsInternal: SourceMapEntry[]|null; readonly #sourceInfos: Map<Platform.DevToolsPath.UrlString, TextSourceMap.SourceInfo>; /** * 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, initiator: PageResourceLoadInitiator) { this.#initiator = initiator; this.#json = payload; this.#compiledURLInternal = compiledURL; this.#sourceMappingURL = sourceMappingURL; this.#baseURL = (sourceMappingURL.startsWith('data:') ? compiledURL : sourceMappingURL); this.#mappingsInternal = null; this.#sourceInfos = new Map(); if (this.#json.sections) { const sectionWithURL = Boolean(this.#json.sections.find(section => Boolean(section.url))); if (sectionWithURL) { Common.Console.Console.instance().warn( `SourceMap "${sourceMappingURL}" contains unsupported "URL" field in one of its sections.`); } } this.eachSection(this.parseSources.bind(this)); } /** * @throws {!Error} */ static async load( sourceMapURL: Platform.DevToolsPath.UrlString, compiledURL: Platform.DevToolsPath.UrlString, initiator: PageResourceLoadInitiator): Promise<TextSourceMap> { let updatedContent; try { const {content} = await PageResourceLoader.instance().loadResource(sourceMapURL, initiator); updatedContent = content; if (content.slice(0, 3) === ')]}') { updatedContent = content.substring(content.indexOf('\n')); } } catch (error) { throw new Error(i18nString(UIStrings.couldNotLoadContentForSS, {PH1: sourceMapURL, PH2: error.message})); } try { const payload = (JSON.parse(updatedContent) as SourceMapV3); return new TextSourceMap(compiledURL, sourceMapURL, payload, initiator); } catch (error) { throw new Error(i18nString(UIStrings.couldNotParseContentForSS, {PH1: sourceMapURL, PH2: error.message})); } } compiledURL(): Platform.DevToolsPath.UrlString { return this.#compiledURLInternal; } url(): Platform.DevToolsPath.UrlString { return this.#sourceMappingURL; } sourceURLs(): Platform.DevToolsPath.UrlString[] { return [...this.#sourceInfos.keys()]; } sourceContentProvider(sourceURL: Platform.DevToolsPath.UrlString, contentType: Common.ResourceType.ResourceType): TextUtils.ContentProvider.ContentProvider { const info = this.#sourceInfos.get(sourceURL); if (info && info.content) { return TextUtils.StaticContentProvider.StaticContentProvider.fromString(sourceURL, contentType, info.content); } return new CompilerSourceMappingContentProvider(sourceURL, contentType, this.#initiator); } embeddedContentByURL(sourceURL: Platform.DevToolsPath.UrlString): string|null { const entry = this.#sourceInfos.get(sourceURL); if (!entry) { return null; } return entry.content; } findEntry(lineNumber: number, columnNumber: number): SourceMapEntry|null { const mappings = this.mappings(); const index = Platform.ArrayUtilities.upperBound( mappings, undefined, (unused, 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 index = Platform.ArrayUtilities.upperBound( mappings, undefined, (unused, entry) => lineNumber - entry.lineNumber || columnNumber - entry.columnNumber); if (!index) { // If the line and column are preceding all the entries, then there is nothing to map. return null; } const sourceURL = mappings[index].sourceURL; if (!sourceURL) { return null; } // Let us compute the range that contains the source position in the compiled code. const endLine = index < mappings.length ? mappings[index].lineNumber : 2 ** 31 - 1; const endColumn = index < mappings.length ? mappings[index].columnNumber : 2 ** 31 - 1; const range = new TextUtils.TextRange.TextRange( mappings[index - 1].lineNumber, mappings[index - 1].columnNumber, endLine, endColumn); // Now try to find the corresponding token in the original code. const reverseMappings = this.reversedMappings(sourceURL); const startSourceLine = mappings[index - 1].sourceLineNumber; const startSourceColumn = mappings[index - 1].sourceColumnNumber; const endReverseIndex = Platform.ArrayUtilities.upperBound( reverseMappings, undefined, (unused, 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, (unused, 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.#ensureMappingsProcessed(); return this.#mappingsInternal ?? []; } private reversedMappings(sourceURL: Platform.DevToolsPath.UrlString): number[] { this.#ensureMappingsProcessed(); return this.#sourceInfos.get(sourceURL)?.reverseMappings ?? []; } #ensureMappingsProcessed(): void { if (this.#mappingsInternal === null) { this.#mappingsInternal = []; this.eachSection(this.parseMap.bind(this)); this.#computeReverseMappings(this.#mappingsInternal); 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.#sourceInfos.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: SourceMapV3, arg1: number, arg2: number) => void): void { if (!this.#json) { return; } if (!this.#json.sections) { callback(this.#json, 0, 0); return; } for (const section of this.#json.sections) { callback(section.map, section.offset.line, section.offset.column); } } private parseSources(sourceMap: SourceMapV3): void { const sourcesList = []; const sourceRoot = sourceMap.sourceRoot || Platform.DevToolsPath.EmptyUrlString; const ignoreList = new Set(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. Appending 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 = Common.ParsedURL.ParsedURL.concatenate(sourceRoot, '/', href); } else { href = Common.ParsedURL.ParsedURL.concatenate(sourceRoot, href); } } let url = Common.ParsedURL.ParsedURL.completeURL(this.#baseURL, href) || href; const source = sourceMap.sourcesContent && sourceMap.sourcesContent[i]; if (url === this.#compiledURLInternal && source) { url = Common.ParsedURL.ParsedURL.concatenate(url, '? [sm]'); } sourcesList.push(url); if (!this.#sourceInfos.has(url)) { const content = source ?? null; const ignoreListHint = ignoreList.has(i); this.#sourceInfos.set(url, new TextSourceMap.SourceInfo(content, ignoreListHint)); } } sourceMapToSourceList.set(sourceMap, sourcesList); } private parseMap(map: SourceMapV3, lineNumber: number, columnNumber: number): void { let sourceIndex = 0; let sourceLineNumber = 0; let sourceColumnNumber = 0; let nameIndex = 0; // TODO(crbug.com/1011811): refactor away map. // `sources` can be undefined if it wasn't previously // processed and added to the list. However, that // is not WAI and we should make sure that we can // only reach this point when we are certain // we have the list available. const sources = sourceMapToSourceList.get(map); const names = map.names || []; const stringCharIterator = new TextSourceMap.StringCharIterator(map.mappings); let sourceURL: Platform.DevToolsPath.UrlString|undefined = sources && sources[sourceIndex]; while (true) { if (stringCharIterator.peek() === ',') { stringCharIterator.next(); } else { while (stringCharIterator.peek() === ';') { lineNumber += 1; columnNumber = 0; stringCharIterator.next(); } if (!stringCharIterator.hasNext()) { break; } } columnNumber += this.decodeVLQ(stringCharIterator); if (!stringCharIterator.hasNext() || this.isSeparator(stringCharIterator.peek())) { this.mappings().push(new SourceMapEntry(lineNumber, columnNumber)); continue; } const sourceIndexDelta = this.decodeVLQ(stringCharIterator); if (sourceIndexDelta) { sourceIndex += sourceIndexDelta; if (sources) { sourceURL = sources[sourceIndex]; } } sourceLineNumber += this.decodeVLQ(stringCharIterator); sourceColumnNumber += this.decodeVLQ(stringCharIterator); if (!stringCharIterator.hasNext() || this.isSeparator(stringCharIterator.peek())) { this.mappings().push( new SourceMapEntry(lineNumber, columnNumber, sourceURL, sourceLineNumber, sourceColumnNumber)); continue; } nameIndex += this.decodeVLQ(stringCharIterator); this.mappings().push(new SourceMapEntry( lineNumber, columnNumber, sourceURL, sourceLineNumber, sourceColumnNumber, names[nameIndex])); } // As per spec, mappings are not necessarily sorted. this.mappings().sort(SourceMapEntry.compare); } private isSeparator(char: string): boolean { return char === ',' || char === ';'; } private decodeVLQ(stringCharIterator: TextSourceMap.StringCharIterator): number { // Read unsigned value. let result = 0; let shift = 0; let digit: number = TextSourceMap._VLQ_CONTINUATION_MASK; while (digit & TextSourceMap._VLQ_CONTINUATION_MASK) { digit = base64Map.get(stringCharIterator.next()) || 0; result += (digit & TextSourceMap._VLQ_BASE_MASK) << shift; shift += TextSourceMap._VLQ_BASE_SHIFT; } // Fix the sign. const negative = result & 1; result >>= 1; return negative ? -result : result; } reverseMapTextRange(url: Platform.DevToolsPath.UrlString, textRange: TextUtils.TextRange.TextRange): TextUtils.TextRange.TextRange|null { function comparator( position: { lineNumber: number, columnNumber: number, }, mappingIndex: number): number { if (position.lineNumber !== mappings[mappingIndex].sourceLineNumber) { return position.lineNumber - mappings[mappingIndex].sourceLineNumber; } return position.columnNumber - mappings[mappingIndex].sourceColumnNumber; } const reverseMappings = this.reversedMappings(url); const mappings = this.mappings(); if (!reverseMappings.length) { return null; } const startIndex = Platform.ArrayUtilities.lowerBound( reverseMappings, {lineNumber: textRange.startLine, columnNumber: textRange.startColumn}, comparator); const endIndex = Platform.ArrayUtilities.upperBound( reverseMappings, {lineNumber: textRange.endLine, columnNumber: textRange.endColumn}, comparator); if (endIndex >= reverseMappings.length) { return null; } const startMapping = mappings[reverseMappings[startIndex]]; const endMapping = mappings[reverseMappings[endIndex]]; return new TextUtils.TextRange.TextRange( startMapping.lineNumber, startMapping.columnNumber, endMapping.lineNumber, endMapping.columnNumber); } 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.#sourceInfos.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; } } export namespace TextSourceMap { // TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration // eslint-disable-next-line @typescript-eslint/naming-convention export const _VLQ_BASE_SHIFT = 5; // TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration // eslint-disable-next-line @typescript-eslint/naming-convention export const _VLQ_BASE_MASK = (1 << 5) - 1; // TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration // eslint-disable-next-line @typescript-eslint/naming-convention export const _VLQ_CONTINUATION_MASK = 1 << 5; export class StringCharIterator { private readonly string: string; private position: number; constructor(string: string) { this.string = string; this.position = 0; } next(): string { return this.string.charAt(this.position++); } peek(): string { return this.string.charAt(this.position); } hasNext(): boolean { return this.position < this.string.length; } } export class SourceInfo { reverseMappings: number[]|null = null; constructor(public content: string|null, public ignoreListHint: boolean) { } } }