From ab7cb271898c988fec8747c7a3a1d92692b2c306 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Thu, 15 Feb 2018 12:26:40 -0800 Subject: [PATCH 01/32] Add compiler option to enable declaration sourcemaps --- src/compiler/commandLineParser.ts | 7 + src/compiler/diagnosticMessages.json | 4 + src/compiler/emitter.ts | 33 +- src/compiler/program.ts | 8 +- src/compiler/types.ts | 1 + src/compiler/utilities.ts | 3 +- src/harness/compilerRunner.ts | 2 +- src/harness/harness.ts | 12 +- src/harness/sourceMapRecorder.ts | 21 +- .../reference/api/tsserverlibrary.d.ts | 1 + tests/baselines/reference/api/typescript.d.ts | 1 + tests/baselines/reference/declarationMaps.js | 43 ++ .../reference/declarationMaps.js.map | 2 + .../reference/declarationMaps.sourcemap.txt | 299 ++++++++++++ .../reference/declarationMaps.symbols | 50 ++ .../baselines/reference/declarationMaps.types | 50 ++ .../reference/declarationMapsMultifile.js | 64 +++ .../reference/declarationMapsMultifile.js.map | 3 + .../declarationMapsMultifile.sourcemap.txt | 201 ++++++++ .../declarationMapsMultifile.symbols | 47 ++ .../reference/declarationMapsMultifile.types | 56 +++ .../reference/declarationMapsOutFile.js | 69 +++ .../reference/declarationMapsOutFile.js.map | 2 + .../declarationMapsOutFile.sourcemap.txt | 192 ++++++++ .../reference/declarationMapsOutFile.symbols | 47 ++ .../reference/declarationMapsOutFile.types | 56 +++ .../reference/declarationMapsOutFile2.js | 49 ++ .../reference/declarationMapsOutFile2.js.map | 2 + .../declarationMapsOutFile2.sourcemap.txt | 136 ++++++ .../reference/declarationMapsOutFile2.symbols | 40 ++ .../reference/declarationMapsOutFile2.types | 49 ++ .../reference/declarationMapsWithSourceMap.js | 49 ++ .../declarationMapsWithSourceMap.js.map | 3 + ...declarationMapsWithSourceMap.sourcemap.txt | 436 ++++++++++++++++++ .../declarationMapsWithSourceMap.symbols | 40 ++ .../declarationMapsWithSourceMap.types | 49 ++ ...clarationMapsWithoutDeclaration.errors.txt | 23 + .../declarationMapsWithoutDeclaration.js | 24 + .../declarationMapsWithoutDeclaration.symbols | 50 ++ .../declarationMapsWithoutDeclaration.types | 50 ++ .../tsconfig.json | 1 + .../tsconfig.json | 1 + .../tsconfig.json | 1 + .../tsconfig.json | 1 + .../tsconfig.json | 1 + .../tsconfig.json | 1 + .../tsconfig.json | 1 + .../tsconfig.json | 1 + tests/cases/compiler/declarationMaps.ts | 20 + .../compiler/declarationMapsMultifile.ts | 19 + .../cases/compiler/declarationMapsOutFile.ts | 21 + .../cases/compiler/declarationMapsOutFile2.ts | 17 + .../compiler/declarationMapsWithSourceMap.ts | 18 + .../declarationMapsWithoutDeclaration.ts | 19 + 54 files changed, 2378 insertions(+), 18 deletions(-) create mode 100644 tests/baselines/reference/declarationMaps.js create mode 100644 tests/baselines/reference/declarationMaps.js.map create mode 100644 tests/baselines/reference/declarationMaps.sourcemap.txt create mode 100644 tests/baselines/reference/declarationMaps.symbols create mode 100644 tests/baselines/reference/declarationMaps.types create mode 100644 tests/baselines/reference/declarationMapsMultifile.js create mode 100644 tests/baselines/reference/declarationMapsMultifile.js.map create mode 100644 tests/baselines/reference/declarationMapsMultifile.sourcemap.txt create mode 100644 tests/baselines/reference/declarationMapsMultifile.symbols create mode 100644 tests/baselines/reference/declarationMapsMultifile.types create mode 100644 tests/baselines/reference/declarationMapsOutFile.js create mode 100644 tests/baselines/reference/declarationMapsOutFile.js.map create mode 100644 tests/baselines/reference/declarationMapsOutFile.sourcemap.txt create mode 100644 tests/baselines/reference/declarationMapsOutFile.symbols create mode 100644 tests/baselines/reference/declarationMapsOutFile.types create mode 100644 tests/baselines/reference/declarationMapsOutFile2.js create mode 100644 tests/baselines/reference/declarationMapsOutFile2.js.map create mode 100644 tests/baselines/reference/declarationMapsOutFile2.sourcemap.txt create mode 100644 tests/baselines/reference/declarationMapsOutFile2.symbols create mode 100644 tests/baselines/reference/declarationMapsOutFile2.types create mode 100644 tests/baselines/reference/declarationMapsWithSourceMap.js create mode 100644 tests/baselines/reference/declarationMapsWithSourceMap.js.map create mode 100644 tests/baselines/reference/declarationMapsWithSourceMap.sourcemap.txt create mode 100644 tests/baselines/reference/declarationMapsWithSourceMap.symbols create mode 100644 tests/baselines/reference/declarationMapsWithSourceMap.types create mode 100644 tests/baselines/reference/declarationMapsWithoutDeclaration.errors.txt create mode 100644 tests/baselines/reference/declarationMapsWithoutDeclaration.js create mode 100644 tests/baselines/reference/declarationMapsWithoutDeclaration.symbols create mode 100644 tests/baselines/reference/declarationMapsWithoutDeclaration.types create mode 100644 tests/cases/compiler/declarationMaps.ts create mode 100644 tests/cases/compiler/declarationMapsMultifile.ts create mode 100644 tests/cases/compiler/declarationMapsOutFile.ts create mode 100644 tests/cases/compiler/declarationMapsOutFile2.ts create mode 100644 tests/cases/compiler/declarationMapsWithSourceMap.ts create mode 100644 tests/cases/compiler/declarationMapsWithoutDeclaration.ts diff --git a/src/compiler/commandLineParser.ts b/src/compiler/commandLineParser.ts index 76a1d0b9db17a..9e805d80d3af1 100644 --- a/src/compiler/commandLineParser.ts +++ b/src/compiler/commandLineParser.ts @@ -194,6 +194,13 @@ namespace ts { category: Diagnostics.Basic_Options, description: Diagnostics.Generates_corresponding_d_ts_file, }, + { + name: "declarationMaps", + type: "boolean", + showInSimplifiedHelpView: true, + category: Diagnostics.Basic_Options, + description: Diagnostics.Generates_a_sourcemap_for_each_corresponding_d_ts_file, + }, { name: "emitDeclarationOnly", type: "boolean", diff --git a/src/compiler/diagnosticMessages.json b/src/compiler/diagnosticMessages.json index f4f7587b78ac9..6f96ac18398da 100644 --- a/src/compiler/diagnosticMessages.json +++ b/src/compiler/diagnosticMessages.json @@ -2768,6 +2768,10 @@ "code": 5068 }, + "Generates a sourcemap for each corresponding '.d.ts' file.": { + "category": "Message", + "code": 6000 + }, "Concatenate and emit output to single file.": { "category": "Message", "code": 6001 diff --git a/src/compiler/emitter.ts b/src/compiler/emitter.ts index a695b55f5ca8a..fd9f0d1dd4eba 100644 --- a/src/compiler/emitter.ts +++ b/src/compiler/emitter.ts @@ -49,7 +49,8 @@ namespace ts { const jsFilePath = options.outFile || options.out; const sourceMapFilePath = getSourceMapFilePath(jsFilePath, options); const declarationFilePath = (forceDtsPaths || options.declaration) ? removeFileExtension(jsFilePath) + Extension.Dts : undefined; - return { jsFilePath, sourceMapFilePath, declarationFilePath }; + const declarationMapPath = (declarationFilePath && options.declarationMaps) ? declarationFilePath + ".map" : undefined; + return { jsFilePath, sourceMapFilePath, declarationFilePath, declarationMapPath }; } else { const jsFilePath = getOwnEmitOutputFilePath(sourceFile, host, getOutputExtension(sourceFile, options)); @@ -57,7 +58,8 @@ namespace ts { // For legacy reasons (ie, we have baselines capturing the behavior), js files don't report a .d.ts output path - this would only matter if `declaration` and `allowJs` were both on, which is currently an error const isJs = isSourceFileJavaScript(sourceFile); const declarationFilePath = ((forceDtsPaths || options.declaration) && !isJs) ? getDeclarationEmitOutputFilePath(sourceFile, host) : undefined; - return { jsFilePath, sourceMapFilePath, declarationFilePath }; + const declarationMapPath = (declarationFilePath && options.declarationMaps) ? declarationFilePath + ".map" : undefined; + return { jsFilePath, sourceMapFilePath, declarationFilePath, declarationMapPath }; } } @@ -83,12 +85,19 @@ namespace ts { return Extension.Js; } + const enum SourceMapEmitKind { + None, + File, + Inline, + DeclarationFile + } + /*@internal*/ // targetSourceFile is when users only want one file in entire project to be emitted. This is used in compileOnSave feature export function emitFiles(resolver: EmitResolver, host: EmitHost, targetSourceFile: SourceFile, emitOnlyDtsFiles?: boolean, transformers?: TransformerFactory[]): EmitResult { const compilerOptions = host.getCompilerOptions(); const moduleKind = getEmitModuleKind(compilerOptions); - const sourceMapDataList: SourceMapData[] = compilerOptions.sourceMap || compilerOptions.inlineSourceMap ? [] : undefined; + const sourceMapDataList: SourceMapData[] = (compilerOptions.sourceMap || compilerOptions.inlineSourceMap || compilerOptions.declarationMaps) ? [] : undefined; const emittedFilesList: string[] = compilerOptions.listEmittedFiles ? [] : undefined; const emitterDiagnostics = createDiagnosticCollection(); const newLine = host.getNewLine(); @@ -162,7 +171,8 @@ namespace ts { onSetSourceFile: setSourceFile, }); - printSourceFileOrBundle(jsFilePath, sourceMapFilePath, isSourceFile(sourceFileOrBundle) ? transform.transformed[0] : createBundle(transform.transformed), printer); + const sourcemapKind = compilerOptions.inlineSourceMap ? SourceMapEmitKind.Inline : compilerOptions.sourceMap ? SourceMapEmitKind.File : SourceMapEmitKind.None; + printSourceFileOrBundle(jsFilePath, sourceMapFilePath, isSourceFile(sourceFileOrBundle) ? transform.transformed[0] : createBundle(transform.transformed), printer, sourcemapKind); // Clean up emit nodes on parse tree transform.dispose(); @@ -186,6 +196,11 @@ namespace ts { // resolver hooks hasGlobalName: resolver.hasGlobalName, + // sourcemap hooks + onEmitSourceMapOfNode: sourceMap.emitNodeWithSourceMap, + onEmitSourceMapOfToken: sourceMap.emitTokenWithSourceMap, + onEmitSourceMapOfPosition: sourceMap.emitPos, + // transform hooks onEmitNode: declarationTransform.emitNodeWithNotification, substituteNode: declarationTransform.substituteNode, @@ -194,13 +209,13 @@ namespace ts { emitSkipped = emitSkipped || declBlocked; if (!declBlocked || emitOnlyDtsFiles) { const previousState = sourceMap.setState(/*disabled*/ true); - printSourceFileOrBundle(declarationFilePath, /*sourceMapFilePath*/ undefined, declarationTransform.transformed[0], declarationPrinter, /*shouldSkipSourcemap*/ true); + printSourceFileOrBundle(declarationFilePath, /*sourceMapFilePath*/ undefined, declarationTransform.transformed[0], declarationPrinter, declarationFilePath ? SourceMapEmitKind.DeclarationFile : SourceMapEmitKind.None); sourceMap.setState(previousState); } declarationTransform.dispose(); } - function printSourceFileOrBundle(jsFilePath: string, sourceMapFilePath: string, sourceFileOrBundle: SourceFile | Bundle, printer: Printer, shouldSkipSourcemap = false) { + function printSourceFileOrBundle(jsFilePath: string, sourceMapFilePath: string, sourceFileOrBundle: SourceFile | Bundle, printer: Printer, sourcemapKind: SourceMapEmitKind) { const bundle = sourceFileOrBundle.kind === SyntaxKind.Bundle ? sourceFileOrBundle : undefined; const sourceFile = sourceFileOrBundle.kind === SyntaxKind.SourceFile ? sourceFileOrBundle : undefined; const sourceFiles = bundle ? bundle.sourceFiles : [sourceFile]; @@ -219,17 +234,17 @@ namespace ts { writer.writeLine(); const sourceMappingURL = sourceMap.getSourceMappingURL(); - if (!shouldSkipSourcemap && sourceMappingURL) { + if (sourceMappingURL) { writer.write(`//# ${"sourceMappingURL"}=${sourceMappingURL}`); // Sometimes tools can sometimes see this line as a source mapping url comment } // Write the source map - if (!shouldSkipSourcemap && compilerOptions.sourceMap && !compilerOptions.inlineSourceMap) { + if (sourceMapFilePath && (sourcemapKind === SourceMapEmitKind.File || sourcemapKind === SourceMapEmitKind.DeclarationFile)) { writeFile(host, emitterDiagnostics, sourceMapFilePath, sourceMap.getText(), /*writeByteOrderMark*/ false, sourceFiles); } // Record source map data for the test harness. - if (!shouldSkipSourcemap && sourceMapDataList) { + if (sourcemapKind !== SourceMapEmitKind.None) { sourceMapDataList.push(sourceMap.getSourceMapData()); } diff --git a/src/compiler/program.ts b/src/compiler/program.ts index 35772b2f04086..2ef128e1b263c 100755 --- a/src/compiler/program.ts +++ b/src/compiler/program.ts @@ -2122,7 +2122,7 @@ namespace ts { createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_with_option_1, "out", "outFile"); } - if (options.mapRoot && !options.sourceMap) { + if (options.mapRoot && !(options.sourceMap || options.declarationMaps)) { // Error to specify --mapRoot without --sourcemap createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "mapRoot", "sourceMap"); } @@ -2136,6 +2136,12 @@ namespace ts { } } + if (options.declarationMaps) { + if (!options.declaration) { + createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "declarationMaps", "declaration"); + } + } + if (options.lib && options.noLib) { createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_with_option_1, "lib", "noLib"); } diff --git a/src/compiler/types.ts b/src/compiler/types.ts index 0dca91f3fd445..b9a903a190d3b 100644 --- a/src/compiler/types.ts +++ b/src/compiler/types.ts @@ -4085,6 +4085,7 @@ namespace ts { /** configFile is set as non enumerable property so as to avoid checking of json source files */ /* @internal */ readonly configFile?: JsonSourceFile; declaration?: boolean; + declarationMaps?: boolean; emitDeclarationOnly?: boolean; declarationDir?: string; /* @internal */ diagnostics?: boolean; diff --git a/src/compiler/utilities.ts b/src/compiler/utilities.ts index 4077d33933e11..cde23352c3636 100644 --- a/src/compiler/utilities.ts +++ b/src/compiler/utilities.ts @@ -2807,7 +2807,8 @@ namespace ts { export interface EmitFileNames { jsFilePath: string; sourceMapFilePath: string; - declarationFilePath: string; + declarationFilePath: string | undefined; + declarationMapPath: string | undefined; } /** diff --git a/src/harness/compilerRunner.ts b/src/harness/compilerRunner.ts index 30c5e47949f44..630116d6a6fb7 100644 --- a/src/harness/compilerRunner.ts +++ b/src/harness/compilerRunner.ts @@ -148,7 +148,7 @@ class CompilerBaselineRunner extends RunnerBase { // Source maps? it("Correct sourcemap content for " + fileName, () => { - if (options.sourceMap || options.inlineSourceMap) { + if (options.sourceMap || options.inlineSourceMap || options.declarationMaps) { Harness.Baseline.runBaseline(justName.replace(/\.tsx?$/, ".sourcemap.txt"), () => { const record = result.getSourceMapRecord(); if ((options.noEmitOnError && result.errors.length !== 0) || record === undefined) { diff --git a/src/harness/harness.ts b/src/harness/harness.ts index 424ee932541a4..d850faff8273c 100644 --- a/src/harness/harness.ts +++ b/src/harness/harness.ts @@ -1633,13 +1633,13 @@ namespace Harness { export function doSourcemapBaseline(baselinePath: string, options: ts.CompilerOptions, result: CompilerResult, harnessSettings: TestCaseParser.CompilerSettings) { if (options.inlineSourceMap) { - if (result.sourceMaps.length > 0) { + if (result.sourceMaps.length > 0 && !options.declarationMaps) { throw new Error("No sourcemap files should be generated if inlineSourceMaps was set."); } return; } - else if (options.sourceMap) { - if (result.sourceMaps.length !== result.files.length) { + else if (options.sourceMap || options.declarationMaps) { + if (result.sourceMaps.length !== (result.files.length * (options.declarationMaps && options.sourceMap ? 2 : 1))) { throw new Error("Number of sourcemap files should be same as js files."); } @@ -1806,6 +1806,10 @@ namespace Harness { return ts.endsWith(fileName, ".js.map") || ts.endsWith(fileName, ".jsx.map"); } + export function isDTSMap(fileName: string) { + return ts.endsWith(fileName, ".d.ts.map"); + } + /** Contains the code and errors of a compilation and some helper methods to check its status. */ export class CompilerResult { public files: GeneratedFile[] = []; @@ -1826,7 +1830,7 @@ namespace Harness { // .js file, add to files this.files.push(emittedFile); } - else if (isJSMap(emittedFile.fileName)) { + else if (isJSMap(emittedFile.fileName) || isDTSMap(emittedFile.fileName)) { this.sourceMaps.push(emittedFile); } else { diff --git a/src/harness/sourceMapRecorder.ts b/src/harness/sourceMapRecorder.ts index 606d4e67acdf3..78a0fe7969df1 100644 --- a/src/harness/sourceMapRecorder.ts +++ b/src/harness/sourceMapRecorder.ts @@ -434,14 +434,31 @@ namespace Harness.SourceMapRecorder { } } - export function getSourceMapRecord(sourceMapDataList: ts.SourceMapData[], program: ts.Program, jsFiles: Compiler.GeneratedFile[]) { + export function getSourceMapRecord(sourceMapDataList: ts.SourceMapData[], program: ts.Program, jsFiles: Compiler.GeneratedFile[], declarationFiles: Compiler.GeneratedFile[]) { const sourceMapRecorder = new Compiler.WriterAggregator(); for (let i = 0; i < sourceMapDataList.length; i++) { const sourceMapData = sourceMapDataList[i]; let prevSourceFile: ts.SourceFile; + let currentFile: Compiler.GeneratedFile; + if (ts.endsWith(sourceMapData.sourceMapFile, ts.Extension.Dts)) { + if (sourceMapDataList.length > jsFiles.length) { + currentFile = declarationFiles[Math.floor(i / 2)]; // When both kinds of source map are present, they alternate js/dts + } + else { + currentFile = declarationFiles[i]; + } + } + else { + if (sourceMapDataList.length > jsFiles.length) { + currentFile = jsFiles[Math.floor(i / 2)]; + } + else { + currentFile = jsFiles[i]; + } + } - SourceMapSpanWriter.initializeSourceMapSpanWriter(sourceMapRecorder, sourceMapData, jsFiles[i]); + SourceMapSpanWriter.initializeSourceMapSpanWriter(sourceMapRecorder, sourceMapData, currentFile); for (const decodedSourceMapping of sourceMapData.sourceMapDecodedMappings) { const currentSourceFile = program.getSourceFile(sourceMapData.inputSourceFileNames[decodedSourceMapping.sourceIndex]); if (currentSourceFile !== prevSourceFile) { diff --git a/tests/baselines/reference/api/tsserverlibrary.d.ts b/tests/baselines/reference/api/tsserverlibrary.d.ts index 3b4767a8a5db8..b1438ec7dffee 100644 --- a/tests/baselines/reference/api/tsserverlibrary.d.ts +++ b/tests/baselines/reference/api/tsserverlibrary.d.ts @@ -2302,6 +2302,7 @@ declare namespace ts { charset?: string; checkJs?: boolean; declaration?: boolean; + declarationMaps?: boolean; emitDeclarationOnly?: boolean; declarationDir?: string; disableSizeLimit?: boolean; diff --git a/tests/baselines/reference/api/typescript.d.ts b/tests/baselines/reference/api/typescript.d.ts index 4d17de3ca426d..b3464dfe8cc09 100644 --- a/tests/baselines/reference/api/typescript.d.ts +++ b/tests/baselines/reference/api/typescript.d.ts @@ -2302,6 +2302,7 @@ declare namespace ts { charset?: string; checkJs?: boolean; declaration?: boolean; + declarationMaps?: boolean; emitDeclarationOnly?: boolean; declarationDir?: string; disableSizeLimit?: boolean; diff --git a/tests/baselines/reference/declarationMaps.js b/tests/baselines/reference/declarationMaps.js new file mode 100644 index 0000000000000..1e8dc97d6c4bf --- /dev/null +++ b/tests/baselines/reference/declarationMaps.js @@ -0,0 +1,43 @@ +//// [declarationMaps.ts] +module m2 { + export interface connectModule { + (res, req, next): void; + } + export interface connectExport { + use: (mod: connectModule) => connectExport; + listen: (port: number) => void; + } + +} + +var m2: { + (): m2.connectExport; + test1: m2.connectModule; + test2(): m2.connectModule; +}; + +export = m2; + +//// [declarationMaps.js] +"use strict"; +var m2; +module.exports = m2; + + +//// [declarationMaps.d.ts] +declare module m2 { + interface connectModule { + (res: any, req: any, next: any): void; + } + interface connectExport { + use: (mod: connectModule) => connectExport; + listen: (port: number) => void; + } +} +declare var m2: { + (): m2.connectExport; + test1: m2.connectModule; + test2(): m2.connectModule; +}; +export = m2; +//# sourceMappingURL=declarationMaps.d.ts.map \ No newline at end of file diff --git a/tests/baselines/reference/declarationMaps.js.map b/tests/baselines/reference/declarationMaps.js.map new file mode 100644 index 0000000000000..973eb12afdab5 --- /dev/null +++ b/tests/baselines/reference/declarationMaps.js.map @@ -0,0 +1,2 @@ +//// [declarationMaps.d.ts.map] +{"version":3,"file":"declarationMaps.d.ts","sourceRoot":"","sources":["declarationMaps.ts"],"names":[],"mappings":"AAAA,eAAO,EAAE,CAAC;IACN,UAAiB,aAAa;QAC1B,CAAC,GAAG,KAAA,EAAE,GAAG,KAAA,EAAE,IAAI,KAAA,GAAG,IAAI,CAAC;KAC1B;IACD,UAAiB,aAAa;QAC1B,GAAG,EAAE,CAAC,GAAG,EAAE,aAAa,KAAK,aAAa,CAAC;QAC3C,MAAM,EAAE,CAAC,IAAI,EAAE,MAAM,KAAK,IAAI,CAAC;KAClC;CAEJ;AAED,QAAA,IAAI,EAAE,EAAE;IACJ,IAAI,EAAE,CAAC,aAAa,CAAC;IACrB,KAAK,EAAE,EAAE,CAAC,aAAa,CAAC;IACxB,KAAK,IAAI,EAAE,CAAC,aAAa,CAAC;CAC7B,CAAC;AAEF,SAAS,EAAE,CAAC"} \ No newline at end of file diff --git a/tests/baselines/reference/declarationMaps.sourcemap.txt b/tests/baselines/reference/declarationMaps.sourcemap.txt new file mode 100644 index 0000000000000..f1c4591aacb47 --- /dev/null +++ b/tests/baselines/reference/declarationMaps.sourcemap.txt @@ -0,0 +1,299 @@ +=================================================================== +JsFile: declarationMaps.d.ts +mapUrl: declarationMaps.d.ts.map +sourceRoot: +sources: declarationMaps.ts +=================================================================== +------------------------------------------------------------------- +emittedFile:tests/cases/compiler/declarationMaps.d.ts +sourceFile:declarationMaps.ts +------------------------------------------------------------------- +>>>declare module m2 { +1 > +2 >^^^^^^^^^^^^^^^ +3 > ^^ +4 > ^ +5 > ^^^^^^^^^^^^-> +1 > +2 >module +3 > m2 +4 > +1 >Emitted(1, 1) Source(1, 1) + SourceIndex(0) +2 >Emitted(1, 16) Source(1, 8) + SourceIndex(0) +3 >Emitted(1, 18) Source(1, 10) + SourceIndex(0) +4 >Emitted(1, 19) Source(1, 11) + SourceIndex(0) +--- +>>> interface connectModule { +1->^^^^ +2 > ^^^^^^^^^^ +3 > ^^^^^^^^^^^^^ +4 > ^^^^^^^^^^^^^^^^^^^^-> +1->{ + > +2 > export interface +3 > connectModule +1->Emitted(2, 5) Source(2, 5) + SourceIndex(0) +2 >Emitted(2, 15) Source(2, 22) + SourceIndex(0) +3 >Emitted(2, 28) Source(2, 35) + SourceIndex(0) +--- +>>> (res: any, req: any, next: any): void; +1->^^^^^^^^ +2 > ^ +3 > ^^^ +4 > ^^^^^ +5 > ^^ +6 > ^^^ +7 > ^^^^^ +8 > ^^ +9 > ^^^^ +10> ^^^^^ +11> ^^^ +12> ^^^^ +13> ^ +1-> { + > +2 > ( +3 > res +4 > +5 > , +6 > req +7 > +8 > , +9 > next +10> +11> ): +12> void +13> ; +1->Emitted(3, 9) Source(3, 9) + SourceIndex(0) +2 >Emitted(3, 10) Source(3, 10) + SourceIndex(0) +3 >Emitted(3, 13) Source(3, 13) + SourceIndex(0) +4 >Emitted(3, 18) Source(3, 13) + SourceIndex(0) +5 >Emitted(3, 20) Source(3, 15) + SourceIndex(0) +6 >Emitted(3, 23) Source(3, 18) + SourceIndex(0) +7 >Emitted(3, 28) Source(3, 18) + SourceIndex(0) +8 >Emitted(3, 30) Source(3, 20) + SourceIndex(0) +9 >Emitted(3, 34) Source(3, 24) + SourceIndex(0) +10>Emitted(3, 39) Source(3, 24) + SourceIndex(0) +11>Emitted(3, 42) Source(3, 27) + SourceIndex(0) +12>Emitted(3, 46) Source(3, 31) + SourceIndex(0) +13>Emitted(3, 47) Source(3, 32) + SourceIndex(0) +--- +>>> } +1 >^^^^^ +2 > ^^^^^^^^^^^^^^^^^^^^^^^^^-> +1 > + > } +1 >Emitted(4, 6) Source(4, 6) + SourceIndex(0) +--- +>>> interface connectExport { +1->^^^^ +2 > ^^^^^^^^^^ +3 > ^^^^^^^^^^^^^ +4 > ^^^^^^^^^^^^^^^^^^^^^^^^^-> +1-> + > +2 > export interface +3 > connectExport +1->Emitted(5, 5) Source(5, 5) + SourceIndex(0) +2 >Emitted(5, 15) Source(5, 22) + SourceIndex(0) +3 >Emitted(5, 28) Source(5, 35) + SourceIndex(0) +--- +>>> use: (mod: connectModule) => connectExport; +1->^^^^^^^^ +2 > ^^^ +3 > ^^ +4 > ^ +5 > ^^^ +6 > ^^ +7 > ^^^^^^^^^^^^^ +8 > ^^^^^ +9 > ^^^^^^^^^^^^^ +10> ^ +1-> { + > +2 > use +3 > : +4 > ( +5 > mod +6 > : +7 > connectModule +8 > ) => +9 > connectExport +10> ; +1->Emitted(6, 9) Source(6, 9) + SourceIndex(0) +2 >Emitted(6, 12) Source(6, 12) + SourceIndex(0) +3 >Emitted(6, 14) Source(6, 14) + SourceIndex(0) +4 >Emitted(6, 15) Source(6, 15) + SourceIndex(0) +5 >Emitted(6, 18) Source(6, 18) + SourceIndex(0) +6 >Emitted(6, 20) Source(6, 20) + SourceIndex(0) +7 >Emitted(6, 33) Source(6, 33) + SourceIndex(0) +8 >Emitted(6, 38) Source(6, 38) + SourceIndex(0) +9 >Emitted(6, 51) Source(6, 51) + SourceIndex(0) +10>Emitted(6, 52) Source(6, 52) + SourceIndex(0) +--- +>>> listen: (port: number) => void; +1 >^^^^^^^^ +2 > ^^^^^^ +3 > ^^ +4 > ^ +5 > ^^^^ +6 > ^^ +7 > ^^^^^^ +8 > ^^^^^ +9 > ^^^^ +10> ^ +1 > + > +2 > listen +3 > : +4 > ( +5 > port +6 > : +7 > number +8 > ) => +9 > void +10> ; +1 >Emitted(7, 9) Source(7, 9) + SourceIndex(0) +2 >Emitted(7, 15) Source(7, 15) + SourceIndex(0) +3 >Emitted(7, 17) Source(7, 17) + SourceIndex(0) +4 >Emitted(7, 18) Source(7, 18) + SourceIndex(0) +5 >Emitted(7, 22) Source(7, 22) + SourceIndex(0) +6 >Emitted(7, 24) Source(7, 24) + SourceIndex(0) +7 >Emitted(7, 30) Source(7, 30) + SourceIndex(0) +8 >Emitted(7, 35) Source(7, 35) + SourceIndex(0) +9 >Emitted(7, 39) Source(7, 39) + SourceIndex(0) +10>Emitted(7, 40) Source(7, 40) + SourceIndex(0) +--- +>>> } +1 >^^^^^ +1 > + > } +1 >Emitted(8, 6) Source(8, 6) + SourceIndex(0) +--- +>>>} +1 >^ +2 > ^^^^^^^^^^^^^^^^^-> +1 > + > + >} +1 >Emitted(9, 2) Source(10, 2) + SourceIndex(0) +--- +>>>declare var m2: { +1-> +2 >^^^^^^^^ +3 > ^^^^ +4 > ^^ +5 > ^^ +6 > ^^^^^^^^^^-> +1-> + > + > +2 > +3 > var +4 > m2 +5 > : +1->Emitted(10, 1) Source(12, 1) + SourceIndex(0) +2 >Emitted(10, 9) Source(12, 1) + SourceIndex(0) +3 >Emitted(10, 13) Source(12, 5) + SourceIndex(0) +4 >Emitted(10, 15) Source(12, 7) + SourceIndex(0) +5 >Emitted(10, 17) Source(12, 9) + SourceIndex(0) +--- +>>> (): m2.connectExport; +1->^^^^ +2 > ^^^^ +3 > ^^ +4 > ^ +5 > ^^^^^^^^^^^^^ +6 > ^ +7 > ^^^^-> +1->{ + > +2 > (): +3 > m2 +4 > . +5 > connectExport +6 > ; +1->Emitted(11, 5) Source(13, 5) + SourceIndex(0) +2 >Emitted(11, 9) Source(13, 9) + SourceIndex(0) +3 >Emitted(11, 11) Source(13, 11) + SourceIndex(0) +4 >Emitted(11, 12) Source(13, 12) + SourceIndex(0) +5 >Emitted(11, 25) Source(13, 25) + SourceIndex(0) +6 >Emitted(11, 26) Source(13, 26) + SourceIndex(0) +--- +>>> test1: m2.connectModule; +1->^^^^ +2 > ^^^^^ +3 > ^^ +4 > ^^ +5 > ^ +6 > ^^^^^^^^^^^^^ +7 > ^ +8 > ^^^-> +1-> + > +2 > test1 +3 > : +4 > m2 +5 > . +6 > connectModule +7 > ; +1->Emitted(12, 5) Source(14, 5) + SourceIndex(0) +2 >Emitted(12, 10) Source(14, 10) + SourceIndex(0) +3 >Emitted(12, 12) Source(14, 12) + SourceIndex(0) +4 >Emitted(12, 14) Source(14, 14) + SourceIndex(0) +5 >Emitted(12, 15) Source(14, 15) + SourceIndex(0) +6 >Emitted(12, 28) Source(14, 28) + SourceIndex(0) +7 >Emitted(12, 29) Source(14, 29) + SourceIndex(0) +--- +>>> test2(): m2.connectModule; +1->^^^^ +2 > ^^^^^ +3 > ^^^^ +4 > ^^ +5 > ^ +6 > ^^^^^^^^^^^^^ +7 > ^ +1-> + > +2 > test2 +3 > (): +4 > m2 +5 > . +6 > connectModule +7 > ; +1->Emitted(13, 5) Source(15, 5) + SourceIndex(0) +2 >Emitted(13, 10) Source(15, 10) + SourceIndex(0) +3 >Emitted(13, 14) Source(15, 14) + SourceIndex(0) +4 >Emitted(13, 16) Source(15, 16) + SourceIndex(0) +5 >Emitted(13, 17) Source(15, 17) + SourceIndex(0) +6 >Emitted(13, 30) Source(15, 30) + SourceIndex(0) +7 >Emitted(13, 31) Source(15, 31) + SourceIndex(0) +--- +>>>}; +1 >^ +2 > ^ +3 > ^^^^^^^^^^^-> +1 > + >} +2 > ; +1 >Emitted(14, 2) Source(16, 2) + SourceIndex(0) +2 >Emitted(14, 3) Source(16, 3) + SourceIndex(0) +--- +>>>export = m2; +1-> +2 >^^^^^^^^^ +3 > ^^ +4 > ^ +5 > ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^-> +1-> + > + > +2 >export = +3 > m2 +4 > ; +1->Emitted(15, 1) Source(18, 1) + SourceIndex(0) +2 >Emitted(15, 10) Source(18, 10) + SourceIndex(0) +3 >Emitted(15, 12) Source(18, 12) + SourceIndex(0) +4 >Emitted(15, 13) Source(18, 13) + SourceIndex(0) +--- +>>>//# sourceMappingURL=declarationMaps.d.ts.map \ No newline at end of file diff --git a/tests/baselines/reference/declarationMaps.symbols b/tests/baselines/reference/declarationMaps.symbols new file mode 100644 index 0000000000000..b7fb7a5555a4d --- /dev/null +++ b/tests/baselines/reference/declarationMaps.symbols @@ -0,0 +1,50 @@ +=== tests/cases/compiler/declarationMaps.ts === +module m2 { +>m2 : Symbol(m2, Decl(declarationMaps.ts, 0, 0), Decl(declarationMaps.ts, 11, 3)) + + export interface connectModule { +>connectModule : Symbol(connectModule, Decl(declarationMaps.ts, 0, 11)) + + (res, req, next): void; +>res : Symbol(res, Decl(declarationMaps.ts, 2, 9)) +>req : Symbol(req, Decl(declarationMaps.ts, 2, 13)) +>next : Symbol(next, Decl(declarationMaps.ts, 2, 18)) + } + export interface connectExport { +>connectExport : Symbol(connectExport, Decl(declarationMaps.ts, 3, 5)) + + use: (mod: connectModule) => connectExport; +>use : Symbol(connectExport.use, Decl(declarationMaps.ts, 4, 36)) +>mod : Symbol(mod, Decl(declarationMaps.ts, 5, 14)) +>connectModule : Symbol(connectModule, Decl(declarationMaps.ts, 0, 11)) +>connectExport : Symbol(connectExport, Decl(declarationMaps.ts, 3, 5)) + + listen: (port: number) => void; +>listen : Symbol(connectExport.listen, Decl(declarationMaps.ts, 5, 51)) +>port : Symbol(port, Decl(declarationMaps.ts, 6, 17)) + } + +} + +var m2: { +>m2 : Symbol(m2, Decl(declarationMaps.ts, 0, 0), Decl(declarationMaps.ts, 11, 3)) + + (): m2.connectExport; +>m2 : Symbol(m2, Decl(declarationMaps.ts, 0, 0), Decl(declarationMaps.ts, 11, 3)) +>connectExport : Symbol(m2.connectExport, Decl(declarationMaps.ts, 3, 5)) + + test1: m2.connectModule; +>test1 : Symbol(test1, Decl(declarationMaps.ts, 12, 25)) +>m2 : Symbol(m2, Decl(declarationMaps.ts, 0, 0), Decl(declarationMaps.ts, 11, 3)) +>connectModule : Symbol(m2.connectModule, Decl(declarationMaps.ts, 0, 11)) + + test2(): m2.connectModule; +>test2 : Symbol(test2, Decl(declarationMaps.ts, 13, 28)) +>m2 : Symbol(m2, Decl(declarationMaps.ts, 0, 0), Decl(declarationMaps.ts, 11, 3)) +>connectModule : Symbol(m2.connectModule, Decl(declarationMaps.ts, 0, 11)) + +}; + +export = m2; +>m2 : Symbol(m2, Decl(declarationMaps.ts, 0, 0), Decl(declarationMaps.ts, 11, 3)) + diff --git a/tests/baselines/reference/declarationMaps.types b/tests/baselines/reference/declarationMaps.types new file mode 100644 index 0000000000000..e66c2364c6b3b --- /dev/null +++ b/tests/baselines/reference/declarationMaps.types @@ -0,0 +1,50 @@ +=== tests/cases/compiler/declarationMaps.ts === +module m2 { +>m2 : { (): connectExport; test1: connectModule; test2(): connectModule; } + + export interface connectModule { +>connectModule : connectModule + + (res, req, next): void; +>res : any +>req : any +>next : any + } + export interface connectExport { +>connectExport : connectExport + + use: (mod: connectModule) => connectExport; +>use : (mod: connectModule) => connectExport +>mod : connectModule +>connectModule : connectModule +>connectExport : connectExport + + listen: (port: number) => void; +>listen : (port: number) => void +>port : number + } + +} + +var m2: { +>m2 : { (): m2.connectExport; test1: m2.connectModule; test2(): m2.connectModule; } + + (): m2.connectExport; +>m2 : any +>connectExport : m2.connectExport + + test1: m2.connectModule; +>test1 : m2.connectModule +>m2 : any +>connectModule : m2.connectModule + + test2(): m2.connectModule; +>test2 : () => m2.connectModule +>m2 : any +>connectModule : m2.connectModule + +}; + +export = m2; +>m2 : { (): m2.connectExport; test1: m2.connectModule; test2(): m2.connectModule; } + diff --git a/tests/baselines/reference/declarationMapsMultifile.js b/tests/baselines/reference/declarationMapsMultifile.js new file mode 100644 index 0000000000000..ee86a044f9d71 --- /dev/null +++ b/tests/baselines/reference/declarationMapsMultifile.js @@ -0,0 +1,64 @@ +//// [tests/cases/compiler/declarationMapsMultifile.ts] //// + +//// [a.ts] +export class Foo { + doThing(x: {a: number}) { + return {b: x.a}; + } + static make() { + return new Foo(); + } +} +//// [index.ts] +import {Foo} from "./a"; + +const c = new Foo(); +c.doThing({a: 42}); + +export let x = c.doThing({a: 12}); +export { c, Foo }; + + +//// [a.js] +"use strict"; +exports.__esModule = true; +var Foo = /** @class */ (function () { + function Foo() { + } + Foo.prototype.doThing = function (x) { + return { b: x.a }; + }; + Foo.make = function () { + return new Foo(); + }; + return Foo; +}()); +exports.Foo = Foo; +//// [index.js] +"use strict"; +exports.__esModule = true; +var a_1 = require("./a"); +exports.Foo = a_1.Foo; +var c = new a_1.Foo(); +exports.c = c; +c.doThing({ a: 42 }); +exports.x = c.doThing({ a: 12 }); + + +//// [a.d.ts] +export declare class Foo { + doThing(x: { + a: number; + }): { + b: number; + }; + static make(): Foo; +} +//# sourceMappingURL=a.d.ts.map//// [index.d.ts] +import { Foo } from "./a"; +declare const c: Foo; +export declare let x: { + b: number; +}; +export { c, Foo }; +//# sourceMappingURL=index.d.ts.map \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsMultifile.js.map b/tests/baselines/reference/declarationMapsMultifile.js.map new file mode 100644 index 0000000000000..6890407554a3d --- /dev/null +++ b/tests/baselines/reference/declarationMapsMultifile.js.map @@ -0,0 +1,3 @@ +//// [a.d.ts.map] +{"version":3,"file":"a.d.ts","sourceRoot":"","sources":["a.ts"],"names":[],"mappings":"AAAA;IACI,OAAO,CAAC,CAAC,EAAE;QAAC,CAAC,EAAE,MAAM,CAAA;KAAC;;;IAGtB,MAAM,CAAC,IAAI;CAGd"}//// [index.d.ts.map] +{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAC,GAAG,EAAC,MAAM,KAAK,CAAC;AAExB,QAAA,MAAM,CAAC,KAAY,CAAC;AAGpB,eAAO,IAAI,CAAC;;CAAqB,CAAC;AAClC,OAAO,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC"} \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsMultifile.sourcemap.txt b/tests/baselines/reference/declarationMapsMultifile.sourcemap.txt new file mode 100644 index 0000000000000..6d03fcbc60920 --- /dev/null +++ b/tests/baselines/reference/declarationMapsMultifile.sourcemap.txt @@ -0,0 +1,201 @@ +=================================================================== +JsFile: a.d.ts +mapUrl: a.d.ts.map +sourceRoot: +sources: a.ts +=================================================================== +------------------------------------------------------------------- +emittedFile:tests/cases/compiler/a.d.ts +sourceFile:a.ts +------------------------------------------------------------------- +>>>export declare class Foo { +1 > +2 >^^^^^^^^^^^^^^^^^-> +1 > +1 >Emitted(1, 1) Source(1, 1) + SourceIndex(0) +--- +>>> doThing(x: { +1->^^^^ +2 > ^^^^^^^ +3 > ^ +4 > ^ +5 > ^^ +6 > ^^^^-> +1->export class Foo { + > +2 > doThing +3 > ( +4 > x +5 > : +1->Emitted(2, 5) Source(2, 5) + SourceIndex(0) +2 >Emitted(2, 12) Source(2, 12) + SourceIndex(0) +3 >Emitted(2, 13) Source(2, 13) + SourceIndex(0) +4 >Emitted(2, 14) Source(2, 14) + SourceIndex(0) +5 >Emitted(2, 16) Source(2, 16) + SourceIndex(0) +--- +>>> a: number; +1->^^^^^^^^ +2 > ^ +3 > ^^ +4 > ^^^^^^ +5 > ^ +1->{ +2 > a +3 > : +4 > number +5 > +1->Emitted(3, 9) Source(2, 17) + SourceIndex(0) +2 >Emitted(3, 10) Source(2, 18) + SourceIndex(0) +3 >Emitted(3, 12) Source(2, 20) + SourceIndex(0) +4 >Emitted(3, 18) Source(2, 26) + SourceIndex(0) +5 >Emitted(3, 19) Source(2, 26) + SourceIndex(0) +--- +>>> }): { +1 >^^^^^ +2 > ^^^^^^^^^^^^^^-> +1 >} +1 >Emitted(4, 6) Source(2, 27) + SourceIndex(0) +--- +>>> b: number; +>>> }; +>>> static make(): Foo; +1->^^^^ +2 > ^^^^^^ +3 > ^ +4 > ^^^^ +1->) { + > return {b: x.a}; + > } + > +2 > static +3 > +4 > make +1->Emitted(7, 5) Source(5, 5) + SourceIndex(0) +2 >Emitted(7, 11) Source(5, 11) + SourceIndex(0) +3 >Emitted(7, 12) Source(5, 12) + SourceIndex(0) +4 >Emitted(7, 16) Source(5, 16) + SourceIndex(0) +--- +>>>} +1 >^ +2 > ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^-> +1 >() { + > return new Foo(); + > } + >} +1 >Emitted(8, 2) Source(8, 2) + SourceIndex(0) +--- +>>>//# sourceMappingURL=a.d.ts.map=================================================================== +JsFile: index.d.ts +mapUrl: index.d.ts.map +sourceRoot: +sources: index.ts +=================================================================== +------------------------------------------------------------------- +emittedFile:tests/cases/compiler/index.d.ts +sourceFile:index.ts +------------------------------------------------------------------- +>>>import { Foo } from "./a"; +1 > +2 >^^^^^^^ +3 > ^^ +4 > ^^^ +5 > ^^ +6 > ^^^^^^ +7 > ^^^^^ +8 > ^ +1 > +2 >import +3 > { +4 > Foo +5 > } +6 > from +7 > "./a" +8 > ; +1 >Emitted(1, 1) Source(1, 1) + SourceIndex(0) +2 >Emitted(1, 8) Source(1, 8) + SourceIndex(0) +3 >Emitted(1, 10) Source(1, 9) + SourceIndex(0) +4 >Emitted(1, 13) Source(1, 12) + SourceIndex(0) +5 >Emitted(1, 15) Source(1, 13) + SourceIndex(0) +6 >Emitted(1, 21) Source(1, 19) + SourceIndex(0) +7 >Emitted(1, 26) Source(1, 24) + SourceIndex(0) +8 >Emitted(1, 27) Source(1, 25) + SourceIndex(0) +--- +>>>declare const c: Foo; +1 > +2 >^^^^^^^^ +3 > ^^^^^^ +4 > ^ +5 > ^^^^^ +6 > ^ +7 > ^^^-> +1 > + > + > +2 > +3 > const +4 > c +5 > = new Foo() +6 > ; +1 >Emitted(2, 1) Source(3, 1) + SourceIndex(0) +2 >Emitted(2, 9) Source(3, 1) + SourceIndex(0) +3 >Emitted(2, 15) Source(3, 7) + SourceIndex(0) +4 >Emitted(2, 16) Source(3, 8) + SourceIndex(0) +5 >Emitted(2, 21) Source(3, 20) + SourceIndex(0) +6 >Emitted(2, 22) Source(3, 21) + SourceIndex(0) +--- +>>>export declare let x: { +1-> +2 >^^^^^^^^^^^^^^^ +3 > ^^^^ +4 > ^ +1-> + >c.doThing({a: 42}); + > + > +2 >export +3 > let +4 > x +1->Emitted(3, 1) Source(6, 1) + SourceIndex(0) +2 >Emitted(3, 16) Source(6, 8) + SourceIndex(0) +3 >Emitted(3, 20) Source(6, 12) + SourceIndex(0) +4 >Emitted(3, 21) Source(6, 13) + SourceIndex(0) +--- +>>> b: number; +>>>}; +1 >^ +2 > ^ +3 > ^^^^^^^^^^^^^^^^^-> +1 > = c.doThing({a: 12}) +2 > ; +1 >Emitted(5, 2) Source(6, 34) + SourceIndex(0) +2 >Emitted(5, 3) Source(6, 35) + SourceIndex(0) +--- +>>>export { c, Foo }; +1-> +2 >^^^^^^^ +3 > ^^ +4 > ^ +5 > ^^ +6 > ^^^ +7 > ^^ +8 > ^ +9 > ^^^^^^^^^^^^^^^^-> +1-> + > +2 >export +3 > { +4 > c +5 > , +6 > Foo +7 > } +8 > ; +1->Emitted(6, 1) Source(7, 1) + SourceIndex(0) +2 >Emitted(6, 8) Source(7, 8) + SourceIndex(0) +3 >Emitted(6, 10) Source(7, 10) + SourceIndex(0) +4 >Emitted(6, 11) Source(7, 11) + SourceIndex(0) +5 >Emitted(6, 13) Source(7, 13) + SourceIndex(0) +6 >Emitted(6, 16) Source(7, 16) + SourceIndex(0) +7 >Emitted(6, 18) Source(7, 18) + SourceIndex(0) +8 >Emitted(6, 19) Source(7, 19) + SourceIndex(0) +--- +>>>//# sourceMappingURL=index.d.ts.map \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsMultifile.symbols b/tests/baselines/reference/declarationMapsMultifile.symbols new file mode 100644 index 0000000000000..51efaafec3a5e --- /dev/null +++ b/tests/baselines/reference/declarationMapsMultifile.symbols @@ -0,0 +1,47 @@ +=== tests/cases/compiler/a.ts === +export class Foo { +>Foo : Symbol(Foo, Decl(a.ts, 0, 0)) + + doThing(x: {a: number}) { +>doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 18)) +>x : Symbol(x, Decl(a.ts, 1, 12)) +>a : Symbol(a, Decl(a.ts, 1, 16)) + + return {b: x.a}; +>b : Symbol(b, Decl(a.ts, 2, 16)) +>x.a : Symbol(a, Decl(a.ts, 1, 16)) +>x : Symbol(x, Decl(a.ts, 1, 12)) +>a : Symbol(a, Decl(a.ts, 1, 16)) + } + static make() { +>make : Symbol(Foo.make, Decl(a.ts, 3, 5)) + + return new Foo(); +>Foo : Symbol(Foo, Decl(a.ts, 0, 0)) + } +} +=== tests/cases/compiler/index.ts === +import {Foo} from "./a"; +>Foo : Symbol(Foo, Decl(index.ts, 0, 8)) + +const c = new Foo(); +>c : Symbol(c, Decl(index.ts, 2, 5)) +>Foo : Symbol(Foo, Decl(index.ts, 0, 8)) + +c.doThing({a: 42}); +>c.doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 18)) +>c : Symbol(c, Decl(index.ts, 2, 5)) +>doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 18)) +>a : Symbol(a, Decl(index.ts, 3, 11)) + +export let x = c.doThing({a: 12}); +>x : Symbol(x, Decl(index.ts, 5, 10)) +>c.doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 18)) +>c : Symbol(c, Decl(index.ts, 2, 5)) +>doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 18)) +>a : Symbol(a, Decl(index.ts, 5, 26)) + +export { c, Foo }; +>c : Symbol(c, Decl(index.ts, 6, 8)) +>Foo : Symbol(Foo, Decl(index.ts, 6, 11)) + diff --git a/tests/baselines/reference/declarationMapsMultifile.types b/tests/baselines/reference/declarationMapsMultifile.types new file mode 100644 index 0000000000000..f77a3983a4df9 --- /dev/null +++ b/tests/baselines/reference/declarationMapsMultifile.types @@ -0,0 +1,56 @@ +=== tests/cases/compiler/a.ts === +export class Foo { +>Foo : Foo + + doThing(x: {a: number}) { +>doThing : (x: { a: number; }) => { b: number; } +>x : { a: number; } +>a : number + + return {b: x.a}; +>{b: x.a} : { b: number; } +>b : number +>x.a : number +>x : { a: number; } +>a : number + } + static make() { +>make : () => Foo + + return new Foo(); +>new Foo() : Foo +>Foo : typeof Foo + } +} +=== tests/cases/compiler/index.ts === +import {Foo} from "./a"; +>Foo : typeof Foo + +const c = new Foo(); +>c : Foo +>new Foo() : Foo +>Foo : typeof Foo + +c.doThing({a: 42}); +>c.doThing({a: 42}) : { b: number; } +>c.doThing : (x: { a: number; }) => { b: number; } +>c : Foo +>doThing : (x: { a: number; }) => { b: number; } +>{a: 42} : { a: number; } +>a : number +>42 : 42 + +export let x = c.doThing({a: 12}); +>x : { b: number; } +>c.doThing({a: 12}) : { b: number; } +>c.doThing : (x: { a: number; }) => { b: number; } +>c : Foo +>doThing : (x: { a: number; }) => { b: number; } +>{a: 12} : { a: number; } +>a : number +>12 : 12 + +export { c, Foo }; +>c : Foo +>Foo : typeof Foo + diff --git a/tests/baselines/reference/declarationMapsOutFile.js b/tests/baselines/reference/declarationMapsOutFile.js new file mode 100644 index 0000000000000..def1f470a72bc --- /dev/null +++ b/tests/baselines/reference/declarationMapsOutFile.js @@ -0,0 +1,69 @@ +//// [tests/cases/compiler/declarationMapsOutFile.ts] //// + +//// [a.ts] +export class Foo { + doThing(x: {a: number}) { + return {b: x.a}; + } + static make() { + return new Foo(); + } +} +//// [index.ts] +import {Foo} from "./a"; + +const c = new Foo(); +c.doThing({a: 42}); + +export let x = c.doThing({a: 12}); +export { c, Foo }; + + +//// [bundle.js] +define("a", ["require", "exports"], function (require, exports) { + "use strict"; + exports.__esModule = true; + var Foo = /** @class */ (function () { + function Foo() { + } + Foo.prototype.doThing = function (x) { + return { b: x.a }; + }; + Foo.make = function () { + return new Foo(); + }; + return Foo; + }()); + exports.Foo = Foo; +}); +define("index", ["require", "exports", "a"], function (require, exports, a_1) { + "use strict"; + exports.__esModule = true; + exports.Foo = a_1.Foo; + var c = new a_1.Foo(); + exports.c = c; + c.doThing({ a: 42 }); + exports.x = c.doThing({ a: 12 }); +}); + + +//// [bundle.d.ts] +declare module "a" { + export class Foo { + doThing(x: { + a: number; + }): { + b: number; + }; + static make(): Foo; + } +} +declare module "index" { + import { Foo } from "a"; + const c: Foo; + export let x: { + b: number; + }; + export { c, Foo }; +} +//# sourceMappingURL=bundle.d.ts.map \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsOutFile.js.map b/tests/baselines/reference/declarationMapsOutFile.js.map new file mode 100644 index 0000000000000..0018f3f9f31d7 --- /dev/null +++ b/tests/baselines/reference/declarationMapsOutFile.js.map @@ -0,0 +1,2 @@ +//// [bundle.d.ts.map] +{"version":3,"file":"bundle.d.ts","sourceRoot":"","sources":["tests/cases/compiler/a.ts","tests/cases/compiler/index.ts"],"names":[],"mappings":";IAAA,MAAM;QACF,OAAO,CAAC,GAAG;YAAC,CAAC,EAAE,MAAM,CAAA;SAAC;;;QAGtB,MAAM,CAAC,IAAI;KAGd;;;ICPD,OAAO,EAAC,GAAG,EAAC,UAAY;IAExB,MAAM,CAAC,KAAY,CAAC;IAGpB,MAAM,CAAC,IAAI,CAAC;;KAAqB,CAAC;IAClC,OAAO,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC"} \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsOutFile.sourcemap.txt b/tests/baselines/reference/declarationMapsOutFile.sourcemap.txt new file mode 100644 index 0000000000000..f1090c4f765a2 --- /dev/null +++ b/tests/baselines/reference/declarationMapsOutFile.sourcemap.txt @@ -0,0 +1,192 @@ +=================================================================== +JsFile: bundle.d.ts +mapUrl: bundle.d.ts.map +sourceRoot: +sources: tests/cases/compiler/a.ts,tests/cases/compiler/index.ts +=================================================================== +------------------------------------------------------------------- +emittedFile:bundle.d.ts +sourceFile:tests/cases/compiler/a.ts +------------------------------------------------------------------- +>>>declare module "a" { +>>> export class Foo { +1 >^^^^ +2 > ^^^^^^ +3 > ^^^^^^^^^^^-> +1 > +2 > export +1 >Emitted(2, 5) Source(1, 1) + SourceIndex(0) +2 >Emitted(2, 11) Source(1, 7) + SourceIndex(0) +--- +>>> doThing(x: { +1->^^^^^^^^ +2 > ^^^^^^^ +3 > ^ +4 > ^^^ +5 > ^^^^-> +1-> class Foo { + > +2 > doThing +3 > ( +4 > x: +1->Emitted(3, 9) Source(2, 5) + SourceIndex(0) +2 >Emitted(3, 16) Source(2, 12) + SourceIndex(0) +3 >Emitted(3, 17) Source(2, 13) + SourceIndex(0) +4 >Emitted(3, 20) Source(2, 16) + SourceIndex(0) +--- +>>> a: number; +1->^^^^^^^^^^^^ +2 > ^ +3 > ^^ +4 > ^^^^^^ +5 > ^ +1->{ +2 > a +3 > : +4 > number +5 > +1->Emitted(4, 13) Source(2, 17) + SourceIndex(0) +2 >Emitted(4, 14) Source(2, 18) + SourceIndex(0) +3 >Emitted(4, 16) Source(2, 20) + SourceIndex(0) +4 >Emitted(4, 22) Source(2, 26) + SourceIndex(0) +5 >Emitted(4, 23) Source(2, 26) + SourceIndex(0) +--- +>>> }): { +1 >^^^^^^^^^ +2 > ^^^^^^^^^^^^^^-> +1 >} +1 >Emitted(5, 10) Source(2, 27) + SourceIndex(0) +--- +>>> b: number; +>>> }; +>>> static make(): Foo; +1->^^^^^^^^ +2 > ^^^^^^ +3 > ^ +4 > ^^^^ +1->) { + > return {b: x.a}; + > } + > +2 > static +3 > +4 > make +1->Emitted(8, 9) Source(5, 5) + SourceIndex(0) +2 >Emitted(8, 15) Source(5, 11) + SourceIndex(0) +3 >Emitted(8, 16) Source(5, 12) + SourceIndex(0) +4 >Emitted(8, 20) Source(5, 16) + SourceIndex(0) +--- +>>> } +1 >^^^^^ +1 >() { + > return new Foo(); + > } + >} +1 >Emitted(9, 6) Source(8, 2) + SourceIndex(0) +--- +------------------------------------------------------------------- +emittedFile:bundle.d.ts +sourceFile:tests/cases/compiler/index.ts +------------------------------------------------------------------- +>>>} +>>>declare module "index" { +>>> import { Foo } from "a"; +1 >^^^^ +2 > ^^^^^^^ +3 > ^^ +4 > ^^^ +5 > ^^ +6 > ^^^^^^^^^^ +1 > +2 > import +3 > { +4 > Foo +5 > } +6 > from "./a"; +1 >Emitted(12, 5) Source(1, 1) + SourceIndex(1) +2 >Emitted(12, 12) Source(1, 8) + SourceIndex(1) +3 >Emitted(12, 14) Source(1, 9) + SourceIndex(1) +4 >Emitted(12, 17) Source(1, 12) + SourceIndex(1) +5 >Emitted(12, 19) Source(1, 13) + SourceIndex(1) +6 >Emitted(12, 29) Source(1, 25) + SourceIndex(1) +--- +>>> const c: Foo; +1 >^^^^ +2 > ^^^^^^ +3 > ^ +4 > ^^^^^ +5 > ^ +6 > ^^^-> +1 > + > + > +2 > const +3 > c +4 > = new Foo() +5 > ; +1 >Emitted(13, 5) Source(3, 1) + SourceIndex(1) +2 >Emitted(13, 11) Source(3, 7) + SourceIndex(1) +3 >Emitted(13, 12) Source(3, 8) + SourceIndex(1) +4 >Emitted(13, 17) Source(3, 20) + SourceIndex(1) +5 >Emitted(13, 18) Source(3, 21) + SourceIndex(1) +--- +>>> export let x: { +1->^^^^ +2 > ^^^^^^ +3 > ^ +4 > ^^^^ +5 > ^ +6 > ^^^-> +1-> + >c.doThing({a: 42}); + > + > +2 > export +3 > +4 > let +5 > x +1->Emitted(14, 5) Source(6, 1) + SourceIndex(1) +2 >Emitted(14, 11) Source(6, 7) + SourceIndex(1) +3 >Emitted(14, 12) Source(6, 8) + SourceIndex(1) +4 >Emitted(14, 16) Source(6, 12) + SourceIndex(1) +5 >Emitted(14, 17) Source(6, 13) + SourceIndex(1) +--- +>>> b: number; +>>> }; +1->^^^^^ +2 > ^ +3 > ^^^^^^^^^^^^^^^^^-> +1-> = c.doThing({a: 12}) +2 > ; +1->Emitted(16, 6) Source(6, 34) + SourceIndex(1) +2 >Emitted(16, 7) Source(6, 35) + SourceIndex(1) +--- +>>> export { c, Foo }; +1->^^^^ +2 > ^^^^^^^ +3 > ^^ +4 > ^ +5 > ^^ +6 > ^^^ +7 > ^^ +8 > ^ +1-> + > +2 > export +3 > { +4 > c +5 > , +6 > Foo +7 > } +8 > ; +1->Emitted(17, 5) Source(7, 1) + SourceIndex(1) +2 >Emitted(17, 12) Source(7, 8) + SourceIndex(1) +3 >Emitted(17, 14) Source(7, 10) + SourceIndex(1) +4 >Emitted(17, 15) Source(7, 11) + SourceIndex(1) +5 >Emitted(17, 17) Source(7, 13) + SourceIndex(1) +6 >Emitted(17, 20) Source(7, 16) + SourceIndex(1) +7 >Emitted(17, 22) Source(7, 18) + SourceIndex(1) +8 >Emitted(17, 23) Source(7, 19) + SourceIndex(1) +--- +>>>} +>>>//# sourceMappingURL=bundle.d.ts.map \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsOutFile.symbols b/tests/baselines/reference/declarationMapsOutFile.symbols new file mode 100644 index 0000000000000..51efaafec3a5e --- /dev/null +++ b/tests/baselines/reference/declarationMapsOutFile.symbols @@ -0,0 +1,47 @@ +=== tests/cases/compiler/a.ts === +export class Foo { +>Foo : Symbol(Foo, Decl(a.ts, 0, 0)) + + doThing(x: {a: number}) { +>doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 18)) +>x : Symbol(x, Decl(a.ts, 1, 12)) +>a : Symbol(a, Decl(a.ts, 1, 16)) + + return {b: x.a}; +>b : Symbol(b, Decl(a.ts, 2, 16)) +>x.a : Symbol(a, Decl(a.ts, 1, 16)) +>x : Symbol(x, Decl(a.ts, 1, 12)) +>a : Symbol(a, Decl(a.ts, 1, 16)) + } + static make() { +>make : Symbol(Foo.make, Decl(a.ts, 3, 5)) + + return new Foo(); +>Foo : Symbol(Foo, Decl(a.ts, 0, 0)) + } +} +=== tests/cases/compiler/index.ts === +import {Foo} from "./a"; +>Foo : Symbol(Foo, Decl(index.ts, 0, 8)) + +const c = new Foo(); +>c : Symbol(c, Decl(index.ts, 2, 5)) +>Foo : Symbol(Foo, Decl(index.ts, 0, 8)) + +c.doThing({a: 42}); +>c.doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 18)) +>c : Symbol(c, Decl(index.ts, 2, 5)) +>doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 18)) +>a : Symbol(a, Decl(index.ts, 3, 11)) + +export let x = c.doThing({a: 12}); +>x : Symbol(x, Decl(index.ts, 5, 10)) +>c.doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 18)) +>c : Symbol(c, Decl(index.ts, 2, 5)) +>doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 18)) +>a : Symbol(a, Decl(index.ts, 5, 26)) + +export { c, Foo }; +>c : Symbol(c, Decl(index.ts, 6, 8)) +>Foo : Symbol(Foo, Decl(index.ts, 6, 11)) + diff --git a/tests/baselines/reference/declarationMapsOutFile.types b/tests/baselines/reference/declarationMapsOutFile.types new file mode 100644 index 0000000000000..f77a3983a4df9 --- /dev/null +++ b/tests/baselines/reference/declarationMapsOutFile.types @@ -0,0 +1,56 @@ +=== tests/cases/compiler/a.ts === +export class Foo { +>Foo : Foo + + doThing(x: {a: number}) { +>doThing : (x: { a: number; }) => { b: number; } +>x : { a: number; } +>a : number + + return {b: x.a}; +>{b: x.a} : { b: number; } +>b : number +>x.a : number +>x : { a: number; } +>a : number + } + static make() { +>make : () => Foo + + return new Foo(); +>new Foo() : Foo +>Foo : typeof Foo + } +} +=== tests/cases/compiler/index.ts === +import {Foo} from "./a"; +>Foo : typeof Foo + +const c = new Foo(); +>c : Foo +>new Foo() : Foo +>Foo : typeof Foo + +c.doThing({a: 42}); +>c.doThing({a: 42}) : { b: number; } +>c.doThing : (x: { a: number; }) => { b: number; } +>c : Foo +>doThing : (x: { a: number; }) => { b: number; } +>{a: 42} : { a: number; } +>a : number +>42 : 42 + +export let x = c.doThing({a: 12}); +>x : { b: number; } +>c.doThing({a: 12}) : { b: number; } +>c.doThing : (x: { a: number; }) => { b: number; } +>c : Foo +>doThing : (x: { a: number; }) => { b: number; } +>{a: 12} : { a: number; } +>a : number +>12 : 12 + +export { c, Foo }; +>c : Foo +>Foo : typeof Foo + diff --git a/tests/baselines/reference/declarationMapsOutFile2.js b/tests/baselines/reference/declarationMapsOutFile2.js new file mode 100644 index 0000000000000..42bb8db558672 --- /dev/null +++ b/tests/baselines/reference/declarationMapsOutFile2.js @@ -0,0 +1,49 @@ +//// [tests/cases/compiler/declarationMapsOutFile2.ts] //// + +//// [a.ts] +class Foo { + doThing(x: {a: number}) { + return {b: x.a}; + } + static make() { + return new Foo(); + } +} +//// [index.ts] +const c = new Foo(); +c.doThing({a: 42}); + +let x = c.doThing({a: 12}); + + +//// [bundle.js] +var Foo = /** @class */ (function () { + function Foo() { + } + Foo.prototype.doThing = function (x) { + return { b: x.a }; + }; + Foo.make = function () { + return new Foo(); + }; + return Foo; +}()); +var c = new Foo(); +c.doThing({ a: 42 }); +var x = c.doThing({ a: 12 }); + + +//// [bundle.d.ts] +declare class Foo { + doThing(x: { + a: number; + }): { + b: number; + }; + static make(): Foo; +} +declare const c: Foo; +declare let x: { + b: number; +}; +//# sourceMappingURL=bundle.d.ts.map \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsOutFile2.js.map b/tests/baselines/reference/declarationMapsOutFile2.js.map new file mode 100644 index 0000000000000..579077e0679cf --- /dev/null +++ b/tests/baselines/reference/declarationMapsOutFile2.js.map @@ -0,0 +1,2 @@ +//// [bundle.d.ts.map] +{"version":3,"file":"bundle.d.ts","sourceRoot":"","sources":["tests/cases/compiler/a.ts","tests/cases/compiler/index.ts"],"names":[],"mappings":"AAAA;IACI,OAAO,CAAC,GAAG;QAAC,CAAC,EAAE,MAAM,CAAA;KAAC;;;IAGtB,MAAM,CAAC,IAAI;CAGd;ACPD,QAAA,MAAM,CAAC,KAAY,CAAC;AAGpB,QAAA,IAAI,CAAC;;CAAqB,CAAC"} \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsOutFile2.sourcemap.txt b/tests/baselines/reference/declarationMapsOutFile2.sourcemap.txt new file mode 100644 index 0000000000000..469e5ff2bd992 --- /dev/null +++ b/tests/baselines/reference/declarationMapsOutFile2.sourcemap.txt @@ -0,0 +1,136 @@ +=================================================================== +JsFile: bundle.d.ts +mapUrl: bundle.d.ts.map +sourceRoot: +sources: tests/cases/compiler/a.ts,tests/cases/compiler/index.ts +=================================================================== +------------------------------------------------------------------- +emittedFile:bundle.d.ts +sourceFile:tests/cases/compiler/a.ts +------------------------------------------------------------------- +>>>declare class Foo { +1 > +2 >^^^^^^^^^^^^^^^^^-> +1 > +1 >Emitted(1, 1) Source(1, 1) + SourceIndex(0) +--- +>>> doThing(x: { +1->^^^^ +2 > ^^^^^^^ +3 > ^ +4 > ^^^ +5 > ^^^^-> +1->class Foo { + > +2 > doThing +3 > ( +4 > x: +1->Emitted(2, 5) Source(2, 5) + SourceIndex(0) +2 >Emitted(2, 12) Source(2, 12) + SourceIndex(0) +3 >Emitted(2, 13) Source(2, 13) + SourceIndex(0) +4 >Emitted(2, 16) Source(2, 16) + SourceIndex(0) +--- +>>> a: number; +1->^^^^^^^^ +2 > ^ +3 > ^^ +4 > ^^^^^^ +5 > ^ +1->{ +2 > a +3 > : +4 > number +5 > +1->Emitted(3, 9) Source(2, 17) + SourceIndex(0) +2 >Emitted(3, 10) Source(2, 18) + SourceIndex(0) +3 >Emitted(3, 12) Source(2, 20) + SourceIndex(0) +4 >Emitted(3, 18) Source(2, 26) + SourceIndex(0) +5 >Emitted(3, 19) Source(2, 26) + SourceIndex(0) +--- +>>> }): { +1 >^^^^^ +2 > ^^^^^^^^^^^^^^-> +1 >} +1 >Emitted(4, 6) Source(2, 27) + SourceIndex(0) +--- +>>> b: number; +>>> }; +>>> static make(): Foo; +1->^^^^ +2 > ^^^^^^ +3 > ^ +4 > ^^^^ +1->) { + > return {b: x.a}; + > } + > +2 > static +3 > +4 > make +1->Emitted(7, 5) Source(5, 5) + SourceIndex(0) +2 >Emitted(7, 11) Source(5, 11) + SourceIndex(0) +3 >Emitted(7, 12) Source(5, 12) + SourceIndex(0) +4 >Emitted(7, 16) Source(5, 16) + SourceIndex(0) +--- +>>>} +1 >^ +2 > ^^^^^^^^^^^^^^^^^^^^^-> +1 >() { + > return new Foo(); + > } + >} +1 >Emitted(8, 2) Source(8, 2) + SourceIndex(0) +--- +------------------------------------------------------------------- +emittedFile:bundle.d.ts +sourceFile:tests/cases/compiler/index.ts +------------------------------------------------------------------- +>>>declare const c: Foo; +1-> +2 >^^^^^^^^ +3 > ^^^^^^ +4 > ^ +5 > ^^^^^ +6 > ^ +1-> +2 > +3 > const +4 > c +5 > = new Foo() +6 > ; +1->Emitted(9, 1) Source(1, 1) + SourceIndex(1) +2 >Emitted(9, 9) Source(1, 1) + SourceIndex(1) +3 >Emitted(9, 15) Source(1, 7) + SourceIndex(1) +4 >Emitted(9, 16) Source(1, 8) + SourceIndex(1) +5 >Emitted(9, 21) Source(1, 20) + SourceIndex(1) +6 >Emitted(9, 22) Source(1, 21) + SourceIndex(1) +--- +>>>declare let x: { +1 > +2 >^^^^^^^^ +3 > ^^^^ +4 > ^ +5 > ^^-> +1 > + >c.doThing({a: 42}); + > + > +2 > +3 > let +4 > x +1 >Emitted(10, 1) Source(4, 1) + SourceIndex(1) +2 >Emitted(10, 9) Source(4, 1) + SourceIndex(1) +3 >Emitted(10, 13) Source(4, 5) + SourceIndex(1) +4 >Emitted(10, 14) Source(4, 6) + SourceIndex(1) +--- +>>> b: number; +>>>}; +1->^ +2 > ^ +3 > ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^-> +1-> = c.doThing({a: 12}) +2 > ; +1->Emitted(12, 2) Source(4, 27) + SourceIndex(1) +2 >Emitted(12, 3) Source(4, 28) + SourceIndex(1) +--- +>>>//# sourceMappingURL=bundle.d.ts.map \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsOutFile2.symbols b/tests/baselines/reference/declarationMapsOutFile2.symbols new file mode 100644 index 0000000000000..7ae52d3824513 --- /dev/null +++ b/tests/baselines/reference/declarationMapsOutFile2.symbols @@ -0,0 +1,40 @@ +=== tests/cases/compiler/a.ts === +class Foo { +>Foo : Symbol(Foo, Decl(a.ts, 0, 0)) + + doThing(x: {a: number}) { +>doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 11)) +>x : Symbol(x, Decl(a.ts, 1, 12)) +>a : Symbol(a, Decl(a.ts, 1, 16)) + + return {b: x.a}; +>b : Symbol(b, Decl(a.ts, 2, 16)) +>x.a : Symbol(a, Decl(a.ts, 1, 16)) +>x : Symbol(x, Decl(a.ts, 1, 12)) +>a : Symbol(a, Decl(a.ts, 1, 16)) + } + static make() { +>make : Symbol(Foo.make, Decl(a.ts, 3, 5)) + + return new Foo(); +>Foo : Symbol(Foo, Decl(a.ts, 0, 0)) + } +} +=== tests/cases/compiler/index.ts === +const c = new Foo(); +>c : Symbol(c, Decl(index.ts, 0, 5)) +>Foo : Symbol(Foo, Decl(a.ts, 0, 0)) + +c.doThing({a: 42}); +>c.doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 11)) +>c : Symbol(c, Decl(index.ts, 0, 5)) +>doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 11)) +>a : Symbol(a, Decl(index.ts, 1, 11)) + +let x = c.doThing({a: 12}); +>x : Symbol(x, Decl(index.ts, 3, 3)) +>c.doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 11)) +>c : Symbol(c, Decl(index.ts, 0, 5)) +>doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 11)) +>a : Symbol(a, Decl(index.ts, 3, 19)) + diff --git a/tests/baselines/reference/declarationMapsOutFile2.types b/tests/baselines/reference/declarationMapsOutFile2.types new file mode 100644 index 0000000000000..c4fc42619950d --- /dev/null +++ b/tests/baselines/reference/declarationMapsOutFile2.types @@ -0,0 +1,49 @@ +=== tests/cases/compiler/a.ts === +class Foo { +>Foo : Foo + + doThing(x: {a: number}) { +>doThing : (x: { a: number; }) => { b: number; } +>x : { a: number; } +>a : number + + return {b: x.a}; +>{b: x.a} : { b: number; } +>b : number +>x.a : number +>x : { a: number; } +>a : number + } + static make() { +>make : () => Foo + + return new Foo(); +>new Foo() : Foo +>Foo : typeof Foo + } +} +=== tests/cases/compiler/index.ts === +const c = new Foo(); +>c : Foo +>new Foo() : Foo +>Foo : typeof Foo + +c.doThing({a: 42}); +>c.doThing({a: 42}) : { b: number; } +>c.doThing : (x: { a: number; }) => { b: number; } +>c : Foo +>doThing : (x: { a: number; }) => { b: number; } +>{a: 42} : { a: number; } +>a : number +>42 : 42 + +let x = c.doThing({a: 12}); +>x : { b: number; } +>c.doThing({a: 12}) : { b: number; } +>c.doThing : (x: { a: number; }) => { b: number; } +>c : Foo +>doThing : (x: { a: number; }) => { b: number; } +>{a: 12} : { a: number; } +>a : number +>12 : 12 + diff --git a/tests/baselines/reference/declarationMapsWithSourceMap.js b/tests/baselines/reference/declarationMapsWithSourceMap.js new file mode 100644 index 0000000000000..38cbb61a10e85 --- /dev/null +++ b/tests/baselines/reference/declarationMapsWithSourceMap.js @@ -0,0 +1,49 @@ +//// [tests/cases/compiler/declarationMapsWithSourceMap.ts] //// + +//// [a.ts] +class Foo { + doThing(x: {a: number}) { + return {b: x.a}; + } + static make() { + return new Foo(); + } +} +//// [index.ts] +const c = new Foo(); +c.doThing({a: 42}); + +let x = c.doThing({a: 12}); + + +//// [bundle.js] +var Foo = /** @class */ (function () { + function Foo() { + } + Foo.prototype.doThing = function (x) { + return { b: x.a }; + }; + Foo.make = function () { + return new Foo(); + }; + return Foo; +}()); +var c = new Foo(); +c.doThing({ a: 42 }); +var x = c.doThing({ a: 12 }); +//# sourceMappingURL=bundle.js.map + +//// [bundle.d.ts] +declare class Foo { + doThing(x: { + a: number; + }): { + b: number; + }; + static make(): Foo; +} +declare const c: Foo; +declare let x: { + b: number; +}; +//# sourceMappingURL=bundle.d.ts.map \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsWithSourceMap.js.map b/tests/baselines/reference/declarationMapsWithSourceMap.js.map new file mode 100644 index 0000000000000..8c10d3f8c4caf --- /dev/null +++ b/tests/baselines/reference/declarationMapsWithSourceMap.js.map @@ -0,0 +1,3 @@ +//// [bundle.js.map] +{"version":3,"file":"bundle.js","sourceRoot":"","sources":["tests/cases/compiler/a.ts","tests/cases/compiler/index.ts"],"names":[],"mappings":"AAAA;IAAA;IAOA,CAAC;IANG,qBAAO,GAAP,UAAQ,CAAc;QAClB,MAAM,CAAC,EAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAC,CAAC;IACpB,CAAC;IACM,QAAI,GAAX;QACI,MAAM,CAAC,IAAI,GAAG,EAAE,CAAC;IACrB,CAAC;IACL,UAAC;AAAD,CAAC,AAPD,IAOC;ACPD,IAAM,CAAC,GAAG,IAAI,GAAG,EAAE,CAAC;AACpB,CAAC,CAAC,OAAO,CAAC,EAAC,CAAC,EAAE,EAAE,EAAC,CAAC,CAAC;AAEnB,IAAI,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,EAAC,CAAC,EAAE,EAAE,EAAC,CAAC,CAAC"}//// [bundle.d.ts.map] +{"version":3,"file":"bundle.d.ts","sourceRoot":"","sources":["tests/cases/compiler/a.ts","tests/cases/compiler/index.ts"],"names":[],"mappings":"AAAA;IACI,OAAO,CAAC,GAAG;QAAC,CAAC,EAAE,MAAM,CAAA;KAAC;;;IAGtB,MAAM,CAAC,IAAI;CAGd;ACPD,QAAA,MAAM,CAAC,KAAY,CAAC;AAGpB,QAAA,IAAI,CAAC;;CAAqB,CAAC"} \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsWithSourceMap.sourcemap.txt b/tests/baselines/reference/declarationMapsWithSourceMap.sourcemap.txt new file mode 100644 index 0000000000000..1ee94a6057a7e --- /dev/null +++ b/tests/baselines/reference/declarationMapsWithSourceMap.sourcemap.txt @@ -0,0 +1,436 @@ +=================================================================== +JsFile: bundle.js +mapUrl: bundle.js.map +sourceRoot: +sources: tests/cases/compiler/a.ts,tests/cases/compiler/index.ts +=================================================================== +------------------------------------------------------------------- +emittedFile:bundle.js +sourceFile:tests/cases/compiler/a.ts +------------------------------------------------------------------- +>>>var Foo = /** @class */ (function () { +1 > +2 >^^^^^^^^^^^^^^^^^^^^^-> +1 > +1 >Emitted(1, 1) Source(1, 1) + SourceIndex(0) +--- +>>> function Foo() { +1->^^^^ +2 > ^^-> +1-> +1->Emitted(2, 5) Source(1, 1) + SourceIndex(0) +--- +>>> } +1->^^^^ +2 > ^ +3 > ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^-> +1->class Foo { + > doThing(x: {a: number}) { + > return {b: x.a}; + > } + > static make() { + > return new Foo(); + > } + > +2 > } +1->Emitted(3, 5) Source(8, 1) + SourceIndex(0) +2 >Emitted(3, 6) Source(8, 2) + SourceIndex(0) +--- +>>> Foo.prototype.doThing = function (x) { +1->^^^^ +2 > ^^^^^^^^^^^^^^^^^^^^^ +3 > ^^^ +4 > ^^^^^^^^^^ +5 > ^ +1-> +2 > doThing +3 > +4 > doThing( +5 > x: {a: number} +1->Emitted(4, 5) Source(2, 5) + SourceIndex(0) +2 >Emitted(4, 26) Source(2, 12) + SourceIndex(0) +3 >Emitted(4, 29) Source(2, 5) + SourceIndex(0) +4 >Emitted(4, 39) Source(2, 13) + SourceIndex(0) +5 >Emitted(4, 40) Source(2, 27) + SourceIndex(0) +--- +>>> return { b: x.a }; +1 >^^^^^^^^ +2 > ^^^^^^ +3 > ^ +4 > ^^ +5 > ^ +6 > ^^ +7 > ^ +8 > ^ +9 > ^ +10> ^^ +11> ^ +1 >) { + > +2 > return +3 > +4 > { +5 > b +6 > : +7 > x +8 > . +9 > a +10> } +11> ; +1 >Emitted(5, 9) Source(3, 9) + SourceIndex(0) +2 >Emitted(5, 15) Source(3, 15) + SourceIndex(0) +3 >Emitted(5, 16) Source(3, 16) + SourceIndex(0) +4 >Emitted(5, 18) Source(3, 17) + SourceIndex(0) +5 >Emitted(5, 19) Source(3, 18) + SourceIndex(0) +6 >Emitted(5, 21) Source(3, 20) + SourceIndex(0) +7 >Emitted(5, 22) Source(3, 21) + SourceIndex(0) +8 >Emitted(5, 23) Source(3, 22) + SourceIndex(0) +9 >Emitted(5, 24) Source(3, 23) + SourceIndex(0) +10>Emitted(5, 26) Source(3, 24) + SourceIndex(0) +11>Emitted(5, 27) Source(3, 25) + SourceIndex(0) +--- +>>> }; +1 >^^^^ +2 > ^ +3 > ^^^^^^^^^^^^^^^^^^^^^^^^-> +1 > + > +2 > } +1 >Emitted(6, 5) Source(4, 5) + SourceIndex(0) +2 >Emitted(6, 6) Source(4, 6) + SourceIndex(0) +--- +>>> Foo.make = function () { +1->^^^^ +2 > ^^^^^^^^ +3 > ^^^ +4 > ^^^^^^^^^^^-> +1-> + > static +2 > make +3 > +1->Emitted(7, 5) Source(5, 12) + SourceIndex(0) +2 >Emitted(7, 13) Source(5, 16) + SourceIndex(0) +3 >Emitted(7, 16) Source(5, 5) + SourceIndex(0) +--- +>>> return new Foo(); +1->^^^^^^^^ +2 > ^^^^^^ +3 > ^ +4 > ^^^^ +5 > ^^^ +6 > ^^ +7 > ^ +1->static make() { + > +2 > return +3 > +4 > new +5 > Foo +6 > () +7 > ; +1->Emitted(8, 9) Source(6, 9) + SourceIndex(0) +2 >Emitted(8, 15) Source(6, 15) + SourceIndex(0) +3 >Emitted(8, 16) Source(6, 16) + SourceIndex(0) +4 >Emitted(8, 20) Source(6, 20) + SourceIndex(0) +5 >Emitted(8, 23) Source(6, 23) + SourceIndex(0) +6 >Emitted(8, 25) Source(6, 25) + SourceIndex(0) +7 >Emitted(8, 26) Source(6, 26) + SourceIndex(0) +--- +>>> }; +1 >^^^^ +2 > ^ +3 > ^^^^^^^^^^^-> +1 > + > +2 > } +1 >Emitted(9, 5) Source(7, 5) + SourceIndex(0) +2 >Emitted(9, 6) Source(7, 6) + SourceIndex(0) +--- +>>> return Foo; +1->^^^^ +2 > ^^^^^^^^^^ +1-> + > +2 > } +1->Emitted(10, 5) Source(8, 1) + SourceIndex(0) +2 >Emitted(10, 15) Source(8, 2) + SourceIndex(0) +--- +>>>}()); +1 > +2 >^ +3 > +4 > ^^^^ +5 > ^^^^^^^^^^^^^^-> +1 > +2 >} +3 > +4 > class Foo { + > doThing(x: {a: number}) { + > return {b: x.a}; + > } + > static make() { + > return new Foo(); + > } + > } +1 >Emitted(11, 1) Source(8, 1) + SourceIndex(0) +2 >Emitted(11, 2) Source(8, 2) + SourceIndex(0) +3 >Emitted(11, 2) Source(1, 1) + SourceIndex(0) +4 >Emitted(11, 6) Source(8, 2) + SourceIndex(0) +--- +------------------------------------------------------------------- +emittedFile:bundle.js +sourceFile:tests/cases/compiler/index.ts +------------------------------------------------------------------- +>>>var c = new Foo(); +1-> +2 >^^^^ +3 > ^ +4 > ^^^ +5 > ^^^^ +6 > ^^^ +7 > ^^ +8 > ^ +9 > ^^^^-> +1-> +2 >const +3 > c +4 > = +5 > new +6 > Foo +7 > () +8 > ; +1->Emitted(12, 1) Source(1, 1) + SourceIndex(1) +2 >Emitted(12, 5) Source(1, 7) + SourceIndex(1) +3 >Emitted(12, 6) Source(1, 8) + SourceIndex(1) +4 >Emitted(12, 9) Source(1, 11) + SourceIndex(1) +5 >Emitted(12, 13) Source(1, 15) + SourceIndex(1) +6 >Emitted(12, 16) Source(1, 18) + SourceIndex(1) +7 >Emitted(12, 18) Source(1, 20) + SourceIndex(1) +8 >Emitted(12, 19) Source(1, 21) + SourceIndex(1) +--- +>>>c.doThing({ a: 42 }); +1-> +2 >^ +3 > ^ +4 > ^^^^^^^ +5 > ^ +6 > ^^ +7 > ^ +8 > ^^ +9 > ^^ +10> ^^ +11> ^ +12> ^ +13> ^^^^^^^^^-> +1-> + > +2 >c +3 > . +4 > doThing +5 > ( +6 > { +7 > a +8 > : +9 > 42 +10> } +11> ) +12> ; +1->Emitted(13, 1) Source(2, 1) + SourceIndex(1) +2 >Emitted(13, 2) Source(2, 2) + SourceIndex(1) +3 >Emitted(13, 3) Source(2, 3) + SourceIndex(1) +4 >Emitted(13, 10) Source(2, 10) + SourceIndex(1) +5 >Emitted(13, 11) Source(2, 11) + SourceIndex(1) +6 >Emitted(13, 13) Source(2, 12) + SourceIndex(1) +7 >Emitted(13, 14) Source(2, 13) + SourceIndex(1) +8 >Emitted(13, 16) Source(2, 15) + SourceIndex(1) +9 >Emitted(13, 18) Source(2, 17) + SourceIndex(1) +10>Emitted(13, 20) Source(2, 18) + SourceIndex(1) +11>Emitted(13, 21) Source(2, 19) + SourceIndex(1) +12>Emitted(13, 22) Source(2, 20) + SourceIndex(1) +--- +>>>var x = c.doThing({ a: 12 }); +1-> +2 >^^^^ +3 > ^ +4 > ^^^ +5 > ^ +6 > ^ +7 > ^^^^^^^ +8 > ^ +9 > ^^ +10> ^ +11> ^^ +12> ^^ +13> ^^ +14> ^ +15> ^ +16> ^^^^-> +1-> + > + > +2 >let +3 > x +4 > = +5 > c +6 > . +7 > doThing +8 > ( +9 > { +10> a +11> : +12> 12 +13> } +14> ) +15> ; +1->Emitted(14, 1) Source(4, 1) + SourceIndex(1) +2 >Emitted(14, 5) Source(4, 5) + SourceIndex(1) +3 >Emitted(14, 6) Source(4, 6) + SourceIndex(1) +4 >Emitted(14, 9) Source(4, 9) + SourceIndex(1) +5 >Emitted(14, 10) Source(4, 10) + SourceIndex(1) +6 >Emitted(14, 11) Source(4, 11) + SourceIndex(1) +7 >Emitted(14, 18) Source(4, 18) + SourceIndex(1) +8 >Emitted(14, 19) Source(4, 19) + SourceIndex(1) +9 >Emitted(14, 21) Source(4, 20) + SourceIndex(1) +10>Emitted(14, 22) Source(4, 21) + SourceIndex(1) +11>Emitted(14, 24) Source(4, 23) + SourceIndex(1) +12>Emitted(14, 26) Source(4, 25) + SourceIndex(1) +13>Emitted(14, 28) Source(4, 26) + SourceIndex(1) +14>Emitted(14, 29) Source(4, 27) + SourceIndex(1) +15>Emitted(14, 30) Source(4, 28) + SourceIndex(1) +--- +>>>//# sourceMappingURL=bundle.js.map=================================================================== +JsFile: bundle.d.ts +mapUrl: bundle.d.ts.map +sourceRoot: +sources: tests/cases/compiler/a.ts,tests/cases/compiler/index.ts +=================================================================== +------------------------------------------------------------------- +emittedFile:bundle.d.ts +sourceFile:tests/cases/compiler/a.ts +------------------------------------------------------------------- +>>>declare class Foo { +1 > +2 >^^^^^^^^^^^^^^^^^-> +1 > +1 >Emitted(1, 1) Source(1, 1) + SourceIndex(0) +--- +>>> doThing(x: { +1->^^^^ +2 > ^^^^^^^ +3 > ^ +4 > ^^^ +5 > ^^^^-> +1->class Foo { + > +2 > doThing +3 > ( +4 > x: +1->Emitted(2, 5) Source(2, 5) + SourceIndex(0) +2 >Emitted(2, 12) Source(2, 12) + SourceIndex(0) +3 >Emitted(2, 13) Source(2, 13) + SourceIndex(0) +4 >Emitted(2, 16) Source(2, 16) + SourceIndex(0) +--- +>>> a: number; +1->^^^^^^^^ +2 > ^ +3 > ^^ +4 > ^^^^^^ +5 > ^ +1->{ +2 > a +3 > : +4 > number +5 > +1->Emitted(3, 9) Source(2, 17) + SourceIndex(0) +2 >Emitted(3, 10) Source(2, 18) + SourceIndex(0) +3 >Emitted(3, 12) Source(2, 20) + SourceIndex(0) +4 >Emitted(3, 18) Source(2, 26) + SourceIndex(0) +5 >Emitted(3, 19) Source(2, 26) + SourceIndex(0) +--- +>>> }): { +1 >^^^^^ +2 > ^^^^^^^^^^^^^^-> +1 >} +1 >Emitted(4, 6) Source(2, 27) + SourceIndex(0) +--- +>>> b: number; +>>> }; +>>> static make(): Foo; +1->^^^^ +2 > ^^^^^^ +3 > ^ +4 > ^^^^ +1->) { + > return {b: x.a}; + > } + > +2 > static +3 > +4 > make +1->Emitted(7, 5) Source(5, 5) + SourceIndex(0) +2 >Emitted(7, 11) Source(5, 11) + SourceIndex(0) +3 >Emitted(7, 12) Source(5, 12) + SourceIndex(0) +4 >Emitted(7, 16) Source(5, 16) + SourceIndex(0) +--- +>>>} +1 >^ +2 > ^^^^^^^^^^^^^^^^^^^^^-> +1 >() { + > return new Foo(); + > } + >} +1 >Emitted(8, 2) Source(8, 2) + SourceIndex(0) +--- +------------------------------------------------------------------- +emittedFile:bundle.d.ts +sourceFile:tests/cases/compiler/index.ts +------------------------------------------------------------------- +>>>declare const c: Foo; +1-> +2 >^^^^^^^^ +3 > ^^^^^^ +4 > ^ +5 > ^^^^^ +6 > ^ +1-> +2 > +3 > const +4 > c +5 > = new Foo() +6 > ; +1->Emitted(9, 1) Source(1, 1) + SourceIndex(1) +2 >Emitted(9, 9) Source(1, 1) + SourceIndex(1) +3 >Emitted(9, 15) Source(1, 7) + SourceIndex(1) +4 >Emitted(9, 16) Source(1, 8) + SourceIndex(1) +5 >Emitted(9, 21) Source(1, 20) + SourceIndex(1) +6 >Emitted(9, 22) Source(1, 21) + SourceIndex(1) +--- +>>>declare let x: { +1 > +2 >^^^^^^^^ +3 > ^^^^ +4 > ^ +5 > ^^-> +1 > + >c.doThing({a: 42}); + > + > +2 > +3 > let +4 > x +1 >Emitted(10, 1) Source(4, 1) + SourceIndex(1) +2 >Emitted(10, 9) Source(4, 1) + SourceIndex(1) +3 >Emitted(10, 13) Source(4, 5) + SourceIndex(1) +4 >Emitted(10, 14) Source(4, 6) + SourceIndex(1) +--- +>>> b: number; +>>>}; +1->^ +2 > ^ +3 > ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^-> +1-> = c.doThing({a: 12}) +2 > ; +1->Emitted(12, 2) Source(4, 27) + SourceIndex(1) +2 >Emitted(12, 3) Source(4, 28) + SourceIndex(1) +--- +>>>//# sourceMappingURL=bundle.d.ts.map \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsWithSourceMap.symbols b/tests/baselines/reference/declarationMapsWithSourceMap.symbols new file mode 100644 index 0000000000000..7ae52d3824513 --- /dev/null +++ b/tests/baselines/reference/declarationMapsWithSourceMap.symbols @@ -0,0 +1,40 @@ +=== tests/cases/compiler/a.ts === +class Foo { +>Foo : Symbol(Foo, Decl(a.ts, 0, 0)) + + doThing(x: {a: number}) { +>doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 11)) +>x : Symbol(x, Decl(a.ts, 1, 12)) +>a : Symbol(a, Decl(a.ts, 1, 16)) + + return {b: x.a}; +>b : Symbol(b, Decl(a.ts, 2, 16)) +>x.a : Symbol(a, Decl(a.ts, 1, 16)) +>x : Symbol(x, Decl(a.ts, 1, 12)) +>a : Symbol(a, Decl(a.ts, 1, 16)) + } + static make() { +>make : Symbol(Foo.make, Decl(a.ts, 3, 5)) + + return new Foo(); +>Foo : Symbol(Foo, Decl(a.ts, 0, 0)) + } +} +=== tests/cases/compiler/index.ts === +const c = new Foo(); +>c : Symbol(c, Decl(index.ts, 0, 5)) +>Foo : Symbol(Foo, Decl(a.ts, 0, 0)) + +c.doThing({a: 42}); +>c.doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 11)) +>c : Symbol(c, Decl(index.ts, 0, 5)) +>doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 11)) +>a : Symbol(a, Decl(index.ts, 1, 11)) + +let x = c.doThing({a: 12}); +>x : Symbol(x, Decl(index.ts, 3, 3)) +>c.doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 11)) +>c : Symbol(c, Decl(index.ts, 0, 5)) +>doThing : Symbol(Foo.doThing, Decl(a.ts, 0, 11)) +>a : Symbol(a, Decl(index.ts, 3, 19)) + diff --git a/tests/baselines/reference/declarationMapsWithSourceMap.types b/tests/baselines/reference/declarationMapsWithSourceMap.types new file mode 100644 index 0000000000000..c4fc42619950d --- /dev/null +++ b/tests/baselines/reference/declarationMapsWithSourceMap.types @@ -0,0 +1,49 @@ +=== tests/cases/compiler/a.ts === +class Foo { +>Foo : Foo + + doThing(x: {a: number}) { +>doThing : (x: { a: number; }) => { b: number; } +>x : { a: number; } +>a : number + + return {b: x.a}; +>{b: x.a} : { b: number; } +>b : number +>x.a : number +>x : { a: number; } +>a : number + } + static make() { +>make : () => Foo + + return new Foo(); +>new Foo() : Foo +>Foo : typeof Foo + } +} +=== tests/cases/compiler/index.ts === +const c = new Foo(); +>c : Foo +>new Foo() : Foo +>Foo : typeof Foo + +c.doThing({a: 42}); +>c.doThing({a: 42}) : { b: number; } +>c.doThing : (x: { a: number; }) => { b: number; } +>c : Foo +>doThing : (x: { a: number; }) => { b: number; } +>{a: 42} : { a: number; } +>a : number +>42 : 42 + +let x = c.doThing({a: 12}); +>x : { b: number; } +>c.doThing({a: 12}) : { b: number; } +>c.doThing : (x: { a: number; }) => { b: number; } +>c : Foo +>doThing : (x: { a: number; }) => { b: number; } +>{a: 12} : { a: number; } +>a : number +>12 : 12 + diff --git a/tests/baselines/reference/declarationMapsWithoutDeclaration.errors.txt b/tests/baselines/reference/declarationMapsWithoutDeclaration.errors.txt new file mode 100644 index 0000000000000..15afcf6622e31 --- /dev/null +++ b/tests/baselines/reference/declarationMapsWithoutDeclaration.errors.txt @@ -0,0 +1,23 @@ +error TS5052: Option 'declarationMaps' cannot be specified without specifying option 'declaration'. + + +!!! error TS5052: Option 'declarationMaps' cannot be specified without specifying option 'declaration'. +==== tests/cases/compiler/declarationMapsWithoutDeclaration.ts (0 errors) ==== + module m2 { + export interface connectModule { + (res, req, next): void; + } + export interface connectExport { + use: (mod: connectModule) => connectExport; + listen: (port: number) => void; + } + + } + + var m2: { + (): m2.connectExport; + test1: m2.connectModule; + test2(): m2.connectModule; + }; + + export = m2; \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsWithoutDeclaration.js b/tests/baselines/reference/declarationMapsWithoutDeclaration.js new file mode 100644 index 0000000000000..c4f402cb4dfcb --- /dev/null +++ b/tests/baselines/reference/declarationMapsWithoutDeclaration.js @@ -0,0 +1,24 @@ +//// [declarationMapsWithoutDeclaration.ts] +module m2 { + export interface connectModule { + (res, req, next): void; + } + export interface connectExport { + use: (mod: connectModule) => connectExport; + listen: (port: number) => void; + } + +} + +var m2: { + (): m2.connectExport; + test1: m2.connectModule; + test2(): m2.connectModule; +}; + +export = m2; + +//// [declarationMapsWithoutDeclaration.js] +"use strict"; +var m2; +module.exports = m2; diff --git a/tests/baselines/reference/declarationMapsWithoutDeclaration.symbols b/tests/baselines/reference/declarationMapsWithoutDeclaration.symbols new file mode 100644 index 0000000000000..b74c089162722 --- /dev/null +++ b/tests/baselines/reference/declarationMapsWithoutDeclaration.symbols @@ -0,0 +1,50 @@ +=== tests/cases/compiler/declarationMapsWithoutDeclaration.ts === +module m2 { +>m2 : Symbol(m2, Decl(declarationMapsWithoutDeclaration.ts, 0, 0), Decl(declarationMapsWithoutDeclaration.ts, 11, 3)) + + export interface connectModule { +>connectModule : Symbol(connectModule, Decl(declarationMapsWithoutDeclaration.ts, 0, 11)) + + (res, req, next): void; +>res : Symbol(res, Decl(declarationMapsWithoutDeclaration.ts, 2, 9)) +>req : Symbol(req, Decl(declarationMapsWithoutDeclaration.ts, 2, 13)) +>next : Symbol(next, Decl(declarationMapsWithoutDeclaration.ts, 2, 18)) + } + export interface connectExport { +>connectExport : Symbol(connectExport, Decl(declarationMapsWithoutDeclaration.ts, 3, 5)) + + use: (mod: connectModule) => connectExport; +>use : Symbol(connectExport.use, Decl(declarationMapsWithoutDeclaration.ts, 4, 36)) +>mod : Symbol(mod, Decl(declarationMapsWithoutDeclaration.ts, 5, 14)) +>connectModule : Symbol(connectModule, Decl(declarationMapsWithoutDeclaration.ts, 0, 11)) +>connectExport : Symbol(connectExport, Decl(declarationMapsWithoutDeclaration.ts, 3, 5)) + + listen: (port: number) => void; +>listen : Symbol(connectExport.listen, Decl(declarationMapsWithoutDeclaration.ts, 5, 51)) +>port : Symbol(port, Decl(declarationMapsWithoutDeclaration.ts, 6, 17)) + } + +} + +var m2: { +>m2 : Symbol(m2, Decl(declarationMapsWithoutDeclaration.ts, 0, 0), Decl(declarationMapsWithoutDeclaration.ts, 11, 3)) + + (): m2.connectExport; +>m2 : Symbol(m2, Decl(declarationMapsWithoutDeclaration.ts, 0, 0), Decl(declarationMapsWithoutDeclaration.ts, 11, 3)) +>connectExport : Symbol(m2.connectExport, Decl(declarationMapsWithoutDeclaration.ts, 3, 5)) + + test1: m2.connectModule; +>test1 : Symbol(test1, Decl(declarationMapsWithoutDeclaration.ts, 12, 25)) +>m2 : Symbol(m2, Decl(declarationMapsWithoutDeclaration.ts, 0, 0), Decl(declarationMapsWithoutDeclaration.ts, 11, 3)) +>connectModule : Symbol(m2.connectModule, Decl(declarationMapsWithoutDeclaration.ts, 0, 11)) + + test2(): m2.connectModule; +>test2 : Symbol(test2, Decl(declarationMapsWithoutDeclaration.ts, 13, 28)) +>m2 : Symbol(m2, Decl(declarationMapsWithoutDeclaration.ts, 0, 0), Decl(declarationMapsWithoutDeclaration.ts, 11, 3)) +>connectModule : Symbol(m2.connectModule, Decl(declarationMapsWithoutDeclaration.ts, 0, 11)) + +}; + +export = m2; +>m2 : Symbol(m2, Decl(declarationMapsWithoutDeclaration.ts, 0, 0), Decl(declarationMapsWithoutDeclaration.ts, 11, 3)) + diff --git a/tests/baselines/reference/declarationMapsWithoutDeclaration.types b/tests/baselines/reference/declarationMapsWithoutDeclaration.types new file mode 100644 index 0000000000000..69b4b3e21e8b0 --- /dev/null +++ b/tests/baselines/reference/declarationMapsWithoutDeclaration.types @@ -0,0 +1,50 @@ +=== tests/cases/compiler/declarationMapsWithoutDeclaration.ts === +module m2 { +>m2 : { (): connectExport; test1: connectModule; test2(): connectModule; } + + export interface connectModule { +>connectModule : connectModule + + (res, req, next): void; +>res : any +>req : any +>next : any + } + export interface connectExport { +>connectExport : connectExport + + use: (mod: connectModule) => connectExport; +>use : (mod: connectModule) => connectExport +>mod : connectModule +>connectModule : connectModule +>connectExport : connectExport + + listen: (port: number) => void; +>listen : (port: number) => void +>port : number + } + +} + +var m2: { +>m2 : { (): m2.connectExport; test1: m2.connectModule; test2(): m2.connectModule; } + + (): m2.connectExport; +>m2 : any +>connectExport : m2.connectExport + + test1: m2.connectModule; +>test1 : m2.connectModule +>m2 : any +>connectModule : m2.connectModule + + test2(): m2.connectModule; +>test2 : () => m2.connectModule +>m2 : any +>connectModule : m2.connectModule + +}; + +export = m2; +>m2 : { (): m2.connectExport; test1: m2.connectModule; test2(): m2.connectModule; } + diff --git a/tests/baselines/reference/tsConfig/Default initialized TSConfig/tsconfig.json b/tests/baselines/reference/tsConfig/Default initialized TSConfig/tsconfig.json index cb57cd1366bb5..e4c98cac5676b 100644 --- a/tests/baselines/reference/tsConfig/Default initialized TSConfig/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Default initialized TSConfig/tsconfig.json @@ -8,6 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ + // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with boolean value compiler options/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with boolean value compiler options/tsconfig.json index 6614dd9e48e1a..3d3c62fa83773 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with boolean value compiler options/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with boolean value compiler options/tsconfig.json @@ -8,6 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ + // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with enum value compiler options/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with enum value compiler options/tsconfig.json index 8c19ad58bed7f..e413105cfa00f 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with enum value compiler options/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with enum value compiler options/tsconfig.json @@ -8,6 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ "jsx": "react", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ + // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with files options/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with files options/tsconfig.json index f5e2b163a21a3..9e941d4bd36c2 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with files options/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with files options/tsconfig.json @@ -8,6 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ + // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option value/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option value/tsconfig.json index ced3eae17b824..18abba4d3200b 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option value/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option value/tsconfig.json @@ -8,6 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ + // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option/tsconfig.json index cb57cd1366bb5..e4c98cac5676b 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option/tsconfig.json @@ -8,6 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ + // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options with enum value/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options with enum value/tsconfig.json index 1356e2258f7b4..dcb1a15965b7f 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options with enum value/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options with enum value/tsconfig.json @@ -8,6 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ + // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options/tsconfig.json index 473280f675ee1..df2966ad60534 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options/tsconfig.json @@ -8,6 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ + // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/cases/compiler/declarationMaps.ts b/tests/cases/compiler/declarationMaps.ts new file mode 100644 index 0000000000000..218f1c9394207 --- /dev/null +++ b/tests/cases/compiler/declarationMaps.ts @@ -0,0 +1,20 @@ +// @declaration: true +// @declarationMaps: true +module m2 { + export interface connectModule { + (res, req, next): void; + } + export interface connectExport { + use: (mod: connectModule) => connectExport; + listen: (port: number) => void; + } + +} + +var m2: { + (): m2.connectExport; + test1: m2.connectModule; + test2(): m2.connectModule; +}; + +export = m2; \ No newline at end of file diff --git a/tests/cases/compiler/declarationMapsMultifile.ts b/tests/cases/compiler/declarationMapsMultifile.ts new file mode 100644 index 0000000000000..f63ac88355f2c --- /dev/null +++ b/tests/cases/compiler/declarationMapsMultifile.ts @@ -0,0 +1,19 @@ +// @declaration: true +// @declarationMaps: true +// @filename: a.ts +export class Foo { + doThing(x: {a: number}) { + return {b: x.a}; + } + static make() { + return new Foo(); + } +} +// @filename: index.ts +import {Foo} from "./a"; + +const c = new Foo(); +c.doThing({a: 42}); + +export let x = c.doThing({a: 12}); +export { c, Foo }; diff --git a/tests/cases/compiler/declarationMapsOutFile.ts b/tests/cases/compiler/declarationMapsOutFile.ts new file mode 100644 index 0000000000000..4e047388bdacc --- /dev/null +++ b/tests/cases/compiler/declarationMapsOutFile.ts @@ -0,0 +1,21 @@ +// @declaration: true +// @declarationMaps: true +// @module: amd +// @outFile: bundle.js +// @filename: a.ts +export class Foo { + doThing(x: {a: number}) { + return {b: x.a}; + } + static make() { + return new Foo(); + } +} +// @filename: index.ts +import {Foo} from "./a"; + +const c = new Foo(); +c.doThing({a: 42}); + +export let x = c.doThing({a: 12}); +export { c, Foo }; diff --git a/tests/cases/compiler/declarationMapsOutFile2.ts b/tests/cases/compiler/declarationMapsOutFile2.ts new file mode 100644 index 0000000000000..6876db0e6ff88 --- /dev/null +++ b/tests/cases/compiler/declarationMapsOutFile2.ts @@ -0,0 +1,17 @@ +// @declaration: true +// @declarationMaps: true +// @outFile: bundle.js +// @filename: a.ts +class Foo { + doThing(x: {a: number}) { + return {b: x.a}; + } + static make() { + return new Foo(); + } +} +// @filename: index.ts +const c = new Foo(); +c.doThing({a: 42}); + +let x = c.doThing({a: 12}); diff --git a/tests/cases/compiler/declarationMapsWithSourceMap.ts b/tests/cases/compiler/declarationMapsWithSourceMap.ts new file mode 100644 index 0000000000000..498bcb41a973e --- /dev/null +++ b/tests/cases/compiler/declarationMapsWithSourceMap.ts @@ -0,0 +1,18 @@ +// @declaration: true +// @declarationMaps: true +// @outFile: bundle.js +// @sourceMap: true +// @filename: a.ts +class Foo { + doThing(x: {a: number}) { + return {b: x.a}; + } + static make() { + return new Foo(); + } +} +// @filename: index.ts +const c = new Foo(); +c.doThing({a: 42}); + +let x = c.doThing({a: 12}); diff --git a/tests/cases/compiler/declarationMapsWithoutDeclaration.ts b/tests/cases/compiler/declarationMapsWithoutDeclaration.ts new file mode 100644 index 0000000000000..c1911eb7b7efa --- /dev/null +++ b/tests/cases/compiler/declarationMapsWithoutDeclaration.ts @@ -0,0 +1,19 @@ +// @declarationMaps: true +module m2 { + export interface connectModule { + (res, req, next): void; + } + export interface connectExport { + use: (mod: connectModule) => connectExport; + listen: (port: number) => void; + } + +} + +var m2: { + (): m2.connectExport; + test1: m2.connectModule; + test2(): m2.connectModule; +}; + +export = m2; \ No newline at end of file From 91cc19cb9ae3063689f057ec1d87991845e722de Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Fri, 16 Feb 2018 17:26:44 -0800 Subject: [PATCH 02/32] Transparent goto definition for sourcemapped declaration files --- src/compiler/core.ts | 4 + src/compiler/emitter.ts | 16 +- src/compiler/scanner.ts | 2 +- src/harness/harness.ts | 7 +- src/services/services.ts | 66 +++- src/services/sourcemaps.ts | 342 ++++++++++++++++++ src/services/tsconfig.json | 1 + src/services/types.ts | 2 + .../reference/api/tsserverlibrary.d.ts | 2 +- tests/baselines/reference/api/typescript.d.ts | 2 +- .../fourslash/declarationMapGoToDefinition.ts | 40 ++ 11 files changed, 466 insertions(+), 18 deletions(-) create mode 100644 src/services/sourcemaps.ts create mode 100644 tests/cases/fourslash/declarationMapGoToDefinition.ts diff --git a/src/compiler/core.ts b/src/compiler/core.ts index 8f38f16c049e4..731af132d76b1 100644 --- a/src/compiler/core.ts +++ b/src/compiler/core.ts @@ -2066,6 +2066,10 @@ namespace ts { return moduleResolution; } + export function getAreDeclarationMapsEnabled(options: CompilerOptions) { + return !!(options.declaration && options.declarationMaps); + } + export function getAllowSyntheticDefaultImports(compilerOptions: CompilerOptions) { const moduleKind = getEmitModuleKind(compilerOptions); return compilerOptions.allowSyntheticDefaultImports !== undefined diff --git a/src/compiler/emitter.ts b/src/compiler/emitter.ts index fd9f0d1dd4eba..af5f38a6a3403 100644 --- a/src/compiler/emitter.ts +++ b/src/compiler/emitter.ts @@ -49,7 +49,7 @@ namespace ts { const jsFilePath = options.outFile || options.out; const sourceMapFilePath = getSourceMapFilePath(jsFilePath, options); const declarationFilePath = (forceDtsPaths || options.declaration) ? removeFileExtension(jsFilePath) + Extension.Dts : undefined; - const declarationMapPath = (declarationFilePath && options.declarationMaps) ? declarationFilePath + ".map" : undefined; + const declarationMapPath = getAreDeclarationMapsEnabled(options) ? declarationFilePath + ".map" : undefined; return { jsFilePath, sourceMapFilePath, declarationFilePath, declarationMapPath }; } else { @@ -58,7 +58,7 @@ namespace ts { // For legacy reasons (ie, we have baselines capturing the behavior), js files don't report a .d.ts output path - this would only matter if `declaration` and `allowJs` were both on, which is currently an error const isJs = isSourceFileJavaScript(sourceFile); const declarationFilePath = ((forceDtsPaths || options.declaration) && !isJs) ? getDeclarationEmitOutputFilePath(sourceFile, host) : undefined; - const declarationMapPath = (declarationFilePath && options.declarationMaps) ? declarationFilePath + ".map" : undefined; + const declarationMapPath = getAreDeclarationMapsEnabled(options) ? declarationFilePath + ".map" : undefined; return { jsFilePath, sourceMapFilePath, declarationFilePath, declarationMapPath }; } } @@ -97,7 +97,7 @@ namespace ts { export function emitFiles(resolver: EmitResolver, host: EmitHost, targetSourceFile: SourceFile, emitOnlyDtsFiles?: boolean, transformers?: TransformerFactory[]): EmitResult { const compilerOptions = host.getCompilerOptions(); const moduleKind = getEmitModuleKind(compilerOptions); - const sourceMapDataList: SourceMapData[] = (compilerOptions.sourceMap || compilerOptions.inlineSourceMap || compilerOptions.declarationMaps) ? [] : undefined; + const sourceMapDataList: SourceMapData[] = (compilerOptions.sourceMap || compilerOptions.inlineSourceMap || getAreDeclarationMapsEnabled(compilerOptions)) ? [] : undefined; const emittedFilesList: string[] = compilerOptions.listEmittedFiles ? [] : undefined; const emitterDiagnostics = createDiagnosticCollection(); const newLine = host.getNewLine(); @@ -122,9 +122,9 @@ namespace ts { sourceMaps: sourceMapDataList }; - function emitSourceFileOrBundle({ jsFilePath, sourceMapFilePath, declarationFilePath }: EmitFileNames, sourceFileOrBundle: SourceFile | Bundle) { + function emitSourceFileOrBundle({ jsFilePath, sourceMapFilePath, declarationFilePath, declarationMapPath }: EmitFileNames, sourceFileOrBundle: SourceFile | Bundle) { emitJsFileOrBundle(sourceFileOrBundle, jsFilePath, sourceMapFilePath); - emitDeclarationFileOrBundle(sourceFileOrBundle, declarationFilePath); + emitDeclarationFileOrBundle(sourceFileOrBundle, declarationFilePath, declarationMapPath); if (!emitSkipped && emittedFilesList) { if (!emitOnlyDtsFiles) { @@ -178,7 +178,7 @@ namespace ts { transform.dispose(); } - function emitDeclarationFileOrBundle(sourceFileOrBundle: SourceFile | Bundle, declarationFilePath: string | undefined) { + function emitDeclarationFileOrBundle(sourceFileOrBundle: SourceFile | Bundle, declarationFilePath: string | undefined, declarationMapPath: string | undefined) { if (!(declarationFilePath && !isInJavaScriptFile(sourceFileOrBundle))) { return; } @@ -208,9 +208,7 @@ namespace ts { const declBlocked = (!!declarationTransform.diagnostics && !!declarationTransform.diagnostics.length) || !!host.isEmitBlocked(declarationFilePath) || !!compilerOptions.noEmit; emitSkipped = emitSkipped || declBlocked; if (!declBlocked || emitOnlyDtsFiles) { - const previousState = sourceMap.setState(/*disabled*/ true); - printSourceFileOrBundle(declarationFilePath, /*sourceMapFilePath*/ undefined, declarationTransform.transformed[0], declarationPrinter, declarationFilePath ? SourceMapEmitKind.DeclarationFile : SourceMapEmitKind.None); - sourceMap.setState(previousState); + printSourceFileOrBundle(declarationFilePath, declarationMapPath, declarationTransform.transformed[0], declarationPrinter, getAreDeclarationMapsEnabled(compilerOptions) ? SourceMapEmitKind.DeclarationFile : SourceMapEmitKind.None); } declarationTransform.dispose(); } diff --git a/src/compiler/scanner.ts b/src/compiler/scanner.ts index 55dfe9be5eaf0..94b9b2146e5ab 100644 --- a/src/compiler/scanner.ts +++ b/src/compiler/scanner.ts @@ -330,7 +330,7 @@ namespace ts { return result; } - export function getPositionOfLineAndCharacter(sourceFile: SourceFile, line: number, character: number): number { + export function getPositionOfLineAndCharacter(sourceFile: SourceFileLike, line: number, character: number): number { return computePositionOfLineAndCharacter(getLineStarts(sourceFile), line, character, sourceFile.text); } diff --git a/src/harness/harness.ts b/src/harness/harness.ts index d850faff8273c..3465a7cef91db 100644 --- a/src/harness/harness.ts +++ b/src/harness/harness.ts @@ -1632,14 +1632,15 @@ namespace Harness { } export function doSourcemapBaseline(baselinePath: string, options: ts.CompilerOptions, result: CompilerResult, harnessSettings: TestCaseParser.CompilerSettings) { + const declMaps = ts.getAreDeclarationMapsEnabled(options); if (options.inlineSourceMap) { - if (result.sourceMaps.length > 0 && !options.declarationMaps) { + if (result.sourceMaps.length > 0 && !declMaps) { throw new Error("No sourcemap files should be generated if inlineSourceMaps was set."); } return; } - else if (options.sourceMap || options.declarationMaps) { - if (result.sourceMaps.length !== (result.files.length * (options.declarationMaps && options.sourceMap ? 2 : 1))) { + else if (options.sourceMap || declMaps) { + if (result.sourceMaps.length !== (result.files.length * (declMaps && options.sourceMap ? 2 : 1))) { throw new Error("Number of sourcemap files should be same as js files."); } diff --git a/src/services/services.ts b/src/services/services.ts index ed8e79a9069ee..541b440a87c5f 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -30,6 +30,7 @@ /// /// /// +/// namespace ts { /** The version of the language service API */ @@ -1536,20 +1537,79 @@ namespace ts { return checker.getSymbolAtLocation(node); } + function getSourceMapper(file: SourceFile) { + if (!host.readFile || !host.fileExists) { + return file.sourceMapper = sourcemaps.identitySourceMapper; + } + if (file.sourceMapper) { + return file.sourceMapper; + } + // TODO (wewigham): Read sourcemappingurl from last line of .d.ts if present + const mapFileName = file.fileName + ".map"; + if (!host.fileExists(mapFileName)) { + return file.sourceMapper = sourcemaps.identitySourceMapper; + } + const doc = host.readFile(mapFileName); + let maps: sourcemaps.SourceMapData; + try { + maps = JSON.parse(doc); + } + catch {} + if (!maps || !maps.sources || !maps.file || !maps.mappings) { + // obviously invalid map + return file.sourceMapper = sourcemaps.identitySourceMapper; + } + return file.sourceMapper = sourcemaps.decode({readFile: s => host.readFile(s), fileExists: s => host.fileExists(s), getCanonicalFileName}, mapFileName, maps, program); + } + + function getTargetOfMappedDeclarationFile(info: DefinitionInfo): DefinitionInfo { + if (endsWith(info.fileName, Extension.Dts)) { + const file = program.getSourceFile(info.fileName); + const mapper = getSourceMapper(file); + const mapLoc: sourcemaps.SourceMappableLocation = {fileName: info.fileName, position: info.textSpan.start}; + const newLoc = mapper.forwardMap(mapLoc); + if (newLoc === mapLoc) return info; + return { + containerKind: info.containerKind, + containerName: info.containerName, + fileName: newLoc.fileName, + kind: info.kind, + name: info.name, + textSpan: { + start: newLoc.position, + length: info.textSpan.length // TODO: How to determine correct length? mapping the end pos could result in a completely different location... + } + }; // TODO: recur if the target file is also a declaration file + } + return info; + } + + function getTargetOfMappedDeclarationFiles(infos: ReadonlyArray): DefinitionInfo[] { + return map(infos, getTargetOfMappedDeclarationFile); + } + /// Goto definition function getDefinitionAtPosition(fileName: string, position: number): DefinitionInfo[] { synchronizeHostData(); - return GoToDefinition.getDefinitionAtPosition(program, getValidSourceFile(fileName), position); + return getTargetOfMappedDeclarationFiles(GoToDefinition.getDefinitionAtPosition(program, getValidSourceFile(fileName), position)); } function getDefinitionAndBoundSpan(fileName: string, position: number): DefinitionInfoAndBoundSpan { synchronizeHostData(); - return GoToDefinition.getDefinitionAndBoundSpan(program, getValidSourceFile(fileName), position); + const result = GoToDefinition.getDefinitionAndBoundSpan(program, getValidSourceFile(fileName), position); + const mappedDefs = getTargetOfMappedDeclarationFiles(result.definitions); + if (mappedDefs === result.definitions) { + return result; + } + return { + definitions: mappedDefs, + textSpan: result.textSpan // TODO: Does this need to be mapped in some way? I don't _think_ so? + } } function getTypeDefinitionAtPosition(fileName: string, position: number): DefinitionInfo[] { synchronizeHostData(); - return GoToDefinition.getTypeDefinitionAtPosition(program.getTypeChecker(), getValidSourceFile(fileName), position); + return getTargetOfMappedDeclarationFiles(GoToDefinition.getTypeDefinitionAtPosition(program.getTypeChecker(), getValidSourceFile(fileName), position)); } /// Goto implementation diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts new file mode 100644 index 0000000000000..426bd3e4d2e38 --- /dev/null +++ b/src/services/sourcemaps.ts @@ -0,0 +1,342 @@ +/* @internal */ +namespace ts.sourcemaps { + export interface SourceMapData { + version?: number; + file?: string; + sourceRoot?: string; + sources: string[]; + sourcesContent?: string[]; + names?: string[]; + mappings: string; + } + + export interface SourceMappableLocation { + fileName: string; + position: number; + } + + export interface SourceMapper { + forwardMap(input: SourceMappableLocation): SourceMappableLocation; + reverseMap(input: SourceMappableLocation): SourceMappableLocation; + } + + export const identitySourceMapper = { forwardMap: identity, reverseMap: identity }; + + export interface SourceMapDecodeHost { + readFile(path: string): string; + fileExists(path: string): boolean; + getCanonicalFileName(path: string): string; + } + + export function decode(host: SourceMapDecodeHost, mapPath: string, map: SourceMapData, program?: Program): SourceMapper { + const currentDirectory = getDirectoryPath(mapPath); + const fallbackCache: Map = createMap(); + let decodedMappings: ProcessedSourceMapSpan[]; + let forwardSortedMappings: ProcessedSourceMapSpan[]; + let reverseSortedMappings: ProcessedSourceMapSpan[]; + + return { + forwardMap, + reverseMap + } + + function reverseMap(loc: SourceMappableLocation): SourceMappableLocation { + const maps = filter(getForwardSortedMappings(), m => comparePaths(loc.fileName, m.sourcePath, currentDirectory) === 0); + if (!length(maps)) return loc; + let targetIndex = binarySearch(maps, {sourcePosition: loc.position}, getSourcePosition, compareValues); + if (targetIndex < 0 && maps.length > 0) { + // if no exact match, closest is 2's compliment of result + targetIndex = ~targetIndex; + } + return { fileName: toPath(map.file, currentDirectory, host.getCanonicalFileName), position: maps[targetIndex].emittedPosition } // Closest span + } + + function forwardMap(loc: SourceMappableLocation): SourceMappableLocation { + const maps = getReverseSortedMappings(); + if (!length(maps)) return loc; + let targetIndex = binarySearch(maps, {emittedPosition: loc.position}, identity, compareProcessedSpanEmittedPositions); + if (targetIndex < 0 && maps.length > 0) { + // if no exact match, closest is 2's compliment of result + targetIndex = ~targetIndex; + } + return { fileName: toPath(maps[targetIndex].sourcePath, currentDirectory, host.getCanonicalFileName), position: maps[targetIndex].sourcePosition } // Closest span + } + + function getSourceFileLike(fileName: string): SourceFileLike | undefined { + // Lookup file in program, if provided + let file: SourceFileLike = program && program.getSourceFile(fileName); + if (!file) { + // Otherwise check the cache + const path = toPath(fileName, currentDirectory, host.getCanonicalFileName); + file = fallbackCache.get(path); + if (!file) { + // And failing that, check the disk + if (host.fileExists(path)) { + const text = host.readFile(path); + fallbackCache.set(path, file = { + text, + lineMap: undefined, + getLineAndCharacterOfPosition(pos) { + return computeLineAndCharacterOfPosition(getLineStarts(this), pos); + } + }); + } + } + } + return file; + } + + function getPositionOfLineAndCharacterUsingName(fileName: string, line: number, character: number) { + const file = getSourceFileLike(fileName); + if (!file) { + return -1; + } + return getPositionOfLineAndCharacter(file, line, character); + } + + function getDecodedMappings() { + return decodedMappings || (decodedMappings = calculateDecodedMappings()); + } + + function getReverseSortedMappings() { + return reverseSortedMappings || (reverseSortedMappings = getDecodedMappings().slice().sort(compareProcessedSpanSourcePositions)); + } + + function getForwardSortedMappings() { + return forwardSortedMappings || (forwardSortedMappings = getDecodedMappings().slice().sort(compareProcessedSpanEmittedPositions)); + } + + function calculateDecodedMappings(): ProcessedSourceMapSpan[] { + const state: DecoderState = { + encodedText: map.mappings, + currentNameIndex: undefined, + sourceMapNamesLength: map.names ? map.names.length : undefined, + currentEmittedColumn: 1, + currentEmittedLine: 1, + currentSourceColumn: 1, + currentSourceLine: 1, + currentSourceIndex: 0, + spans: [], + decodingIndex: 0, + processSpan, + } + while (!hasCompletedDecoding(state)) { + decodeSingleSpan(state); + if (state.error) { + // TODO: Report diagnostic on invalid source map or silently fail? + return []; + } + } + return state.spans; + } + + function compareProcessedSpanSourcePositions(a: ProcessedSourceMapSpan, b: ProcessedSourceMapSpan) { + return comparePaths(a.sourcePath, b.sourcePath, currentDirectory) || + compareValues(a.sourcePosition, b.sourcePosition); + } + + function compareProcessedSpanEmittedPositions(a: ProcessedSourceMapSpan, b: ProcessedSourceMapSpan) { + return compareValues(a.emittedPosition, b.emittedPosition); + } + + function processSpan(span: RawSourceMapSpan): ProcessedSourceMapSpan { + const sourcePath = map.sources[span.sourceIndex]; + return { + emittedPosition: getPositionOfLineAndCharacterUsingName(map.file, span.emittedLine - 1, span.emittedColumn - 1), + sourcePosition: getPositionOfLineAndCharacterUsingName(sourcePath, span.sourceLine - 1, span.sourceColumn - 1), + sourcePath, + name: span.nameIndex ? map.names[span.nameIndex] : undefined + }; + } + } + + function getSourcePosition(span: ProcessedSourceMapSpan) { + return span.sourcePosition; + } + + interface ProcessedSourceMapSpan { + emittedPosition: number; + sourcePosition: number; + sourcePath: string; + name?: string; + } + + interface RawSourceMapSpan { + emittedLine: number; + emittedColumn: number; + sourceLine: number; + sourceColumn: number; + sourceIndex: number; + nameIndex?: number; + } + + interface DecoderState { + decodingIndex: number; + currentEmittedLine: number; + currentEmittedColumn: number; + currentSourceLine: number; + currentSourceColumn: number; + currentSourceIndex: number; + currentNameIndex: number; + encodedText: string; + sourceMapNamesLength?: number; + error?: string; + spans: T[]; + processSpan: (span: RawSourceMapSpan) => T + } + + function hasCompletedDecoding(state: DecoderState) { + return state.decodingIndex === state.encodedText.length; + } + + function decodeSingleSpan(state: DecoderState): void { + while (state.decodingIndex < state.encodedText.length) { + const char = state.encodedText.charAt(state.decodingIndex); + if (char === ";") { + // New line + state.currentEmittedLine++; + state.currentEmittedColumn = 1; + state.decodingIndex++; + continue; + } + + if (char === ",") { + // Next entry is on same line - no action needed + state.decodingIndex++; + continue; + } + + // Read the current span + // 1. Column offset from prev read jsColumn + state.currentEmittedColumn += base64VLQFormatDecode(); + // Incorrect emittedColumn dont support this map + if (createErrorIfCondition(state.currentEmittedColumn < 1, "Invalid emittedColumn found")) { + return; + } + // Dont support reading mappings that dont have information about original source and its line numbers + if (createErrorIfCondition(isSourceMappingSegmentEnd(state.encodedText, state.decodingIndex), "Unsupported Error Format: No entries after emitted column")) { + return; + } + + // 2. Relative sourceIndex + state.currentSourceIndex += base64VLQFormatDecode(); + // Incorrect sourceIndex dont support this map + if (createErrorIfCondition(state.currentSourceIndex < 0, "Invalid sourceIndex found")) { + return; + } + // Dont support reading mappings that dont have information about original source span + if (createErrorIfCondition(isSourceMappingSegmentEnd(state.encodedText, state.decodingIndex), "Unsupported Error Format: No entries after sourceIndex")) { + return; + } + + // 3. Relative sourceLine 0 based + state.currentSourceLine += base64VLQFormatDecode(); + // Incorrect sourceLine dont support this map + if (createErrorIfCondition(state.currentSourceLine < 1, "Invalid sourceLine found")) { + return; + } + // Dont support reading mappings that dont have information about original source and its line numbers + if (createErrorIfCondition(isSourceMappingSegmentEnd(state.encodedText, state.decodingIndex), "Unsupported Error Format: No entries after emitted Line")) { + return; + } + + // 4. Relative sourceColumn 0 based + state.currentSourceColumn += base64VLQFormatDecode(); + // Incorrect sourceColumn dont support this map + if (createErrorIfCondition(state.currentSourceColumn < 1, "Invalid sourceLine found")) { + return; + } + // 5. Check if there is name: + if (!isSourceMappingSegmentEnd(state.encodedText, state.decodingIndex)) { + if (state.currentNameIndex === undefined) { + state.currentNameIndex = 0; + } + state.currentNameIndex += base64VLQFormatDecode(); + // Incorrect nameIndex dont support this map + if (createErrorIfCondition(state.currentNameIndex < 0 || state.currentNameIndex >= state.sourceMapNamesLength, "Invalid name index for the source map entry")) { + return; + } + } + // Dont support reading mappings that dont have information about original source and its line numbers + if (createErrorIfCondition(!isSourceMappingSegmentEnd(state.encodedText, state.decodingIndex), "Unsupported Error Format: There are more entries after " + (state.currentNameIndex === undefined ? "sourceColumn" : "nameIndex"))) { + return; + } + + // Entry should be complete + captureSpan(); + return; + } + + createErrorIfCondition(/*condition*/ true, "No encoded entry found"); + return; + + function captureSpan() { + state.spans.push(state.processSpan({ + emittedColumn: state.currentEmittedColumn, + emittedLine: state.currentEmittedLine, + sourceColumn: state.currentSourceColumn, + sourceIndex: state.currentSourceIndex, + sourceLine: state.currentSourceLine, + nameIndex: state.currentNameIndex + })); + } + + function createErrorIfCondition(condition: boolean, errormsg: string) { + if (state.error) { + // An error was already reported + return true; + } + + if (condition) { + state.error = errormsg; + } + + return condition; + } + + function base64VLQFormatDecode() { + let moreDigits = true; + let shiftCount = 0; + let value = 0; + + for (; moreDigits; state.decodingIndex++) { + if (createErrorIfCondition(state.decodingIndex >= state.encodedText.length, "Error in decoding base64VLQFormatDecode, past the mapping string")) { + return; + } + + // 6 digit number + const currentByte = base64FormatDecode(state.encodedText.charAt(state.decodingIndex)); + + // If msb is set, we still have more bits to continue + moreDigits = (currentByte & 32) !== 0; + + // least significant 5 bits are the next msbs in the final value. + value = value | ((currentByte & 31) << shiftCount); + shiftCount += 5; + } + + // Least significant bit if 1 represents negative and rest of the msb is actual absolute value + if ((value & 1) === 0) { + // + number + value = value >> 1; + } + else { + // - number + value = value >> 1; + value = -value; + } + + return value; + } + } + + function base64FormatDecode(char: string) { + return "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/".indexOf(char); + } + + function isSourceMappingSegmentEnd(encodedText: string, pos: number) { + return (pos === encodedText.length || + encodedText.charAt(pos) === "," || + encodedText.charAt(pos) === ";"); + } +} \ No newline at end of file diff --git a/src/services/tsconfig.json b/src/services/tsconfig.json index 7ea6c1fa976f1..4f838df6341fb 100644 --- a/src/services/tsconfig.json +++ b/src/services/tsconfig.json @@ -72,6 +72,7 @@ "shims.ts", "signatureHelp.ts", "suggestionDiagnostics.ts", + "sourcemaps.ts", "symbolDisplay.ts", "textChanges.ts", "refactorProvider.ts", diff --git a/src/services/types.ts b/src/services/types.ts index dba10960cdf76..bf2cba183ff3d 100644 --- a/src/services/types.ts +++ b/src/services/types.ts @@ -75,6 +75,8 @@ namespace ts { getLineStarts(): ReadonlyArray; getPositionOfLineAndCharacter(line: number, character: number): number; update(newText: string, textChangeRange: TextChangeRange): SourceFile; + + /* @internal */ sourceMapper?: sourcemaps.SourceMapper; } export interface SourceFileLike { diff --git a/tests/baselines/reference/api/tsserverlibrary.d.ts b/tests/baselines/reference/api/tsserverlibrary.d.ts index b1438ec7dffee..7786312436d14 100644 --- a/tests/baselines/reference/api/tsserverlibrary.d.ts +++ b/tests/baselines/reference/api/tsserverlibrary.d.ts @@ -3279,7 +3279,7 @@ declare namespace ts { tryScan(callback: () => T): T; } function tokenToString(t: SyntaxKind): string | undefined; - function getPositionOfLineAndCharacter(sourceFile: SourceFile, line: number, character: number): number; + function getPositionOfLineAndCharacter(sourceFile: SourceFileLike, line: number, character: number): number; function getLineAndCharacterOfPosition(sourceFile: SourceFileLike, position: number): LineAndCharacter; function isWhiteSpaceLike(ch: number): boolean; /** Does not include line breaks. For that, see isWhiteSpaceLike. */ diff --git a/tests/baselines/reference/api/typescript.d.ts b/tests/baselines/reference/api/typescript.d.ts index b3464dfe8cc09..699ea71cfc867 100644 --- a/tests/baselines/reference/api/typescript.d.ts +++ b/tests/baselines/reference/api/typescript.d.ts @@ -2934,7 +2934,7 @@ declare namespace ts { tryScan(callback: () => T): T; } function tokenToString(t: SyntaxKind): string | undefined; - function getPositionOfLineAndCharacter(sourceFile: SourceFile, line: number, character: number): number; + function getPositionOfLineAndCharacter(sourceFile: SourceFileLike, line: number, character: number): number; function getLineAndCharacterOfPosition(sourceFile: SourceFileLike, position: number): LineAndCharacter; function isWhiteSpaceLike(ch: number): boolean; /** Does not include line breaks. For that, see isWhiteSpaceLike. */ diff --git a/tests/cases/fourslash/declarationMapGoToDefinition.ts b/tests/cases/fourslash/declarationMapGoToDefinition.ts new file mode 100644 index 0000000000000..a83fd7e8396e3 --- /dev/null +++ b/tests/cases/fourslash/declarationMapGoToDefinition.ts @@ -0,0 +1,40 @@ +/// +// @Filename: index.ts +////export class Foo { +//// member: string; +//// /*2*/methodName(propName: SomeType): void {} +//// otherMethod() { +//// if (Math.random() > 0.5) { +//// return {x: 42}; +//// } +//// return {y: "yes"}; +//// } +////} +//// +////export interface SomeType { +//// member: number; +////} +// @Filename: indexdef.d.ts.map +////{"version":3,"file":"indexdef.d.ts","sourceRoot":"","sources":["index.ts"],"names":[],"mappings":"AAAA;IACI,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI;IACpC,WAAW;;;;;;;CAMd;AAED,MAAM,WAAW,QAAQ;IACrB,MAAM,EAAE,MAAM,CAAC;CAClB"} +// @Filename: indexdef.d.ts +////export declare class Foo { +//// member: string; +//// methodName(propName: SomeType): void; +//// otherMethod(): { +//// x: number; +//// y?: undefined; +//// } | { +//// y: string; +//// x?: undefined; +//// }; +////} +////export interface SomeType { +//// member: number; +////} +//////# sourceMappingURL=indexdef.d.ts.map +// @Filename: mymodule.ts +////import * as mod from "./indexdef"; +////const instance = new mod.Foo(); +////instance.[|/*1*/methodName|]({member: 12}); + +verify.goToDefinition("1", "2"); From c291828ddc46ecfe078d689f7e0e8af8bd4066c6 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Fri, 16 Mar 2018 14:08:02 -0700 Subject: [PATCH 03/32] Post-rebase touchups --- src/compiler/emitter.ts | 18 +++- src/harness/harness.ts | 2 +- src/services/services.ts | 1 + .../reference/declarationMapsOutFile.js.map | 2 +- .../declarationMapsOutFile.sourcemap.txt | 11 +- .../reference/declarationMapsOutFile2.js.map | 2 +- .../declarationMapsOutFile2.sourcemap.txt | 11 +- .../declarationMapsWithSourceMap.js.map | 4 +- ...declarationMapsWithSourceMap.sourcemap.txt | 101 +++++++++--------- 9 files changed, 84 insertions(+), 68 deletions(-) diff --git a/src/compiler/emitter.ts b/src/compiler/emitter.ts index af5f38a6a3403..b66bc3851cb65 100644 --- a/src/compiler/emitter.ts +++ b/src/compiler/emitter.ts @@ -200,6 +200,7 @@ namespace ts { onEmitSourceMapOfNode: sourceMap.emitNodeWithSourceMap, onEmitSourceMapOfToken: sourceMap.emitTokenWithSourceMap, onEmitSourceMapOfPosition: sourceMap.emitPos, + onSetSourceFile: setSourceFile, // transform hooks onEmitNode: declarationTransform.emitNodeWithNotification, @@ -213,11 +214,16 @@ namespace ts { declarationTransform.dispose(); } - function printSourceFileOrBundle(jsFilePath: string, sourceMapFilePath: string, sourceFileOrBundle: SourceFile | Bundle, printer: Printer, sourcemapKind: SourceMapEmitKind) { + function printSourceFileOrBundle(jsFilePath: string, sourceMapFilePath: string | undefined, sourceFileOrBundle: SourceFile | Bundle, printer: Printer, sourcemapKind: SourceMapEmitKind) { const bundle = sourceFileOrBundle.kind === SyntaxKind.Bundle ? sourceFileOrBundle : undefined; const sourceFile = sourceFileOrBundle.kind === SyntaxKind.SourceFile ? sourceFileOrBundle : undefined; const sourceFiles = bundle ? bundle.sourceFiles : [sourceFile]; - sourceMap.initialize(jsFilePath, sourceMapFilePath, sourceFileOrBundle); + if (sourcemapKind !== SourceMapEmitKind.None) { + sourceMap.initialize(jsFilePath, sourceMapFilePath, sourceFileOrBundle); + } + else { + sourceMap.setState(/*disabled*/ true); + } if (bundle) { bundledHelpers = createMap(); @@ -250,7 +256,13 @@ namespace ts { writeFile(host, emitterDiagnostics, jsFilePath, writer.getText(), compilerOptions.emitBOM, sourceFiles); // Reset state - sourceMap.reset(); + + if (sourcemapKind !== SourceMapEmitKind.None) { + sourceMap.reset(); + } + else { + sourceMap.setState(/*disabled*/ false); + } writer.clear(); currentSourceFile = undefined; diff --git a/src/harness/harness.ts b/src/harness/harness.ts index 3465a7cef91db..9e6978f2a15ff 100644 --- a/src/harness/harness.ts +++ b/src/harness/harness.ts @@ -1844,7 +1844,7 @@ namespace Harness { public getSourceMapRecord() { if (this.sourceMapData && this.sourceMapData.length > 0) { - return SourceMapRecorder.getSourceMapRecord(this.sourceMapData, this.program, this.files); + return SourceMapRecorder.getSourceMapRecord(this.sourceMapData, this.program, this.files, this.declFilesCode); } } } diff --git a/src/services/services.ts b/src/services/services.ts index 541b440a87c5f..35f052c2bd98f 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1597,6 +1597,7 @@ namespace ts { function getDefinitionAndBoundSpan(fileName: string, position: number): DefinitionInfoAndBoundSpan { synchronizeHostData(); const result = GoToDefinition.getDefinitionAndBoundSpan(program, getValidSourceFile(fileName), position); + if (!result) return result; const mappedDefs = getTargetOfMappedDeclarationFiles(result.definitions); if (mappedDefs === result.definitions) { return result; diff --git a/tests/baselines/reference/declarationMapsOutFile.js.map b/tests/baselines/reference/declarationMapsOutFile.js.map index 0018f3f9f31d7..051318f806401 100644 --- a/tests/baselines/reference/declarationMapsOutFile.js.map +++ b/tests/baselines/reference/declarationMapsOutFile.js.map @@ -1,2 +1,2 @@ //// [bundle.d.ts.map] -{"version":3,"file":"bundle.d.ts","sourceRoot":"","sources":["tests/cases/compiler/a.ts","tests/cases/compiler/index.ts"],"names":[],"mappings":";IAAA,MAAM;QACF,OAAO,CAAC,GAAG;YAAC,CAAC,EAAE,MAAM,CAAA;SAAC;;;QAGtB,MAAM,CAAC,IAAI;KAGd;;;ICPD,OAAO,EAAC,GAAG,EAAC,UAAY;IAExB,MAAM,CAAC,KAAY,CAAC;IAGpB,MAAM,CAAC,IAAI,CAAC;;KAAqB,CAAC;IAClC,OAAO,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC"} \ No newline at end of file +{"version":3,"file":"bundle.d.ts","sourceRoot":"","sources":["tests/cases/compiler/a.ts","tests/cases/compiler/index.ts"],"names":[],"mappings":";IAAA,MAAM;QACF,OAAO,CAAC,CAAC,EAAE;YAAC,CAAC,EAAE,MAAM,CAAA;SAAC;;;QAGtB,MAAM,CAAC,IAAI;KAGd;;;ICPD,OAAO,EAAC,GAAG,EAAC,UAAY;IAExB,MAAM,CAAC,KAAY,CAAC;IAGpB,MAAM,CAAC,IAAI,CAAC;;KAAqB,CAAC;IAClC,OAAO,EAAE,CAAC,EAAE,GAAG,EAAE,CAAC"} \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsOutFile.sourcemap.txt b/tests/baselines/reference/declarationMapsOutFile.sourcemap.txt index f1090c4f765a2..39d83719b20f3 100644 --- a/tests/baselines/reference/declarationMapsOutFile.sourcemap.txt +++ b/tests/baselines/reference/declarationMapsOutFile.sourcemap.txt @@ -22,17 +22,20 @@ sourceFile:tests/cases/compiler/a.ts 1->^^^^^^^^ 2 > ^^^^^^^ 3 > ^ -4 > ^^^ -5 > ^^^^-> +4 > ^ +5 > ^^ +6 > ^^^^-> 1-> class Foo { > 2 > doThing 3 > ( -4 > x: +4 > x +5 > : 1->Emitted(3, 9) Source(2, 5) + SourceIndex(0) 2 >Emitted(3, 16) Source(2, 12) + SourceIndex(0) 3 >Emitted(3, 17) Source(2, 13) + SourceIndex(0) -4 >Emitted(3, 20) Source(2, 16) + SourceIndex(0) +4 >Emitted(3, 18) Source(2, 14) + SourceIndex(0) +5 >Emitted(3, 20) Source(2, 16) + SourceIndex(0) --- >>> a: number; 1->^^^^^^^^^^^^ diff --git a/tests/baselines/reference/declarationMapsOutFile2.js.map b/tests/baselines/reference/declarationMapsOutFile2.js.map index 579077e0679cf..22845f90fb202 100644 --- a/tests/baselines/reference/declarationMapsOutFile2.js.map +++ b/tests/baselines/reference/declarationMapsOutFile2.js.map @@ -1,2 +1,2 @@ //// [bundle.d.ts.map] -{"version":3,"file":"bundle.d.ts","sourceRoot":"","sources":["tests/cases/compiler/a.ts","tests/cases/compiler/index.ts"],"names":[],"mappings":"AAAA;IACI,OAAO,CAAC,GAAG;QAAC,CAAC,EAAE,MAAM,CAAA;KAAC;;;IAGtB,MAAM,CAAC,IAAI;CAGd;ACPD,QAAA,MAAM,CAAC,KAAY,CAAC;AAGpB,QAAA,IAAI,CAAC;;CAAqB,CAAC"} \ No newline at end of file +{"version":3,"file":"bundle.d.ts","sourceRoot":"","sources":["tests/cases/compiler/a.ts","tests/cases/compiler/index.ts"],"names":[],"mappings":"AAAA;IACI,OAAO,CAAC,CAAC,EAAE;QAAC,CAAC,EAAE,MAAM,CAAA;KAAC;;;IAGtB,MAAM,CAAC,IAAI;CAGd;ACPD,QAAA,MAAM,CAAC,KAAY,CAAC;AAGpB,QAAA,IAAI,CAAC;;CAAqB,CAAC"} \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsOutFile2.sourcemap.txt b/tests/baselines/reference/declarationMapsOutFile2.sourcemap.txt index 469e5ff2bd992..d7a829917e653 100644 --- a/tests/baselines/reference/declarationMapsOutFile2.sourcemap.txt +++ b/tests/baselines/reference/declarationMapsOutFile2.sourcemap.txt @@ -18,17 +18,20 @@ sourceFile:tests/cases/compiler/a.ts 1->^^^^ 2 > ^^^^^^^ 3 > ^ -4 > ^^^ -5 > ^^^^-> +4 > ^ +5 > ^^ +6 > ^^^^-> 1->class Foo { > 2 > doThing 3 > ( -4 > x: +4 > x +5 > : 1->Emitted(2, 5) Source(2, 5) + SourceIndex(0) 2 >Emitted(2, 12) Source(2, 12) + SourceIndex(0) 3 >Emitted(2, 13) Source(2, 13) + SourceIndex(0) -4 >Emitted(2, 16) Source(2, 16) + SourceIndex(0) +4 >Emitted(2, 14) Source(2, 14) + SourceIndex(0) +5 >Emitted(2, 16) Source(2, 16) + SourceIndex(0) --- >>> a: number; 1->^^^^^^^^ diff --git a/tests/baselines/reference/declarationMapsWithSourceMap.js.map b/tests/baselines/reference/declarationMapsWithSourceMap.js.map index 8c10d3f8c4caf..17ef6a6004ccc 100644 --- a/tests/baselines/reference/declarationMapsWithSourceMap.js.map +++ b/tests/baselines/reference/declarationMapsWithSourceMap.js.map @@ -1,3 +1,3 @@ //// [bundle.js.map] -{"version":3,"file":"bundle.js","sourceRoot":"","sources":["tests/cases/compiler/a.ts","tests/cases/compiler/index.ts"],"names":[],"mappings":"AAAA;IAAA;IAOA,CAAC;IANG,qBAAO,GAAP,UAAQ,CAAc;QAClB,MAAM,CAAC,EAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAC,CAAC;IACpB,CAAC;IACM,QAAI,GAAX;QACI,MAAM,CAAC,IAAI,GAAG,EAAE,CAAC;IACrB,CAAC;IACL,UAAC;AAAD,CAAC,AAPD,IAOC;ACPD,IAAM,CAAC,GAAG,IAAI,GAAG,EAAE,CAAC;AACpB,CAAC,CAAC,OAAO,CAAC,EAAC,CAAC,EAAE,EAAE,EAAC,CAAC,CAAC;AAEnB,IAAI,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,EAAC,CAAC,EAAE,EAAE,EAAC,CAAC,CAAC"}//// [bundle.d.ts.map] -{"version":3,"file":"bundle.d.ts","sourceRoot":"","sources":["tests/cases/compiler/a.ts","tests/cases/compiler/index.ts"],"names":[],"mappings":"AAAA;IACI,OAAO,CAAC,GAAG;QAAC,CAAC,EAAE,MAAM,CAAA;KAAC;;;IAGtB,MAAM,CAAC,IAAI;CAGd;ACPD,QAAA,MAAM,CAAC,KAAY,CAAC;AAGpB,QAAA,IAAI,CAAC;;CAAqB,CAAC"} \ No newline at end of file +{"version":3,"file":"bundle.js","sourceRoot":"","sources":["tests/cases/compiler/a.ts","tests/cases/compiler/index.ts"],"names":[],"mappings":"AAAA;IAAA;IAOA,CAAC;IANG,qBAAO,GAAP,UAAQ,CAAc;QAClB,OAAO,EAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAAC,CAAC;IACpB,CAAC;IACM,QAAI,GAAX;QACI,OAAO,IAAI,GAAG,EAAE,CAAC;IACrB,CAAC;IACL,UAAC;AAAD,CAAC,AAPD,IAOC;ACPD,IAAM,CAAC,GAAG,IAAI,GAAG,EAAE,CAAC;AACpB,CAAC,CAAC,OAAO,CAAC,EAAC,CAAC,EAAE,EAAE,EAAC,CAAC,CAAC;AAEnB,IAAI,CAAC,GAAG,CAAC,CAAC,OAAO,CAAC,EAAC,CAAC,EAAE,EAAE,EAAC,CAAC,CAAC"}//// [bundle.d.ts.map] +{"version":3,"file":"bundle.d.ts","sourceRoot":"","sources":["tests/cases/compiler/a.ts","tests/cases/compiler/index.ts"],"names":[],"mappings":"AAAA;IACI,OAAO,CAAC,CAAC,EAAE;QAAC,CAAC,EAAE,MAAM,CAAA;KAAC;;;IAGtB,MAAM,CAAC,IAAI;CAGd;ACPD,QAAA,MAAM,CAAC,KAAY,CAAC;AAGpB,QAAA,IAAI,CAAC;;CAAqB,CAAC"} \ No newline at end of file diff --git a/tests/baselines/reference/declarationMapsWithSourceMap.sourcemap.txt b/tests/baselines/reference/declarationMapsWithSourceMap.sourcemap.txt index 1ee94a6057a7e..0f9df308ef6cb 100644 --- a/tests/baselines/reference/declarationMapsWithSourceMap.sourcemap.txt +++ b/tests/baselines/reference/declarationMapsWithSourceMap.sourcemap.txt @@ -55,39 +55,36 @@ sourceFile:tests/cases/compiler/a.ts --- >>> return { b: x.a }; 1 >^^^^^^^^ -2 > ^^^^^^ -3 > ^ -4 > ^^ -5 > ^ -6 > ^^ -7 > ^ -8 > ^ -9 > ^ -10> ^^ -11> ^ +2 > ^^^^^^^ +3 > ^^ +4 > ^ +5 > ^^ +6 > ^ +7 > ^ +8 > ^ +9 > ^^ +10> ^ 1 >) { > -2 > return -3 > -4 > { -5 > b -6 > : -7 > x -8 > . -9 > a -10> } -11> ; +2 > return +3 > { +4 > b +5 > : +6 > x +7 > . +8 > a +9 > } +10> ; 1 >Emitted(5, 9) Source(3, 9) + SourceIndex(0) -2 >Emitted(5, 15) Source(3, 15) + SourceIndex(0) -3 >Emitted(5, 16) Source(3, 16) + SourceIndex(0) -4 >Emitted(5, 18) Source(3, 17) + SourceIndex(0) -5 >Emitted(5, 19) Source(3, 18) + SourceIndex(0) -6 >Emitted(5, 21) Source(3, 20) + SourceIndex(0) -7 >Emitted(5, 22) Source(3, 21) + SourceIndex(0) -8 >Emitted(5, 23) Source(3, 22) + SourceIndex(0) -9 >Emitted(5, 24) Source(3, 23) + SourceIndex(0) -10>Emitted(5, 26) Source(3, 24) + SourceIndex(0) -11>Emitted(5, 27) Source(3, 25) + SourceIndex(0) +2 >Emitted(5, 16) Source(3, 16) + SourceIndex(0) +3 >Emitted(5, 18) Source(3, 17) + SourceIndex(0) +4 >Emitted(5, 19) Source(3, 18) + SourceIndex(0) +5 >Emitted(5, 21) Source(3, 20) + SourceIndex(0) +6 >Emitted(5, 22) Source(3, 21) + SourceIndex(0) +7 >Emitted(5, 23) Source(3, 22) + SourceIndex(0) +8 >Emitted(5, 24) Source(3, 23) + SourceIndex(0) +9 >Emitted(5, 26) Source(3, 24) + SourceIndex(0) +10>Emitted(5, 27) Source(3, 25) + SourceIndex(0) --- >>> }; 1 >^^^^ @@ -114,27 +111,24 @@ sourceFile:tests/cases/compiler/a.ts --- >>> return new Foo(); 1->^^^^^^^^ -2 > ^^^^^^ -3 > ^ -4 > ^^^^ -5 > ^^^ -6 > ^^ -7 > ^ +2 > ^^^^^^^ +3 > ^^^^ +4 > ^^^ +5 > ^^ +6 > ^ 1->static make() { > -2 > return -3 > -4 > new -5 > Foo -6 > () -7 > ; +2 > return +3 > new +4 > Foo +5 > () +6 > ; 1->Emitted(8, 9) Source(6, 9) + SourceIndex(0) -2 >Emitted(8, 15) Source(6, 15) + SourceIndex(0) -3 >Emitted(8, 16) Source(6, 16) + SourceIndex(0) -4 >Emitted(8, 20) Source(6, 20) + SourceIndex(0) -5 >Emitted(8, 23) Source(6, 23) + SourceIndex(0) -6 >Emitted(8, 25) Source(6, 25) + SourceIndex(0) -7 >Emitted(8, 26) Source(6, 26) + SourceIndex(0) +2 >Emitted(8, 16) Source(6, 16) + SourceIndex(0) +3 >Emitted(8, 20) Source(6, 20) + SourceIndex(0) +4 >Emitted(8, 23) Source(6, 23) + SourceIndex(0) +5 >Emitted(8, 25) Source(6, 25) + SourceIndex(0) +6 >Emitted(8, 26) Source(6, 26) + SourceIndex(0) --- >>> }; 1 >^^^^ @@ -318,17 +312,20 @@ sourceFile:tests/cases/compiler/a.ts 1->^^^^ 2 > ^^^^^^^ 3 > ^ -4 > ^^^ -5 > ^^^^-> +4 > ^ +5 > ^^ +6 > ^^^^-> 1->class Foo { > 2 > doThing 3 > ( -4 > x: +4 > x +5 > : 1->Emitted(2, 5) Source(2, 5) + SourceIndex(0) 2 >Emitted(2, 12) Source(2, 12) + SourceIndex(0) 3 >Emitted(2, 13) Source(2, 13) + SourceIndex(0) -4 >Emitted(2, 16) Source(2, 16) + SourceIndex(0) +4 >Emitted(2, 14) Source(2, 14) + SourceIndex(0) +5 >Emitted(2, 16) Source(2, 16) + SourceIndex(0) --- >>> a: number; 1->^^^^^^^^ From 2b26a27525e884b5de62f611cee5067940e253f7 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Fri, 16 Mar 2018 14:10:37 -0700 Subject: [PATCH 04/32] Rename API methods --- src/services/services.ts | 2 +- src/services/sourcemaps.ts | 14 +++++++------- 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/src/services/services.ts b/src/services/services.ts index 35f052c2bd98f..2cf4f2359eb34 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1567,7 +1567,7 @@ namespace ts { const file = program.getSourceFile(info.fileName); const mapper = getSourceMapper(file); const mapLoc: sourcemaps.SourceMappableLocation = {fileName: info.fileName, position: info.textSpan.start}; - const newLoc = mapper.forwardMap(mapLoc); + const newLoc = mapper.getOriginalPosition(mapLoc); if (newLoc === mapLoc) return info; return { containerKind: info.containerKind, diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts index 426bd3e4d2e38..5e556a0f2e507 100644 --- a/src/services/sourcemaps.ts +++ b/src/services/sourcemaps.ts @@ -16,11 +16,11 @@ namespace ts.sourcemaps { } export interface SourceMapper { - forwardMap(input: SourceMappableLocation): SourceMappableLocation; - reverseMap(input: SourceMappableLocation): SourceMappableLocation; + getOriginalPosition(input: SourceMappableLocation): SourceMappableLocation; + getGeneratedPosition(input: SourceMappableLocation): SourceMappableLocation; } - export const identitySourceMapper = { forwardMap: identity, reverseMap: identity }; + export const identitySourceMapper = { getOriginalPosition: identity, getGeneratedPosition: identity }; export interface SourceMapDecodeHost { readFile(path: string): string; @@ -36,11 +36,11 @@ namespace ts.sourcemaps { let reverseSortedMappings: ProcessedSourceMapSpan[]; return { - forwardMap, - reverseMap + getOriginalPosition, + getGeneratedPosition } - function reverseMap(loc: SourceMappableLocation): SourceMappableLocation { + function getGeneratedPosition(loc: SourceMappableLocation): SourceMappableLocation { const maps = filter(getForwardSortedMappings(), m => comparePaths(loc.fileName, m.sourcePath, currentDirectory) === 0); if (!length(maps)) return loc; let targetIndex = binarySearch(maps, {sourcePosition: loc.position}, getSourcePosition, compareValues); @@ -51,7 +51,7 @@ namespace ts.sourcemaps { return { fileName: toPath(map.file, currentDirectory, host.getCanonicalFileName), position: maps[targetIndex].emittedPosition } // Closest span } - function forwardMap(loc: SourceMappableLocation): SourceMappableLocation { + function getOriginalPosition(loc: SourceMappableLocation): SourceMappableLocation { const maps = getReverseSortedMappings(); if (!length(maps)) return loc; let targetIndex = binarySearch(maps, {emittedPosition: loc.position}, identity, compareProcessedSpanEmittedPositions); From ec44da5006d9ceec855105a3c0c3f6dc75039bf6 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Fri, 16 Mar 2018 15:07:34 -0700 Subject: [PATCH 05/32] Fix lints --- src/compiler/emitter.ts | 1 - src/services/services.ts | 10 ++++++---- src/services/sourcemaps.ts | 14 +++++++------- 3 files changed, 13 insertions(+), 12 deletions(-) diff --git a/src/compiler/emitter.ts b/src/compiler/emitter.ts index b66bc3851cb65..bba032d77b96c 100644 --- a/src/compiler/emitter.ts +++ b/src/compiler/emitter.ts @@ -256,7 +256,6 @@ namespace ts { writeFile(host, emitterDiagnostics, jsFilePath, writer.getText(), compilerOptions.emitBOM, sourceFiles); // Reset state - if (sourcemapKind !== SourceMapEmitKind.None) { sourceMap.reset(); } diff --git a/src/services/services.ts b/src/services/services.ts index 2cf4f2359eb34..e5877356ec8de 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1554,19 +1554,21 @@ namespace ts { try { maps = JSON.parse(doc); } - catch {} + catch { + // swallow error + } if (!maps || !maps.sources || !maps.file || !maps.mappings) { // obviously invalid map return file.sourceMapper = sourcemaps.identitySourceMapper; } - return file.sourceMapper = sourcemaps.decode({readFile: s => host.readFile(s), fileExists: s => host.fileExists(s), getCanonicalFileName}, mapFileName, maps, program); + return file.sourceMapper = sourcemaps.decode({ readFile: s => host.readFile(s), fileExists: s => host.fileExists(s), getCanonicalFileName }, mapFileName, maps, program); } function getTargetOfMappedDeclarationFile(info: DefinitionInfo): DefinitionInfo { if (endsWith(info.fileName, Extension.Dts)) { const file = program.getSourceFile(info.fileName); const mapper = getSourceMapper(file); - const mapLoc: sourcemaps.SourceMappableLocation = {fileName: info.fileName, position: info.textSpan.start}; + const mapLoc: sourcemaps.SourceMappableLocation = { fileName: info.fileName, position: info.textSpan.start }; const newLoc = mapper.getOriginalPosition(mapLoc); if (newLoc === mapLoc) return info; return { @@ -1605,7 +1607,7 @@ namespace ts { return { definitions: mappedDefs, textSpan: result.textSpan // TODO: Does this need to be mapped in some way? I don't _think_ so? - } + }; } function getTypeDefinitionAtPosition(fileName: string, position: number): DefinitionInfo[] { diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts index 5e556a0f2e507..e25e569e84eb4 100644 --- a/src/services/sourcemaps.ts +++ b/src/services/sourcemaps.ts @@ -38,28 +38,28 @@ namespace ts.sourcemaps { return { getOriginalPosition, getGeneratedPosition - } + }; function getGeneratedPosition(loc: SourceMappableLocation): SourceMappableLocation { const maps = filter(getForwardSortedMappings(), m => comparePaths(loc.fileName, m.sourcePath, currentDirectory) === 0); if (!length(maps)) return loc; - let targetIndex = binarySearch(maps, {sourcePosition: loc.position}, getSourcePosition, compareValues); + let targetIndex = binarySearch(maps, { sourcePosition: loc.position }, getSourcePosition, compareValues); if (targetIndex < 0 && maps.length > 0) { // if no exact match, closest is 2's compliment of result targetIndex = ~targetIndex; } - return { fileName: toPath(map.file, currentDirectory, host.getCanonicalFileName), position: maps[targetIndex].emittedPosition } // Closest span + return { fileName: toPath(map.file, currentDirectory, host.getCanonicalFileName), position: maps[targetIndex].emittedPosition }; // Closest span } function getOriginalPosition(loc: SourceMappableLocation): SourceMappableLocation { const maps = getReverseSortedMappings(); if (!length(maps)) return loc; - let targetIndex = binarySearch(maps, {emittedPosition: loc.position}, identity, compareProcessedSpanEmittedPositions); + let targetIndex = binarySearch(maps, { emittedPosition: loc.position }, identity, compareProcessedSpanEmittedPositions); if (targetIndex < 0 && maps.length > 0) { // if no exact match, closest is 2's compliment of result targetIndex = ~targetIndex; } - return { fileName: toPath(maps[targetIndex].sourcePath, currentDirectory, host.getCanonicalFileName), position: maps[targetIndex].sourcePosition } // Closest span + return { fileName: toPath(maps[targetIndex].sourcePath, currentDirectory, host.getCanonicalFileName), position: maps[targetIndex].sourcePosition }; // Closest span } function getSourceFileLike(fileName: string): SourceFileLike | undefined { @@ -119,7 +119,7 @@ namespace ts.sourcemaps { spans: [], decodingIndex: 0, processSpan, - } + }; while (!hasCompletedDecoding(state)) { decodeSingleSpan(state); if (state.error) { @@ -182,7 +182,7 @@ namespace ts.sourcemaps { sourceMapNamesLength?: number; error?: string; spans: T[]; - processSpan: (span: RawSourceMapSpan) => T + processSpan: (span: RawSourceMapSpan) => T; } function hasCompletedDecoding(state: DecoderState) { From ffb7f855e5f7580961b536727e58d8d056b0c952 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Fri, 16 Mar 2018 15:25:06 -0700 Subject: [PATCH 06/32] Fix typo in name XD --- src/services/services.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/services/services.ts b/src/services/services.ts index e5877356ec8de..3fee892e40663 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1544,7 +1544,7 @@ namespace ts { if (file.sourceMapper) { return file.sourceMapper; } - // TODO (wewigham): Read sourcemappingurl from last line of .d.ts if present + // TODO (weswigham): Read sourcemappingurl from last line of .d.ts if present const mapFileName = file.fileName + ".map"; if (!host.fileExists(mapFileName)) { return file.sourceMapper = sourcemaps.identitySourceMapper; From 589ac28e9c5a19bba89f3f163b132a92096ce15c Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Fri, 16 Mar 2018 15:49:41 -0700 Subject: [PATCH 07/32] Log sourcemap decode errors --- src/services/sourcemaps.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts index e25e569e84eb4..81d82a9adcd91 100644 --- a/src/services/sourcemaps.ts +++ b/src/services/sourcemaps.ts @@ -26,6 +26,7 @@ namespace ts.sourcemaps { readFile(path: string): string; fileExists(path: string): boolean; getCanonicalFileName(path: string): string; + log(text: string): void; } export function decode(host: SourceMapDecodeHost, mapPath: string, map: SourceMapData, program?: Program): SourceMapper { @@ -123,7 +124,7 @@ namespace ts.sourcemaps { while (!hasCompletedDecoding(state)) { decodeSingleSpan(state); if (state.error) { - // TODO: Report diagnostic on invalid source map or silently fail? + host.log(`Encountered error while decoding sourcemap found at ${mapPath}: ${state.error}`); return []; } } From 5584a2d45b9a772cbf98bca046749fc0bd9325cf Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Fri, 16 Mar 2018 16:38:34 -0700 Subject: [PATCH 08/32] Share the cache more, but also invalidate it more --- src/services/services.ts | 64 +++++++++++++++++++++++++++++++++----- src/services/sourcemaps.ts | 22 +++---------- src/services/types.ts | 1 + 3 files changed, 61 insertions(+), 26 deletions(-) diff --git a/src/services/services.ts b/src/services/services.ts index 3fee892e40663..425f103893edd 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1170,6 +1170,35 @@ namespace ts { } } + /* @internal */ + export interface SourceFileLikeCache { + get(path: Path): SourceFileLike | undefined; + } + + /* @internal */ + export function getSourceFileLikeCache(host: { readFile?: (path: string) => string, fileExists?: (path: string) => boolean }): SourceFileLikeCache { + const cached = createMap(); + return { + get(path: Path) { + if (cached.has(path)) { + return cached.get(path); + } + if (!host.fileExists || !host.readFile || !host.fileExists(path)) return; + // And failing that, check the disk + const text = host.readFile(path); + const file: SourceFileLike = { + text, + lineMap: undefined, + getLineAndCharacterOfPosition(pos) { + return computeLineAndCharacterOfPosition(getLineStarts(this), pos); + } + }; + cached.set(path, file); + return file; + } + }; + } + export function createLanguageService(host: LanguageServiceHost, documentRegistry: DocumentRegistry = createDocumentRegistry(host.useCaseSensitiveFileNames && host.useCaseSensitiveFileNames(), host.getCurrentDirectory())): LanguageService { @@ -1187,6 +1216,8 @@ namespace ts { localizedDiagnosticMessages = host.getLocalizedDiagnosticMessages(); } + let sourcemappedFileCache: SourceFileLikeCache; + function log(message: string) { if (host.log) { host.log(message); @@ -1294,6 +1325,11 @@ namespace ts { // It needs to be cleared to allow all collected snapshots to be released hostCache = undefined; + // We reset this cache on structure invalidation so we don't hold on to outdated files for long; however we can't use the `compilerHost` above, + // Because it only functions until `hostCache` is cleared, while we'll potentially need the functionality to lazily read sourcemap files during + // the course of whatever called `synchronizeHostData` + sourcemappedFileCache = getSourceFileLikeCache(host); + // Make sure all the nodes in the program are both bound, and have their parent // pointers set property. program.getTypeChecker(); @@ -1537,7 +1573,7 @@ namespace ts { return checker.getSymbolAtLocation(node); } - function getSourceMapper(file: SourceFile) { + function getSourceMapper(fileName: string, file: { sourceMapper?: sourcemaps.SourceMapper }) { if (!host.readFile || !host.fileExists) { return file.sourceMapper = sourcemaps.identitySourceMapper; } @@ -1545,7 +1581,7 @@ namespace ts { return file.sourceMapper; } // TODO (weswigham): Read sourcemappingurl from last line of .d.ts if present - const mapFileName = file.fileName + ".map"; + const mapFileName = fileName + ".map"; if (!host.fileExists(mapFileName)) { return file.sourceMapper = sourcemaps.identitySourceMapper; } @@ -1561,17 +1597,29 @@ namespace ts { // obviously invalid map return file.sourceMapper = sourcemaps.identitySourceMapper; } - return file.sourceMapper = sourcemaps.decode({ readFile: s => host.readFile(s), fileExists: s => host.fileExists(s), getCanonicalFileName }, mapFileName, maps, program); + return file.sourceMapper = sourcemaps.decode({ + readFile: s => host.readFile(s), + fileExists: s => host.fileExists(s), + getCanonicalFileName, + log, + }, mapFileName, maps, program, sourcemappedFileCache); } function getTargetOfMappedDeclarationFile(info: DefinitionInfo): DefinitionInfo { if (endsWith(info.fileName, Extension.Dts)) { - const file = program.getSourceFile(info.fileName); - const mapper = getSourceMapper(file); + let file: SourceFileLike = program.getSourceFile(info.fileName); + if (!file) { + const path = toPath(info.fileName, currentDirectory, getCanonicalFileName); + file = sourcemappedFileCache.get(path); + } + if (!file) { + return info; + } + const mapper = getSourceMapper(info.fileName, file); const mapLoc: sourcemaps.SourceMappableLocation = { fileName: info.fileName, position: info.textSpan.start }; const newLoc = mapper.getOriginalPosition(mapLoc); if (newLoc === mapLoc) return info; - return { + return getTargetOfMappedDeclarationFile({ containerKind: info.containerKind, containerName: info.containerName, fileName: newLoc.fileName, @@ -1579,9 +1627,9 @@ namespace ts { name: info.name, textSpan: { start: newLoc.position, - length: info.textSpan.length // TODO: How to determine correct length? mapping the end pos could result in a completely different location... + length: info.textSpan.length } - }; // TODO: recur if the target file is also a declaration file + }); } return info; } diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts index 81d82a9adcd91..a47d2ca8d1626 100644 --- a/src/services/sourcemaps.ts +++ b/src/services/sourcemaps.ts @@ -29,9 +29,8 @@ namespace ts.sourcemaps { log(text: string): void; } - export function decode(host: SourceMapDecodeHost, mapPath: string, map: SourceMapData, program?: Program): SourceMapper { + export function decode(host: SourceMapDecodeHost, mapPath: string, map: SourceMapData, program?: Program, fallbackCache = getSourceFileLikeCache(host)): SourceMapper { const currentDirectory = getDirectoryPath(mapPath); - const fallbackCache: Map = createMap(); let decodedMappings: ProcessedSourceMapSpan[]; let forwardSortedMappings: ProcessedSourceMapSpan[]; let reverseSortedMappings: ProcessedSourceMapSpan[]; @@ -65,24 +64,11 @@ namespace ts.sourcemaps { function getSourceFileLike(fileName: string): SourceFileLike | undefined { // Lookup file in program, if provided - let file: SourceFileLike = program && program.getSourceFile(fileName); + const file: SourceFileLike = program && program.getSourceFile(fileName); if (!file) { - // Otherwise check the cache + // Otherwise check the cache (which may hit disk) const path = toPath(fileName, currentDirectory, host.getCanonicalFileName); - file = fallbackCache.get(path); - if (!file) { - // And failing that, check the disk - if (host.fileExists(path)) { - const text = host.readFile(path); - fallbackCache.set(path, file = { - text, - lineMap: undefined, - getLineAndCharacterOfPosition(pos) { - return computeLineAndCharacterOfPosition(getLineStarts(this), pos); - } - }); - } - } + return fallbackCache.get(path); } return file; } diff --git a/src/services/types.ts b/src/services/types.ts index bf2cba183ff3d..f0f9dcb40580d 100644 --- a/src/services/types.ts +++ b/src/services/types.ts @@ -81,6 +81,7 @@ namespace ts { export interface SourceFileLike { getLineAndCharacterOfPosition(pos: number): LineAndCharacter; + /*@internal*/ sourceMapper?: sourcemaps.SourceMapper; } export interface SourceMapSource { From 238ba98e4a296e9b21e3ae7d02adcae3612da798 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Fri, 16 Mar 2018 16:40:15 -0700 Subject: [PATCH 09/32] Remove todo --- src/services/services.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/services/services.ts b/src/services/services.ts index 425f103893edd..2068374b470b5 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1654,7 +1654,7 @@ namespace ts { } return { definitions: mappedDefs, - textSpan: result.textSpan // TODO: Does this need to be mapped in some way? I don't _think_ so? + textSpan: result.textSpan }; } From ff158cfd9531958199c4c2a8ef1413763d54a01a Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Fri, 16 Mar 2018 17:04:49 -0700 Subject: [PATCH 10/32] Enable mapping on go to implementation as well --- src/services/services.ts | 83 +++++++++++++++++++++++++++------------- 1 file changed, 56 insertions(+), 27 deletions(-) diff --git a/src/services/services.ts b/src/services/services.ts index 2068374b470b5..3aed149eba1a5 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1605,37 +1605,48 @@ namespace ts { }, mapFileName, maps, program, sourcemappedFileCache); } - function getTargetOfMappedDeclarationFile(info: DefinitionInfo): DefinitionInfo { - if (endsWith(info.fileName, Extension.Dts)) { - let file: SourceFileLike = program.getSourceFile(info.fileName); - if (!file) { - const path = toPath(info.fileName, currentDirectory, getCanonicalFileName); - file = sourcemappedFileCache.get(path); - } - if (!file) { - return info; - } - const mapper = getSourceMapper(info.fileName, file); - const mapLoc: sourcemaps.SourceMappableLocation = { fileName: info.fileName, position: info.textSpan.start }; - const newLoc = mapper.getOriginalPosition(mapLoc); - if (newLoc === mapLoc) return info; - return getTargetOfMappedDeclarationFile({ - containerKind: info.containerKind, - containerName: info.containerName, - fileName: newLoc.fileName, - kind: info.kind, - name: info.name, - textSpan: { - start: newLoc.position, - length: info.textSpan.length + function makeGetTargetOfMappedPosition( + extract: (original: TIn) => sourcemaps.SourceMappableLocation, + create: (result: sourcemaps.SourceMappableLocation, original: TIn) => TIn + ) { + return getTargetOfMappedPosition; + function getTargetOfMappedPosition(input: TIn): TIn { + const info = extract(input); + if (endsWith(info.fileName, Extension.Dts)) { + let file: SourceFileLike = program.getSourceFile(info.fileName); + if (!file) { + const path = toPath(info.fileName, currentDirectory, getCanonicalFileName); + file = sourcemappedFileCache.get(path); } - }); + if (!file) { + return input; + } + const mapper = getSourceMapper(info.fileName, file); + const newLoc = mapper.getOriginalPosition(info); + if (newLoc === info) return input; + return getTargetOfMappedPosition(create(newLoc, input)); + } + return input; } - return info; } + const getTargetOfMappedDeclarationInfo = makeGetTargetOfMappedPosition( + (info: DefinitionInfo) => ({ fileName: info.fileName, position: info.textSpan.start }), + (newLoc, info) => ({ + containerKind: info.containerKind, + containerName: info.containerName, + fileName: newLoc.fileName, + kind: info.kind, + name: info.name, + textSpan: { + start: newLoc.position, + length: info.textSpan.length + } + }) + ); + function getTargetOfMappedDeclarationFiles(infos: ReadonlyArray): DefinitionInfo[] { - return map(infos, getTargetOfMappedDeclarationFile); + return map(infos, getTargetOfMappedDeclarationInfo); } /// Goto definition @@ -1664,9 +1675,27 @@ namespace ts { } /// Goto implementation + + const getTargetOfMappedImplementationLocation = makeGetTargetOfMappedPosition( + (info: ImplementationLocation) => ({ fileName: info.fileName, position: info.textSpan.start }), + (newLoc, info) => ({ + fileName: newLoc.fileName, + kind: info.kind, + displayParts: info.displayParts, + textSpan: { + start: newLoc.position, + length: info.textSpan.length + } + }) + ); + + function getTargetOfMappedImplementationLocations(infos: ReadonlyArray): ImplementationLocation[] { + return map(infos, getTargetOfMappedImplementationLocation); + } + function getImplementationAtPosition(fileName: string, position: number): ImplementationLocation[] { synchronizeHostData(); - return FindAllReferences.getImplementationsAtPosition(program, cancellationToken, program.getSourceFiles(), getValidSourceFile(fileName), position); + return getTargetOfMappedImplementationLocations(FindAllReferences.getImplementationsAtPosition(program, cancellationToken, program.getSourceFiles(), getValidSourceFile(fileName), position)); } /// References and Occurrences From b9f61494e259cc855007445808bb1bda293e8b72 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 12:48:18 -0700 Subject: [PATCH 11/32] Allow fourslash to test declaration maps mroe easily --- src/harness/fourslash.ts | 9 +++-- ...ionMapsGeneratedMapsEnableMapping.baseline | 34 +++++++++++++++++++ ...clarationMapsGeneratedMapsEnableMapping.ts | 29 ++++++++++++++++ tests/cases/fourslash/fourslash.ts | 2 +- 4 files changed, 70 insertions(+), 4 deletions(-) create mode 100644 tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping.baseline create mode 100644 tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping.ts diff --git a/src/harness/fourslash.ts b/src/harness/fourslash.ts index c10f49659891a..ec5a701198a9a 100644 --- a/src/harness/fourslash.ts +++ b/src/harness/fourslash.ts @@ -1601,7 +1601,7 @@ Actual: ${stringify(fullActual)}`); }); } - public baselineGetEmitOutput() { + public baselineGetEmitOutput(insertResultsIntoVfs?: boolean) { // Find file to be emitted const emitFiles: FourSlashFile[] = []; // List of FourSlashFile that has emitThisFile flag on @@ -1650,6 +1650,9 @@ Actual: ${stringify(fullActual)}`); for (const outputFile of emitOutput.outputFiles) { const fileName = "FileName : " + outputFile.name + Harness.IO.newLine(); resultString = resultString + fileName + outputFile.text; + if (insertResultsIntoVfs) { + this.languageServiceAdapterHost.addScript(ts.getNormalizedAbsolutePath(outputFile.name, "/"), outputFile.text, /*isRootFile*/ true); + } } resultString += Harness.IO.newLine(); }); @@ -4155,8 +4158,8 @@ namespace FourSlashInterface { this.state.baselineCurrentFileNameOrDottedNameSpans(); } - public baselineGetEmitOutput() { - this.state.baselineGetEmitOutput(); + public baselineGetEmitOutput(insertResultsIntoVfs?: boolean) { + this.state.baselineGetEmitOutput(insertResultsIntoVfs); } public baselineQuickInfo() { diff --git a/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping.baseline b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping.baseline new file mode 100644 index 0000000000000..10cc53e95face --- /dev/null +++ b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping.baseline @@ -0,0 +1,34 @@ +EmitSkipped: false +FileName : ./dist/index.js +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +var Foo = /** @class */ (function () { + function Foo() { + } + Foo.prototype.methodName = function (propName) { }; + Foo.prototype.otherMethod = function () { + if (Math.random() > 0.5) { + return { x: 42 }; + } + return { y: "yes" }; + }; + return Foo; +}()); +exports.Foo = Foo; +FileName : ./dist/index.d.ts.map +{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../tests/cases/fourslash/index.ts"],"names":[],"mappings":"AAAA;IACI,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI;IACpC,WAAW;;;;;;;CAMd;AAED,MAAM,WAAW,QAAQ;IACrB,MAAM,EAAE,MAAM,CAAC;CAClB"}FileName : ./dist/index.d.ts +export declare class Foo { + member: string; + methodName(propName: SomeType): void; + otherMethod(): { + x: number; + y?: undefined; + } | { + y: string; + x?: undefined; + }; +} +export interface SomeType { + member: number; +} +//# sourceMappingURL=index.d.ts.map diff --git a/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping.ts b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping.ts new file mode 100644 index 0000000000000..1f284aa4d47bd --- /dev/null +++ b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping.ts @@ -0,0 +1,29 @@ +/// +// @BaselineFile: declarationMapsGeneratedMapsEnableMapping.baseline +// @outDir: ./dist +// @declaration: true +// @declarationMaps: true +// @Filename: index.ts +// @emitThisFile: true +////export class Foo { +//// member: string; +//// /*2*/methodName(propName: SomeType): void {} +//// otherMethod() { +//// if (Math.random() > 0.5) { +//// return {x: 42}; +//// } +//// return {y: "yes"}; +//// } +////} +//// +////export interface SomeType { +//// member: number; +////} +// @Filename: mymodule.ts +////import * as mod from "/dist/index"; +////const instance = new mod.Foo(); +////instance.[|/*1*/methodName|]({member: 12}); + +verify.baselineGetEmitOutput(/*insertResultIntoVfs*/ true); +verify.goToDefinition("1", "2"); +verify.goToType("1", "2"); diff --git a/tests/cases/fourslash/fourslash.ts b/tests/cases/fourslash/fourslash.ts index fd53a5acf2e8a..aa11493cb6d73 100644 --- a/tests/cases/fourslash/fourslash.ts +++ b/tests/cases/fourslash/fourslash.ts @@ -287,7 +287,7 @@ declare namespace FourSlashInterface { numberOfErrorsInCurrentFile(expected: number): void; baselineCurrentFileBreakpointLocations(): void; baselineCurrentFileNameOrDottedNameSpans(): void; - baselineGetEmitOutput(): void; + baselineGetEmitOutput(insertResultsIntoVfs?: boolean): void; baselineQuickInfo(): void; nameOrDottedNameSpanTextIs(text: string): void; outliningSpansInCurrentFile(spans: Range[]): void; From 53f72de2b6a5bba392f8f6a2b555de98d25adcb7 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 13:01:33 -0700 Subject: [PATCH 12/32] more test --- .../declarationMapsGeneratedMapsEnableMapping.ts | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping.ts b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping.ts index 1f284aa4d47bd..23185af0e6555 100644 --- a/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping.ts +++ b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping.ts @@ -25,5 +25,9 @@ ////instance.[|/*1*/methodName|]({member: 12}); verify.baselineGetEmitOutput(/*insertResultIntoVfs*/ true); -verify.goToDefinition("1", "2"); -verify.goToType("1", "2"); +verify.goToDefinition("1", "2"); // getDefinitionAndBoundSpan +verify.goToType("1", "2"); // getTypeDefinitionAtPosition +goTo.marker("1"); +verify.goToDefinitionIs("2"); // getDefinitionAtPosition +goTo.implementation(); // getImplementationAtPosition +verify.caretAtMarker("2"); From 6070108495ab6e2554a6f13a2fb8bca80a05b82a Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 13:29:26 -0700 Subject: [PATCH 13/32] Handle sourceRoot --- src/services/sourcemaps.ts | 21 +++++------ ...onMapsGeneratedMapsEnableMapping2.baseline | 35 +++++++++++++++++++ ...larationMapsGeneratedMapsEnableMapping2.ts | 35 +++++++++++++++++++ 3 files changed, 81 insertions(+), 10 deletions(-) create mode 100644 tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping2.baseline create mode 100644 tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping2.ts diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts index a47d2ca8d1626..c6d3075a01043 100644 --- a/src/services/sourcemaps.ts +++ b/src/services/sourcemaps.ts @@ -31,6 +31,7 @@ namespace ts.sourcemaps { export function decode(host: SourceMapDecodeHost, mapPath: string, map: SourceMapData, program?: Program, fallbackCache = getSourceFileLikeCache(host)): SourceMapper { const currentDirectory = getDirectoryPath(mapPath); + const sourceRoot = map.sourceRoot || currentDirectory; let decodedMappings: ProcessedSourceMapSpan[]; let forwardSortedMappings: ProcessedSourceMapSpan[]; let reverseSortedMappings: ProcessedSourceMapSpan[]; @@ -41,14 +42,14 @@ namespace ts.sourcemaps { }; function getGeneratedPosition(loc: SourceMappableLocation): SourceMappableLocation { - const maps = filter(getForwardSortedMappings(), m => comparePaths(loc.fileName, m.sourcePath, currentDirectory) === 0); + const maps = filter(getForwardSortedMappings(), m => comparePaths(loc.fileName, m.sourcePath, sourceRoot) === 0); if (!length(maps)) return loc; let targetIndex = binarySearch(maps, { sourcePosition: loc.position }, getSourcePosition, compareValues); if (targetIndex < 0 && maps.length > 0) { // if no exact match, closest is 2's compliment of result targetIndex = ~targetIndex; } - return { fileName: toPath(map.file, currentDirectory, host.getCanonicalFileName), position: maps[targetIndex].emittedPosition }; // Closest span + return { fileName: toPath(map.file, sourceRoot, host.getCanonicalFileName), position: maps[targetIndex].emittedPosition }; // Closest span } function getOriginalPosition(loc: SourceMappableLocation): SourceMappableLocation { @@ -59,22 +60,22 @@ namespace ts.sourcemaps { // if no exact match, closest is 2's compliment of result targetIndex = ~targetIndex; } - return { fileName: toPath(maps[targetIndex].sourcePath, currentDirectory, host.getCanonicalFileName), position: maps[targetIndex].sourcePosition }; // Closest span + return { fileName: toPath(maps[targetIndex].sourcePath, sourceRoot, host.getCanonicalFileName), position: maps[targetIndex].sourcePosition }; // Closest span } - function getSourceFileLike(fileName: string): SourceFileLike | undefined { + function getSourceFileLike(fileName: string, location: string): SourceFileLike | undefined { // Lookup file in program, if provided const file: SourceFileLike = program && program.getSourceFile(fileName); if (!file) { // Otherwise check the cache (which may hit disk) - const path = toPath(fileName, currentDirectory, host.getCanonicalFileName); + const path = toPath(fileName, location, host.getCanonicalFileName); return fallbackCache.get(path); } return file; } - function getPositionOfLineAndCharacterUsingName(fileName: string, line: number, character: number) { - const file = getSourceFileLike(fileName); + function getPositionOfLineAndCharacterUsingName(fileName: string, directory: string, line: number, character: number) { + const file = getSourceFileLike(fileName, directory); if (!file) { return -1; } @@ -118,7 +119,7 @@ namespace ts.sourcemaps { } function compareProcessedSpanSourcePositions(a: ProcessedSourceMapSpan, b: ProcessedSourceMapSpan) { - return comparePaths(a.sourcePath, b.sourcePath, currentDirectory) || + return comparePaths(a.sourcePath, b.sourcePath, sourceRoot) || compareValues(a.sourcePosition, b.sourcePosition); } @@ -129,8 +130,8 @@ namespace ts.sourcemaps { function processSpan(span: RawSourceMapSpan): ProcessedSourceMapSpan { const sourcePath = map.sources[span.sourceIndex]; return { - emittedPosition: getPositionOfLineAndCharacterUsingName(map.file, span.emittedLine - 1, span.emittedColumn - 1), - sourcePosition: getPositionOfLineAndCharacterUsingName(sourcePath, span.sourceLine - 1, span.sourceColumn - 1), + emittedPosition: getPositionOfLineAndCharacterUsingName(map.file, currentDirectory, span.emittedLine - 1, span.emittedColumn - 1), + sourcePosition: getPositionOfLineAndCharacterUsingName(sourcePath, sourceRoot, span.sourceLine - 1, span.sourceColumn - 1), sourcePath, name: span.nameIndex ? map.names[span.nameIndex] : undefined }; diff --git a/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping2.baseline b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping2.baseline new file mode 100644 index 0000000000000..484dd75768221 --- /dev/null +++ b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping2.baseline @@ -0,0 +1,35 @@ +EmitSkipped: false +FileName : ./dist/index.js.map +{"version":3,"file":"index.js","sourceRoot":"/tests/cases/fourslash/","sources":["index.ts"],"names":[],"mappings":";;AAAA;IAAA;IASA,CAAC;IAPG,wBAAU,GAAV,UAAW,QAAkB,IAAS,CAAC;IACvC,yBAAW,GAAX;QACI,IAAI,IAAI,CAAC,MAAM,EAAE,GAAG,GAAG,EAAE;YACrB,OAAO,EAAC,CAAC,EAAE,EAAE,EAAC,CAAC;SAClB;QACD,OAAO,EAAC,CAAC,EAAE,KAAK,EAAC,CAAC;IACtB,CAAC;IACL,UAAC;AAAD,CAAC,AATD,IASC;AATY,kBAAG"}FileName : ./dist/index.js +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +var Foo = /** @class */ (function () { + function Foo() { + } + Foo.prototype.methodName = function (propName) { }; + Foo.prototype.otherMethod = function () { + if (Math.random() > 0.5) { + return { x: 42 }; + } + return { y: "yes" }; + }; + return Foo; +}()); +exports.Foo = Foo; +//# sourceMappingURL=index.js.mapFileName : ./dist/index.d.ts.map +{"version":3,"file":"index.d.ts","sourceRoot":"/tests/cases/fourslash/","sources":["index.ts"],"names":[],"mappings":"AAAA;IACI,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI;IACpC,WAAW;;;;;;;CAMd;AAED,MAAM,WAAW,QAAQ;IACrB,MAAM,EAAE,MAAM,CAAC;CAClB"}FileName : ./dist/index.d.ts +export declare class Foo { + member: string; + methodName(propName: SomeType): void; + otherMethod(): { + x: number; + y?: undefined; + } | { + y: string; + x?: undefined; + }; +} +export interface SomeType { + member: number; +} +//# sourceMappingURL=index.d.ts.map diff --git a/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping2.ts b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping2.ts new file mode 100644 index 0000000000000..6dfd7a997d07c --- /dev/null +++ b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping2.ts @@ -0,0 +1,35 @@ +/// +// @BaselineFile: declarationMapsGeneratedMapsEnableMapping2.baseline +// @outDir: ./dist +// @sourceMap: true +// @sourceRoot: /tests/cases/fourslash/ +// @declaration: true +// @declarationMaps: true +// @Filename: index.ts +// @emitThisFile: true +////export class Foo { +//// member: string; +//// /*2*/methodName(propName: SomeType): void {} +//// otherMethod() { +//// if (Math.random() > 0.5) { +//// return {x: 42}; +//// } +//// return {y: "yes"}; +//// } +////} +//// +////export interface SomeType { +//// member: number; +////} +// @Filename: mymodule.ts +////import * as mod from "/dist/index"; +////const instance = new mod.Foo(); +////instance.[|/*1*/methodName|]({member: 12}); + +verify.baselineGetEmitOutput(/*insertResultIntoVfs*/ true); +verify.goToDefinition("1", "2"); // getDefinitionAndBoundSpan +verify.goToType("1", "2"); // getTypeDefinitionAtPosition +goTo.marker("1"); +verify.goToDefinitionIs("2"); // getDefinitionAtPosition +goTo.implementation(); // getImplementationAtPosition +verify.caretAtMarker("2"); From 0a6355345bae01101aea326d2568e6243013e93a Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 13:38:43 -0700 Subject: [PATCH 14/32] Add tests documenting current behavior with other sourcemapping flags --- ...onMapsGeneratedMapsEnableMapping3.baseline | 34 ++++++++++++++++++ ...neratedMapsEnableMapping_NoInline.baseline | 34 ++++++++++++++++++ ...MapsEnableMapping_NoInlineSources.baseline | 34 ++++++++++++++++++ .../declarationMapsEnableMapping_NoInline.ts | 34 ++++++++++++++++++ ...rationMapsEnableMapping_NoInlineSources.ts | 35 +++++++++++++++++++ ...larationMapsGeneratedMapsEnableMapping3.ts | 34 ++++++++++++++++++ 6 files changed, 205 insertions(+) create mode 100644 tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping3.baseline create mode 100644 tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInline.baseline create mode 100644 tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInlineSources.baseline create mode 100644 tests/cases/fourslash/declarationMapsEnableMapping_NoInline.ts create mode 100644 tests/cases/fourslash/declarationMapsEnableMapping_NoInlineSources.ts create mode 100644 tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping3.ts diff --git a/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping3.baseline b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping3.baseline new file mode 100644 index 0000000000000..80860234baf0f --- /dev/null +++ b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping3.baseline @@ -0,0 +1,34 @@ +EmitSkipped: false +FileName : ./dist/index.js +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +var Foo = /** @class */ (function () { + function Foo() { + } + Foo.prototype.methodName = function (propName) { }; + Foo.prototype.otherMethod = function () { + if (Math.random() > 0.5) { + return { x: 42 }; + } + return { y: "yes" }; + }; + return Foo; +}()); +exports.Foo = Foo; +FileName : ./dist/index.d.ts.map +{"version":3,"file":"index.d.ts","sourceRoot":"/tests/cases/fourslash/","sources":["index.ts"],"names":[],"mappings":"AAAA;IACI,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI;IACpC,WAAW;;;;;;;CAMd;AAED,MAAM,WAAW,QAAQ;IACrB,MAAM,EAAE,MAAM,CAAC;CAClB"}FileName : ./dist/index.d.ts +export declare class Foo { + member: string; + methodName(propName: SomeType): void; + otherMethod(): { + x: number; + y?: undefined; + } | { + y: string; + x?: undefined; + }; +} +export interface SomeType { + member: number; +} +//# sourceMappingURL=index.d.ts.map diff --git a/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInline.baseline b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInline.baseline new file mode 100644 index 0000000000000..a174b5fff2b7c --- /dev/null +++ b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInline.baseline @@ -0,0 +1,34 @@ +EmitSkipped: false +FileName : ./dist/index.js +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +var Foo = /** @class */ (function () { + function Foo() { + } + Foo.prototype.methodName = function (propName) { }; + Foo.prototype.otherMethod = function () { + if (Math.random() > 0.5) { + return { x: 42 }; + } + return { y: "yes" }; + }; + return Foo; +}()); +exports.Foo = Foo; +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi90ZXN0cy9jYXNlcy9mb3Vyc2xhc2gvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFBQTtJQUFBO0lBU0EsQ0FBQztJQVBHLHdCQUFVLEdBQVYsVUFBVyxRQUFrQixJQUFTLENBQUM7SUFDdkMseUJBQVcsR0FBWDtRQUNJLElBQUksSUFBSSxDQUFDLE1BQU0sRUFBRSxHQUFHLEdBQUcsRUFBRTtZQUNyQixPQUFPLEVBQUMsQ0FBQyxFQUFFLEVBQUUsRUFBQyxDQUFDO1NBQ2xCO1FBQ0QsT0FBTyxFQUFDLENBQUMsRUFBRSxLQUFLLEVBQUMsQ0FBQztJQUN0QixDQUFDO0lBQ0wsVUFBQztBQUFELENBQUMsQUFURCxJQVNDO0FBVFksa0JBQUcifQ==FileName : ./dist/index.d.ts.map +{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../tests/cases/fourslash/index.ts"],"names":[],"mappings":"AAAA;IACI,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI;IACpC,WAAW;;;;;;;CAMd;AAED,MAAM,WAAW,QAAQ;IACrB,MAAM,EAAE,MAAM,CAAC;CAClB"}FileName : ./dist/index.d.ts +export declare class Foo { + member: string; + methodName(propName: SomeType): void; + otherMethod(): { + x: number; + y?: undefined; + } | { + y: string; + x?: undefined; + }; +} +export interface SomeType { + member: number; +} +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguZC50cyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3Rlc3RzL2Nhc2VzL2ZvdXJzbGFzaC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTtJQUNJLE1BQU0sRUFBRSxNQUFNLENBQUM7SUFDZixVQUFVLENBQUMsUUFBUSxFQUFFLFFBQVEsR0FBRyxJQUFJO0lBQ3BDLFdBQVc7Ozs7Ozs7Q0FNZDtBQUVELE1BQU0sV0FBVyxRQUFRO0lBQ3JCLE1BQU0sRUFBRSxNQUFNLENBQUM7Q0FDbEIifQ== diff --git a/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInlineSources.baseline b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInlineSources.baseline new file mode 100644 index 0000000000000..1dd1e6d9b1ec2 --- /dev/null +++ b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInlineSources.baseline @@ -0,0 +1,34 @@ +EmitSkipped: false +FileName : ./dist/index.js +"use strict"; +Object.defineProperty(exports, "__esModule", { value: true }); +var Foo = /** @class */ (function () { + function Foo() { + } + Foo.prototype.methodName = function (propName) { }; + Foo.prototype.otherMethod = function () { + if (Math.random() > 0.5) { + return { x: 42 }; + } + return { y: "yes" }; + }; + return Foo; +}()); +exports.Foo = Foo; +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi90ZXN0cy9jYXNlcy9mb3Vyc2xhc2gvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFBQTtJQUFBO0lBU0EsQ0FBQztJQVBHLHdCQUFVLEdBQVYsVUFBVyxRQUFrQixJQUFTLENBQUM7SUFDdkMseUJBQVcsR0FBWDtRQUNJLElBQUksSUFBSSxDQUFDLE1BQU0sRUFBRSxHQUFHLEdBQUcsRUFBRTtZQUNyQixPQUFPLEVBQUMsQ0FBQyxFQUFFLEVBQUUsRUFBQyxDQUFDO1NBQ2xCO1FBQ0QsT0FBTyxFQUFDLENBQUMsRUFBRSxLQUFLLEVBQUMsQ0FBQztJQUN0QixDQUFDO0lBQ0wsVUFBQztBQUFELENBQUMsQUFURCxJQVNDO0FBVFksa0JBQUciLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY2xhc3MgRm9vIHtcbiAgICBtZW1iZXI6IHN0cmluZztcbiAgICBtZXRob2ROYW1lKHByb3BOYW1lOiBTb21lVHlwZSk6IHZvaWQge31cbiAgICBvdGhlck1ldGhvZCgpIHtcbiAgICAgICAgaWYgKE1hdGgucmFuZG9tKCkgPiAwLjUpIHtcbiAgICAgICAgICAgIHJldHVybiB7eDogNDJ9O1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB7eTogXCJ5ZXNcIn07XG4gICAgfVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIFNvbWVUeXBlIHtcbiAgICBtZW1iZXI6IG51bWJlcjtcbn0iXX0=FileName : ./dist/index.d.ts.map +{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../tests/cases/fourslash/index.ts"],"names":[],"mappings":"AAAA;IACI,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI;IACpC,WAAW;;;;;;;CAMd;AAED,MAAM,WAAW,QAAQ;IACrB,MAAM,EAAE,MAAM,CAAC;CAClB","sourcesContent":["export class Foo {\n member: string;\n methodName(propName: SomeType): void {}\n otherMethod() {\n if (Math.random() > 0.5) {\n return {x: 42};\n }\n return {y: \"yes\"};\n }\n}\n\nexport interface SomeType {\n member: number;\n}"]}FileName : ./dist/index.d.ts +export declare class Foo { + member: string; + methodName(propName: SomeType): void; + otherMethod(): { + x: number; + y?: undefined; + } | { + y: string; + x?: undefined; + }; +} +export interface SomeType { + member: number; +} +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguZC50cyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3Rlc3RzL2Nhc2VzL2ZvdXJzbGFzaC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTtJQUNJLE1BQU0sRUFBRSxNQUFNLENBQUM7SUFDZixVQUFVLENBQUMsUUFBUSxFQUFFLFFBQVEsR0FBRyxJQUFJO0lBQ3BDLFdBQVc7Ozs7Ozs7Q0FNZDtBQUVELE1BQU0sV0FBVyxRQUFRO0lBQ3JCLE1BQU0sRUFBRSxNQUFNLENBQUM7Q0FDbEIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY2xhc3MgRm9vIHtcbiAgICBtZW1iZXI6IHN0cmluZztcbiAgICBtZXRob2ROYW1lKHByb3BOYW1lOiBTb21lVHlwZSk6IHZvaWQge31cbiAgICBvdGhlck1ldGhvZCgpIHtcbiAgICAgICAgaWYgKE1hdGgucmFuZG9tKCkgPiAwLjUpIHtcbiAgICAgICAgICAgIHJldHVybiB7eDogNDJ9O1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB7eTogXCJ5ZXNcIn07XG4gICAgfVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIFNvbWVUeXBlIHtcbiAgICBtZW1iZXI6IG51bWJlcjtcbn0iXX0= diff --git a/tests/cases/fourslash/declarationMapsEnableMapping_NoInline.ts b/tests/cases/fourslash/declarationMapsEnableMapping_NoInline.ts new file mode 100644 index 0000000000000..1422a098b5d88 --- /dev/null +++ b/tests/cases/fourslash/declarationMapsEnableMapping_NoInline.ts @@ -0,0 +1,34 @@ +/// +// @BaselineFile: declarationMapsGeneratedMapsEnableMapping_NoInline.baseline +// @outDir: ./dist +// @inlineSourceMap: true +// @declaration: true +// @declarationMaps: true +// @Filename: index.ts +// @emitThisFile: true +////export class Foo { +//// member: string; +//// /*2*/methodName(propName: SomeType): void {} +//// otherMethod() { +//// if (Math.random() > 0.5) { +//// return {x: 42}; +//// } +//// return {y: "yes"}; +//// } +////} +//// +////export interface SomeType { +//// member: number; +////} +// @Filename: mymodule.ts +////import * as mod from "/dist/index"; +////const instance = new mod.Foo(); +////instance.[|/*1*/methodName|]({member: 12}); + +verify.baselineGetEmitOutput(/*insertResultIntoVfs*/ true); +verify.goToDefinition("1", "2"); // getDefinitionAndBoundSpan +verify.goToType("1", "2"); // getTypeDefinitionAtPosition +goTo.marker("1"); +verify.goToDefinitionIs("2"); // getDefinitionAtPosition +goTo.implementation(); // getImplementationAtPosition +verify.caretAtMarker("2"); \ No newline at end of file diff --git a/tests/cases/fourslash/declarationMapsEnableMapping_NoInlineSources.ts b/tests/cases/fourslash/declarationMapsEnableMapping_NoInlineSources.ts new file mode 100644 index 0000000000000..10ad967e35e62 --- /dev/null +++ b/tests/cases/fourslash/declarationMapsEnableMapping_NoInlineSources.ts @@ -0,0 +1,35 @@ +/// +// @BaselineFile: declarationMapsGeneratedMapsEnableMapping_NoInlineSources.baseline +// @outDir: ./dist +// @inlineSourceMap: true +// @inlineSources: true +// @declaration: true +// @declarationMaps: true +// @Filename: index.ts +// @emitThisFile: true +////export class Foo { +//// member: string; +//// /*2*/methodName(propName: SomeType): void {} +//// otherMethod() { +//// if (Math.random() > 0.5) { +//// return {x: 42}; +//// } +//// return {y: "yes"}; +//// } +////} +//// +////export interface SomeType { +//// member: number; +////} +// @Filename: mymodule.ts +////import * as mod from "/dist/index"; +////const instance = new mod.Foo(); +////instance.[|/*1*/methodName|]({member: 12}); + +verify.baselineGetEmitOutput(/*insertResultIntoVfs*/ true); +verify.goToDefinition("1", "2"); // getDefinitionAndBoundSpan +verify.goToType("1", "2"); // getTypeDefinitionAtPosition +goTo.marker("1"); +verify.goToDefinitionIs("2"); // getDefinitionAtPosition +goTo.implementation(); // getImplementationAtPosition +verify.caretAtMarker("2"); \ No newline at end of file diff --git a/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping3.ts b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping3.ts new file mode 100644 index 0000000000000..883f806e09475 --- /dev/null +++ b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping3.ts @@ -0,0 +1,34 @@ +/// +// @BaselineFile: declarationMapsGeneratedMapsEnableMapping3.baseline +// @outDir: ./dist +// @sourceRoot: /tests/cases/fourslash/ +// @declaration: true +// @declarationMaps: true +// @Filename: index.ts +// @emitThisFile: true +////export class Foo { +//// member: string; +//// /*2*/methodName(propName: SomeType): void {} +//// otherMethod() { +//// if (Math.random() > 0.5) { +//// return {x: 42}; +//// } +//// return {y: "yes"}; +//// } +////} +//// +////export interface SomeType { +//// member: number; +////} +// @Filename: mymodule.ts +////import * as mod from "/dist/index"; +////const instance = new mod.Foo(); +////instance.[|/*1*/methodName|]({member: 12}); + +verify.baselineGetEmitOutput(/*insertResultIntoVfs*/ true); +verify.goToDefinition("1", "2"); // getDefinitionAndBoundSpan +verify.goToType("1", "2"); // getTypeDefinitionAtPosition +goTo.marker("1"); +verify.goToDefinitionIs("2"); // getDefinitionAtPosition +goTo.implementation(); // getImplementationAtPosition +verify.caretAtMarker("2"); From d8480b2515da9996a17b4496b1b3cf398d363924 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 17:35:39 -0700 Subject: [PATCH 15/32] Ignore inline options for declaration file maps, simplify dispatch in emitter --- src/compiler/emitter.ts | 61 ++++++++----------- src/compiler/sourcemap.ts | 42 ++++++------- src/compiler/utilities.ts | 2 +- ...neratedMapsEnableMapping_NoInline.baseline | 2 +- ...MapsEnableMapping_NoInlineSources.baseline | 4 +- 5 files changed, 49 insertions(+), 62 deletions(-) diff --git a/src/compiler/emitter.ts b/src/compiler/emitter.ts index bba032d77b96c..188c6d450383d 100644 --- a/src/compiler/emitter.ts +++ b/src/compiler/emitter.ts @@ -64,7 +64,7 @@ namespace ts { } function getSourceMapFilePath(jsFilePath: string, options: CompilerOptions) { - return options.sourceMap ? jsFilePath + ".map" : undefined; + return (options.sourceMap && !options.inlineSourceMap) ? jsFilePath + ".map" : undefined; } // JavaScript files are always LanguageVariant.JSX, as JSX syntax is allowed in .js files also. @@ -85,13 +85,6 @@ namespace ts { return Extension.Js; } - const enum SourceMapEmitKind { - None, - File, - Inline, - DeclarationFile - } - /*@internal*/ // targetSourceFile is when users only want one file in entire project to be emitted. This is used in compileOnSave feature export function emitFiles(resolver: EmitResolver, host: EmitHost, targetSourceFile: SourceFile, emitOnlyDtsFiles?: boolean, transformers?: TransformerFactory[]): EmitResult { @@ -103,6 +96,13 @@ namespace ts { const newLine = host.getNewLine(); const writer = createTextWriter(newLine); const sourceMap = createSourceMapWriter(host, writer); + const declarationSourceMap = createSourceMapWriter(host, writer, { + sourceMap: compilerOptions.declarationMaps, + sourceRoot: compilerOptions.sourceRoot, + mapRoot: compilerOptions.mapRoot, + extendedDiagnostics: compilerOptions.extendedDiagnostics, + // Explicitly do not passthru either `inline` option + }); let currentSourceFile: SourceFile; let bundledHelpers: Map; @@ -171,8 +171,7 @@ namespace ts { onSetSourceFile: setSourceFile, }); - const sourcemapKind = compilerOptions.inlineSourceMap ? SourceMapEmitKind.Inline : compilerOptions.sourceMap ? SourceMapEmitKind.File : SourceMapEmitKind.None; - printSourceFileOrBundle(jsFilePath, sourceMapFilePath, isSourceFile(sourceFileOrBundle) ? transform.transformed[0] : createBundle(transform.transformed), printer, sourcemapKind); + printSourceFileOrBundle(jsFilePath, sourceMapFilePath, isSourceFile(sourceFileOrBundle) ? transform.transformed[0] : createBundle(transform.transformed), printer, sourceMap); // Clean up emit nodes on parse tree transform.dispose(); @@ -197,10 +196,10 @@ namespace ts { hasGlobalName: resolver.hasGlobalName, // sourcemap hooks - onEmitSourceMapOfNode: sourceMap.emitNodeWithSourceMap, - onEmitSourceMapOfToken: sourceMap.emitTokenWithSourceMap, - onEmitSourceMapOfPosition: sourceMap.emitPos, - onSetSourceFile: setSourceFile, + onEmitSourceMapOfNode: declarationSourceMap.emitNodeWithSourceMap, + onEmitSourceMapOfToken: declarationSourceMap.emitTokenWithSourceMap, + onEmitSourceMapOfPosition: declarationSourceMap.emitPos, + onSetSourceFile: setSourceFileDeclarationMaps, // transform hooks onEmitNode: declarationTransform.emitNodeWithNotification, @@ -209,21 +208,16 @@ namespace ts { const declBlocked = (!!declarationTransform.diagnostics && !!declarationTransform.diagnostics.length) || !!host.isEmitBlocked(declarationFilePath) || !!compilerOptions.noEmit; emitSkipped = emitSkipped || declBlocked; if (!declBlocked || emitOnlyDtsFiles) { - printSourceFileOrBundle(declarationFilePath, declarationMapPath, declarationTransform.transformed[0], declarationPrinter, getAreDeclarationMapsEnabled(compilerOptions) ? SourceMapEmitKind.DeclarationFile : SourceMapEmitKind.None); + printSourceFileOrBundle(declarationFilePath, declarationMapPath, declarationTransform.transformed[0], declarationPrinter, declarationSourceMap); } declarationTransform.dispose(); } - function printSourceFileOrBundle(jsFilePath: string, sourceMapFilePath: string | undefined, sourceFileOrBundle: SourceFile | Bundle, printer: Printer, sourcemapKind: SourceMapEmitKind) { + function printSourceFileOrBundle(jsFilePath: string, sourceMapFilePath: string | undefined, sourceFileOrBundle: SourceFile | Bundle, printer: Printer, mapRecorder: SourceMapWriter) { const bundle = sourceFileOrBundle.kind === SyntaxKind.Bundle ? sourceFileOrBundle : undefined; const sourceFile = sourceFileOrBundle.kind === SyntaxKind.SourceFile ? sourceFileOrBundle : undefined; const sourceFiles = bundle ? bundle.sourceFiles : [sourceFile]; - if (sourcemapKind !== SourceMapEmitKind.None) { - sourceMap.initialize(jsFilePath, sourceMapFilePath, sourceFileOrBundle); - } - else { - sourceMap.setState(/*disabled*/ true); - } + mapRecorder.initialize(jsFilePath, sourceMapFilePath || "", sourceFileOrBundle, sourceMapDataList); if (bundle) { bundledHelpers = createMap(); @@ -237,31 +231,21 @@ namespace ts { writer.writeLine(); - const sourceMappingURL = sourceMap.getSourceMappingURL(); + const sourceMappingURL = mapRecorder.getSourceMappingURL(); if (sourceMappingURL) { writer.write(`//# ${"sourceMappingURL"}=${sourceMappingURL}`); // Sometimes tools can sometimes see this line as a source mapping url comment } // Write the source map - if (sourceMapFilePath && (sourcemapKind === SourceMapEmitKind.File || sourcemapKind === SourceMapEmitKind.DeclarationFile)) { - writeFile(host, emitterDiagnostics, sourceMapFilePath, sourceMap.getText(), /*writeByteOrderMark*/ false, sourceFiles); - } - - // Record source map data for the test harness. - if (sourcemapKind !== SourceMapEmitKind.None) { - sourceMapDataList.push(sourceMap.getSourceMapData()); + if (sourceMapFilePath) { + writeFile(host, emitterDiagnostics, sourceMapFilePath, mapRecorder.getText(), /*writeByteOrderMark*/ false, sourceFiles); } // Write the output file writeFile(host, emitterDiagnostics, jsFilePath, writer.getText(), compilerOptions.emitBOM, sourceFiles); // Reset state - if (sourcemapKind !== SourceMapEmitKind.None) { - sourceMap.reset(); - } - else { - sourceMap.setState(/*disabled*/ false); - } + mapRecorder.reset(); writer.clear(); currentSourceFile = undefined; @@ -274,6 +258,11 @@ namespace ts { sourceMap.setSourceFile(node); } + function setSourceFileDeclarationMaps(node: SourceFile) { + currentSourceFile = node; + declarationSourceMap.setSourceFile(node); + } + function emitHelpers(node: Node, writeLines: (text: string) => void) { let helpersEmitted = false; const bundle = node.kind === SyntaxKind.Bundle ? node : undefined; diff --git a/src/compiler/sourcemap.ts b/src/compiler/sourcemap.ts index ee5e713a9ef6b..83c95dd645926 100644 --- a/src/compiler/sourcemap.ts +++ b/src/compiler/sourcemap.ts @@ -10,7 +10,7 @@ namespace ts { * @param sourceMapFilePath The path to the output source map file. * @param sourceFileOrBundle The input source file or bundle for the program. */ - initialize(filePath: string, sourceMapFilePath: string, sourceFileOrBundle: SourceFile | Bundle): void; + initialize(filePath: string, sourceMapFilePath: string, sourceFileOrBundle: SourceFile | Bundle, sourceMapOutput?: SourceMapData[]): void; /** * Reset the SourceMapWriter to an empty state. @@ -62,16 +62,6 @@ namespace ts { * Gets the SourceMappingURL for the source map. */ getSourceMappingURL(): string; - - /** - * Gets test data for source maps. - */ - getSourceMapData(): SourceMapData; - - /** - * @returns the previous disabled state - */ - setState(disabled: boolean): boolean; } // Used for initialize lastEncodedSourceMapSpan and reset lastEncodedSourceMapSpan when updateLastEncodedAndRecordedSpans @@ -83,8 +73,16 @@ namespace ts { sourceIndex: 0 }; - export function createSourceMapWriter(host: EmitHost, writer: EmitTextWriter): SourceMapWriter { - const compilerOptions = host.getCompilerOptions(); + export interface SourceMapOptions { + sourceMap?: boolean; + inlineSourceMap?: boolean; + inlineSources?: boolean; + sourceRoot?: string; + mapRoot?: string; + extendedDiagnostics?: boolean; + } + + export function createSourceMapWriter(host: EmitHost, writer: EmitTextWriter, compilerOptions: SourceMapOptions = host.getCompilerOptions()): SourceMapWriter { const extendedDiagnostics = compilerOptions.extendedDiagnostics; let currentSource: SourceMapSource; let currentSourceText: string; @@ -100,27 +98,20 @@ namespace ts { // Source map data let sourceMapData: SourceMapData; + let sourceMapDataList: SourceMapData[]; let disabled: boolean = !(compilerOptions.sourceMap || compilerOptions.inlineSourceMap); return { initialize, reset, - getSourceMapData: () => sourceMapData, setSourceFile, emitPos, emitNodeWithSourceMap, emitTokenWithSourceMap, getText, getSourceMappingURL, - setState, }; - function setState(state: boolean) { - const last = disabled; - disabled = state; - return last; - } - /** * Skips trivia such as comments and white-space that can optionally overriden by the source map source */ @@ -135,7 +126,7 @@ namespace ts { * @param sourceMapFilePath The path to the output source map file. * @param sourceFileOrBundle The input source file or bundle for the program. */ - function initialize(filePath: string, sourceMapFilePath: string, sourceFileOrBundle: SourceFile | Bundle) { + function initialize(filePath: string, sourceMapFilePath: string, sourceFileOrBundle: SourceFile | Bundle, outputSourceMapDataList?: SourceMapData[]) { if (disabled) { return; } @@ -143,6 +134,7 @@ namespace ts { if (sourceMapData) { reset(); } + sourceMapDataList = outputSourceMapDataList; currentSource = undefined; currentSourceText = undefined; @@ -211,6 +203,11 @@ namespace ts { return; } + // Record source map data for the test harness. + if (sourceMapDataList) { + sourceMapDataList.push(sourceMapData); + } + currentSource = undefined; sourceMapDir = undefined; sourceMapSourceIndex = undefined; @@ -218,6 +215,7 @@ namespace ts { lastEncodedSourceMapSpan = undefined; lastEncodedNameIndex = undefined; sourceMapData = undefined; + sourceMapDataList = undefined; } // Encoding for sourcemap span diff --git a/src/compiler/utilities.ts b/src/compiler/utilities.ts index 025c810a07fb4..e756fa5763baa 100644 --- a/src/compiler/utilities.ts +++ b/src/compiler/utilities.ts @@ -2818,7 +2818,7 @@ namespace ts { export interface EmitFileNames { jsFilePath: string; - sourceMapFilePath: string; + sourceMapFilePath: string | undefined; declarationFilePath: string | undefined; declarationMapPath: string | undefined; } diff --git a/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInline.baseline b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInline.baseline index a174b5fff2b7c..d8d978c954f87 100644 --- a/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInline.baseline +++ b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInline.baseline @@ -31,4 +31,4 @@ export declare class Foo { export interface SomeType { member: number; } -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguZC50cyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3Rlc3RzL2Nhc2VzL2ZvdXJzbGFzaC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTtJQUNJLE1BQU0sRUFBRSxNQUFNLENBQUM7SUFDZixVQUFVLENBQUMsUUFBUSxFQUFFLFFBQVEsR0FBRyxJQUFJO0lBQ3BDLFdBQVc7Ozs7Ozs7Q0FNZDtBQUVELE1BQU0sV0FBVyxRQUFRO0lBQ3JCLE1BQU0sRUFBRSxNQUFNLENBQUM7Q0FDbEIifQ== +//# sourceMappingURL=index.d.ts.map diff --git a/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInlineSources.baseline b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInlineSources.baseline index 1dd1e6d9b1ec2..6114662156a11 100644 --- a/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInlineSources.baseline +++ b/tests/baselines/reference/declarationMapsGeneratedMapsEnableMapping_NoInlineSources.baseline @@ -16,7 +16,7 @@ var Foo = /** @class */ (function () { }()); exports.Foo = Foo; //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi90ZXN0cy9jYXNlcy9mb3Vyc2xhc2gvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7QUFBQTtJQUFBO0lBU0EsQ0FBQztJQVBHLHdCQUFVLEdBQVYsVUFBVyxRQUFrQixJQUFTLENBQUM7SUFDdkMseUJBQVcsR0FBWDtRQUNJLElBQUksSUFBSSxDQUFDLE1BQU0sRUFBRSxHQUFHLEdBQUcsRUFBRTtZQUNyQixPQUFPLEVBQUMsQ0FBQyxFQUFFLEVBQUUsRUFBQyxDQUFDO1NBQ2xCO1FBQ0QsT0FBTyxFQUFDLENBQUMsRUFBRSxLQUFLLEVBQUMsQ0FBQztJQUN0QixDQUFDO0lBQ0wsVUFBQztBQUFELENBQUMsQUFURCxJQVNDO0FBVFksa0JBQUciLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY2xhc3MgRm9vIHtcbiAgICBtZW1iZXI6IHN0cmluZztcbiAgICBtZXRob2ROYW1lKHByb3BOYW1lOiBTb21lVHlwZSk6IHZvaWQge31cbiAgICBvdGhlck1ldGhvZCgpIHtcbiAgICAgICAgaWYgKE1hdGgucmFuZG9tKCkgPiAwLjUpIHtcbiAgICAgICAgICAgIHJldHVybiB7eDogNDJ9O1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB7eTogXCJ5ZXNcIn07XG4gICAgfVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIFNvbWVUeXBlIHtcbiAgICBtZW1iZXI6IG51bWJlcjtcbn0iXX0=FileName : ./dist/index.d.ts.map -{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../tests/cases/fourslash/index.ts"],"names":[],"mappings":"AAAA;IACI,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI;IACpC,WAAW;;;;;;;CAMd;AAED,MAAM,WAAW,QAAQ;IACrB,MAAM,EAAE,MAAM,CAAC;CAClB","sourcesContent":["export class Foo {\n member: string;\n methodName(propName: SomeType): void {}\n otherMethod() {\n if (Math.random() > 0.5) {\n return {x: 42};\n }\n return {y: \"yes\"};\n }\n}\n\nexport interface SomeType {\n member: number;\n}"]}FileName : ./dist/index.d.ts +{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../tests/cases/fourslash/index.ts"],"names":[],"mappings":"AAAA;IACI,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,CAAC,QAAQ,EAAE,QAAQ,GAAG,IAAI;IACpC,WAAW;;;;;;;CAMd;AAED,MAAM,WAAW,QAAQ;IACrB,MAAM,EAAE,MAAM,CAAC;CAClB"}FileName : ./dist/index.d.ts export declare class Foo { member: string; methodName(propName: SomeType): void; @@ -31,4 +31,4 @@ export declare class Foo { export interface SomeType { member: number; } -//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguZC50cyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3Rlc3RzL2Nhc2VzL2ZvdXJzbGFzaC9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTtJQUNJLE1BQU0sRUFBRSxNQUFNLENBQUM7SUFDZixVQUFVLENBQUMsUUFBUSxFQUFFLFFBQVEsR0FBRyxJQUFJO0lBQ3BDLFdBQVc7Ozs7Ozs7Q0FNZDtBQUVELE1BQU0sV0FBVyxRQUFRO0lBQ3JCLE1BQU0sRUFBRSxNQUFNLENBQUM7Q0FDbEIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY2xhc3MgRm9vIHtcbiAgICBtZW1iZXI6IHN0cmluZztcbiAgICBtZXRob2ROYW1lKHByb3BOYW1lOiBTb21lVHlwZSk6IHZvaWQge31cbiAgICBvdGhlck1ldGhvZCgpIHtcbiAgICAgICAgaWYgKE1hdGgucmFuZG9tKCkgPiAwLjUpIHtcbiAgICAgICAgICAgIHJldHVybiB7eDogNDJ9O1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB7eTogXCJ5ZXNcIn07XG4gICAgfVxufVxuXG5leHBvcnQgaW50ZXJmYWNlIFNvbWVUeXBlIHtcbiAgICBtZW1iZXI6IG51bWJlcjtcbn0iXX0= +//# sourceMappingURL=index.d.ts.map From ca88d5ebd80f06d23d750137ca4c6ae5fd596538 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 17:47:22 -0700 Subject: [PATCH 16/32] Change program diagnostic --- src/compiler/diagnosticMessages.json | 4 ++++ src/compiler/program.ts | 22 +++++++++---------- .../optionsInlineSourceMapMapRoot.errors.txt | 4 ++-- ...SourceRootWithNoSourceMapOption.errors.txt | 4 ++-- ...SourceRootWithNoSourceMapOption.errors.txt | 4 ++-- .../mapRootWithNoSourceMapOption.errors.txt | 4 ++-- .../mapRootWithNoSourceMapOption.errors.txt | 4 ++-- 7 files changed, 24 insertions(+), 22 deletions(-) diff --git a/src/compiler/diagnosticMessages.json b/src/compiler/diagnosticMessages.json index 6f96ac18398da..0905963541df6 100644 --- a/src/compiler/diagnosticMessages.json +++ b/src/compiler/diagnosticMessages.json @@ -2767,6 +2767,10 @@ "category": "Error", "code": 5068 }, + "Option '{0}' cannot be specified without specifying option '{1}' or option '{2}'.": { + "category": "Error", + "code": 5069 + }, "Generates a sourcemap for each corresponding '.d.ts' file.": { "category": "Message", diff --git a/src/compiler/program.ts b/src/compiler/program.ts index 664329a4da0e4..34e5d65b7921c 100755 --- a/src/compiler/program.ts +++ b/src/compiler/program.ts @@ -2113,7 +2113,7 @@ namespace ts { if (options.mapRoot && !(options.sourceMap || options.declarationMaps)) { // Error to specify --mapRoot without --sourcemap - createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "mapRoot", "sourceMap"); + createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1_or_option_2, "mapRoot", "sourceMap", "declarationMaps"); } if (options.declarationDir) { @@ -2125,10 +2125,8 @@ namespace ts { } } - if (options.declarationMaps) { - if (!options.declaration) { - createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "declarationMaps", "declaration"); - } + if (options.declarationMaps && !options.declaration) { + createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "declarationMaps", "declaration"); } if (options.lib && options.noLib) { @@ -2305,21 +2303,21 @@ namespace ts { return emptyArray; } - function createDiagnosticForOptionName(message: DiagnosticMessage, option1: string, option2?: string) { - createDiagnosticForOption(/*onKey*/ true, option1, option2, message, option1, option2); + function createDiagnosticForOptionName(message: DiagnosticMessage, option1: string, option2?: string, option3?: string) { + createDiagnosticForOption(/*onKey*/ true, option1, option2, message, option1, option2, option3); } function createOptionValueDiagnostic(option1: string, message: DiagnosticMessage, arg0: string) { createDiagnosticForOption(/*onKey*/ false, option1, /*option2*/ undefined, message, arg0); } - function createDiagnosticForOption(onKey: boolean, option1: string, option2: string, message: DiagnosticMessage, arg0: string | number, arg1?: string | number) { + function createDiagnosticForOption(onKey: boolean, option1: string, option2: string, message: DiagnosticMessage, arg0: string | number, arg1?: string | number, arg2?: string | number) { const compilerOptionsObjectLiteralSyntax = getCompilerOptionsObjectLiteralSyntax(); const needCompilerDiagnostic = !compilerOptionsObjectLiteralSyntax || - !createOptionDiagnosticInObjectLiteralSyntax(compilerOptionsObjectLiteralSyntax, onKey, option1, option2, message, arg0, arg1); + !createOptionDiagnosticInObjectLiteralSyntax(compilerOptionsObjectLiteralSyntax, onKey, option1, option2, message, arg0, arg1, arg2); if (needCompilerDiagnostic) { - programDiagnostics.add(createCompilerDiagnostic(message, arg0, arg1)); + programDiagnostics.add(createCompilerDiagnostic(message, arg0, arg1, arg2)); } } @@ -2338,10 +2336,10 @@ namespace ts { return _compilerOptionsObjectLiteralSyntax; } - function createOptionDiagnosticInObjectLiteralSyntax(objectLiteral: ObjectLiteralExpression, onKey: boolean, key1: string, key2: string, message: DiagnosticMessage, arg0: string | number, arg1?: string | number): boolean { + function createOptionDiagnosticInObjectLiteralSyntax(objectLiteral: ObjectLiteralExpression, onKey: boolean, key1: string, key2: string, message: DiagnosticMessage, arg0: string | number, arg1?: string | number, arg2?: string | number): boolean { const props = getPropertyAssignment(objectLiteral, key1, key2); for (const prop of props) { - programDiagnostics.add(createDiagnosticForNodeInSourceFile(options.configFile, onKey ? prop.name : prop.initializer, message, arg0, arg1)); + programDiagnostics.add(createDiagnosticForNodeInSourceFile(options.configFile, onKey ? prop.name : prop.initializer, message, arg0, arg1, arg2)); } return !!props.length; } diff --git a/tests/baselines/reference/optionsInlineSourceMapMapRoot.errors.txt b/tests/baselines/reference/optionsInlineSourceMapMapRoot.errors.txt index 77a221a8cbbd7..0520441c0cfbd 100644 --- a/tests/baselines/reference/optionsInlineSourceMapMapRoot.errors.txt +++ b/tests/baselines/reference/optionsInlineSourceMapMapRoot.errors.txt @@ -1,8 +1,8 @@ -error TS5052: Option 'mapRoot' cannot be specified without specifying option 'sourceMap'. error TS5053: Option 'mapRoot' cannot be specified with option 'inlineSourceMap'. +error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. -!!! error TS5052: Option 'mapRoot' cannot be specified without specifying option 'sourceMap'. !!! error TS5053: Option 'mapRoot' cannot be specified with option 'inlineSourceMap'. +!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. ==== tests/cases/compiler/optionsInlineSourceMapMapRoot.ts (0 errors) ==== var a = 10; \ No newline at end of file diff --git a/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/amd/mapRootSourceRootWithNoSourceMapOption.errors.txt b/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/amd/mapRootSourceRootWithNoSourceMapOption.errors.txt index 2a780bdeda0b3..a47daea28c7e8 100644 --- a/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/amd/mapRootSourceRootWithNoSourceMapOption.errors.txt +++ b/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/amd/mapRootSourceRootWithNoSourceMapOption.errors.txt @@ -1,9 +1,9 @@ error TS5051: Option 'sourceRoot can only be used when either option '--inlineSourceMap' or option '--sourceMap' is provided. -error TS5052: Option 'mapRoot' cannot be specified without specifying option 'sourceMap'. +error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. !!! error TS5051: Option 'sourceRoot can only be used when either option '--inlineSourceMap' or option '--sourceMap' is provided. -!!! error TS5052: Option 'mapRoot' cannot be specified without specifying option 'sourceMap'. +!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. ==== m1.ts (0 errors) ==== var m1_a1 = 10; class m1_c1 { diff --git a/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/node/mapRootSourceRootWithNoSourceMapOption.errors.txt b/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/node/mapRootSourceRootWithNoSourceMapOption.errors.txt index 2a780bdeda0b3..a47daea28c7e8 100644 --- a/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/node/mapRootSourceRootWithNoSourceMapOption.errors.txt +++ b/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/node/mapRootSourceRootWithNoSourceMapOption.errors.txt @@ -1,9 +1,9 @@ error TS5051: Option 'sourceRoot can only be used when either option '--inlineSourceMap' or option '--sourceMap' is provided. -error TS5052: Option 'mapRoot' cannot be specified without specifying option 'sourceMap'. +error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. !!! error TS5051: Option 'sourceRoot can only be used when either option '--inlineSourceMap' or option '--sourceMap' is provided. -!!! error TS5052: Option 'mapRoot' cannot be specified without specifying option 'sourceMap'. +!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. ==== m1.ts (0 errors) ==== var m1_a1 = 10; class m1_c1 { diff --git a/tests/baselines/reference/project/mapRootWithNoSourceMapOption/amd/mapRootWithNoSourceMapOption.errors.txt b/tests/baselines/reference/project/mapRootWithNoSourceMapOption/amd/mapRootWithNoSourceMapOption.errors.txt index 97e3ed02cf915..983c75ac5d3bd 100644 --- a/tests/baselines/reference/project/mapRootWithNoSourceMapOption/amd/mapRootWithNoSourceMapOption.errors.txt +++ b/tests/baselines/reference/project/mapRootWithNoSourceMapOption/amd/mapRootWithNoSourceMapOption.errors.txt @@ -1,7 +1,7 @@ -error TS5052: Option 'mapRoot' cannot be specified without specifying option 'sourceMap'. +error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. -!!! error TS5052: Option 'mapRoot' cannot be specified without specifying option 'sourceMap'. +!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. ==== m1.ts (0 errors) ==== var m1_a1 = 10; class m1_c1 { diff --git a/tests/baselines/reference/project/mapRootWithNoSourceMapOption/node/mapRootWithNoSourceMapOption.errors.txt b/tests/baselines/reference/project/mapRootWithNoSourceMapOption/node/mapRootWithNoSourceMapOption.errors.txt index 97e3ed02cf915..983c75ac5d3bd 100644 --- a/tests/baselines/reference/project/mapRootWithNoSourceMapOption/node/mapRootWithNoSourceMapOption.errors.txt +++ b/tests/baselines/reference/project/mapRootWithNoSourceMapOption/node/mapRootWithNoSourceMapOption.errors.txt @@ -1,7 +1,7 @@ -error TS5052: Option 'mapRoot' cannot be specified without specifying option 'sourceMap'. +error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. -!!! error TS5052: Option 'mapRoot' cannot be specified without specifying option 'sourceMap'. +!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. ==== m1.ts (0 errors) ==== var m1_a1 = 10; class m1_c1 { From 2b231d75449222bbc8a49c18bd5e0793eab9cb50 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 17:48:05 -0700 Subject: [PATCH 17/32] Fix nit --- src/services/services.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/services/services.ts b/src/services/services.ts index 36c4ace29f974..2e954675f2ed0 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1176,7 +1176,7 @@ namespace ts { } /* @internal */ - export function getSourceFileLikeCache(host: { readFile?: (path: string) => string, fileExists?: (path: string) => boolean }): SourceFileLikeCache { + export function createSourceFileLikeCache(host: { readFile?: (path: string) => string, fileExists?: (path: string) => boolean }): SourceFileLikeCache { const cached = createMap(); return { get(path: Path) { @@ -1328,7 +1328,7 @@ namespace ts { // We reset this cache on structure invalidation so we don't hold on to outdated files for long; however we can't use the `compilerHost` above, // Because it only functions until `hostCache` is cleared, while we'll potentially need the functionality to lazily read sourcemap files during // the course of whatever called `synchronizeHostData` - sourcemappedFileCache = getSourceFileLikeCache(host); + sourcemappedFileCache = createSourceFileLikeCache(host); // Make sure all the nodes in the program are both bound, and have their parent // pointers set property. From fed61322b62687d1320bcb63d2c83f5e425d7b1a Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 17:49:04 -0700 Subject: [PATCH 18/32] Use charCodeAt --- src/services/sourcemaps.ts | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts index c6d3075a01043..2cf4595966e34 100644 --- a/src/services/sourcemaps.ts +++ b/src/services/sourcemaps.ts @@ -29,7 +29,7 @@ namespace ts.sourcemaps { log(text: string): void; } - export function decode(host: SourceMapDecodeHost, mapPath: string, map: SourceMapData, program?: Program, fallbackCache = getSourceFileLikeCache(host)): SourceMapper { + export function decode(host: SourceMapDecodeHost, mapPath: string, map: SourceMapData, program?: Program, fallbackCache = createSourceFileLikeCache(host)): SourceMapper { const currentDirectory = getDirectoryPath(mapPath); const sourceRoot = map.sourceRoot || currentDirectory; let decodedMappings: ProcessedSourceMapSpan[]; @@ -179,8 +179,8 @@ namespace ts.sourcemaps { function decodeSingleSpan(state: DecoderState): void { while (state.decodingIndex < state.encodedText.length) { - const char = state.encodedText.charAt(state.decodingIndex); - if (char === ";") { + const char = state.encodedText.charCodeAt(state.decodingIndex); + if (char === CharacterCodes.semicolon) { // New line state.currentEmittedLine++; state.currentEmittedColumn = 1; @@ -188,7 +188,7 @@ namespace ts.sourcemaps { continue; } - if (char === ",") { + if (char === CharacterCodes.comma) { // Next entry is on same line - no action needed state.decodingIndex++; continue; From 509e4f3bbb07d5837969bc430d28b68db6fcb011 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 17:50:42 -0700 Subject: [PATCH 19/32] Rename internal methods + veriables --- src/services/sourcemaps.ts | 16 ++++++++-------- 1 file changed, 8 insertions(+), 8 deletions(-) diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts index 2cf4595966e34..91c2f6cc30b20 100644 --- a/src/services/sourcemaps.ts +++ b/src/services/sourcemaps.ts @@ -33,8 +33,8 @@ namespace ts.sourcemaps { const currentDirectory = getDirectoryPath(mapPath); const sourceRoot = map.sourceRoot || currentDirectory; let decodedMappings: ProcessedSourceMapSpan[]; - let forwardSortedMappings: ProcessedSourceMapSpan[]; - let reverseSortedMappings: ProcessedSourceMapSpan[]; + let generatedOrderedMappings: ProcessedSourceMapSpan[]; + let sourceOrderedMappings: ProcessedSourceMapSpan[]; return { getOriginalPosition, @@ -42,7 +42,7 @@ namespace ts.sourcemaps { }; function getGeneratedPosition(loc: SourceMappableLocation): SourceMappableLocation { - const maps = filter(getForwardSortedMappings(), m => comparePaths(loc.fileName, m.sourcePath, sourceRoot) === 0); + const maps = filter(getGeneratedOrderedMappings(), m => comparePaths(loc.fileName, m.sourcePath, sourceRoot) === 0); if (!length(maps)) return loc; let targetIndex = binarySearch(maps, { sourcePosition: loc.position }, getSourcePosition, compareValues); if (targetIndex < 0 && maps.length > 0) { @@ -53,7 +53,7 @@ namespace ts.sourcemaps { } function getOriginalPosition(loc: SourceMappableLocation): SourceMappableLocation { - const maps = getReverseSortedMappings(); + const maps = getSourceOrderedMappings(); if (!length(maps)) return loc; let targetIndex = binarySearch(maps, { emittedPosition: loc.position }, identity, compareProcessedSpanEmittedPositions); if (targetIndex < 0 && maps.length > 0) { @@ -86,12 +86,12 @@ namespace ts.sourcemaps { return decodedMappings || (decodedMappings = calculateDecodedMappings()); } - function getReverseSortedMappings() { - return reverseSortedMappings || (reverseSortedMappings = getDecodedMappings().slice().sort(compareProcessedSpanSourcePositions)); + function getSourceOrderedMappings() { + return sourceOrderedMappings || (sourceOrderedMappings = getDecodedMappings().slice().sort(compareProcessedSpanSourcePositions)); } - function getForwardSortedMappings() { - return forwardSortedMappings || (forwardSortedMappings = getDecodedMappings().slice().sort(compareProcessedSpanEmittedPositions)); + function getGeneratedOrderedMappings() { + return generatedOrderedMappings || (generatedOrderedMappings = getDecodedMappings().slice().sort(compareProcessedSpanEmittedPositions)); } function calculateDecodedMappings(): ProcessedSourceMapSpan[] { From 0c44ba15284c2bedc0c99ee675e22c32644d17ac Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 17:55:49 -0700 Subject: [PATCH 20/32] Avoid filter --- src/services/sourcemaps.ts | 11 +++++------ 1 file changed, 5 insertions(+), 6 deletions(-) diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts index 91c2f6cc30b20..fb8556dc5fa5a 100644 --- a/src/services/sourcemaps.ts +++ b/src/services/sourcemaps.ts @@ -42,13 +42,16 @@ namespace ts.sourcemaps { }; function getGeneratedPosition(loc: SourceMappableLocation): SourceMappableLocation { - const maps = filter(getGeneratedOrderedMappings(), m => comparePaths(loc.fileName, m.sourcePath, sourceRoot) === 0); + const maps = getGeneratedOrderedMappings(); if (!length(maps)) return loc; - let targetIndex = binarySearch(maps, { sourcePosition: loc.position }, getSourcePosition, compareValues); + let targetIndex = binarySearch(maps, { sourcePath: loc.fileName, sourcePosition: loc.position }, identity, compareProcessedSpanSourcePositions); if (targetIndex < 0 && maps.length > 0) { // if no exact match, closest is 2's compliment of result targetIndex = ~targetIndex; } + if (!maps[targetIndex] || comparePaths(loc.fileName, maps[targetIndex].sourcePath, sourceRoot) !== 0) { + return loc; + } return { fileName: toPath(map.file, sourceRoot, host.getCanonicalFileName), position: maps[targetIndex].emittedPosition }; // Closest span } @@ -138,10 +141,6 @@ namespace ts.sourcemaps { } } - function getSourcePosition(span: ProcessedSourceMapSpan) { - return span.sourcePosition; - } - interface ProcessedSourceMapSpan { emittedPosition: number; sourcePosition: number; From c8620e3dce13e33b39bb5a83b698c32b018557b3 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 18:01:31 -0700 Subject: [PATCH 21/32] span -> position --- src/services/sourcemaps.ts | 64 +++++++++++++++++++------------------- 1 file changed, 32 insertions(+), 32 deletions(-) diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts index fb8556dc5fa5a..6faf8de1df0f9 100644 --- a/src/services/sourcemaps.ts +++ b/src/services/sourcemaps.ts @@ -32,9 +32,9 @@ namespace ts.sourcemaps { export function decode(host: SourceMapDecodeHost, mapPath: string, map: SourceMapData, program?: Program, fallbackCache = createSourceFileLikeCache(host)): SourceMapper { const currentDirectory = getDirectoryPath(mapPath); const sourceRoot = map.sourceRoot || currentDirectory; - let decodedMappings: ProcessedSourceMapSpan[]; - let generatedOrderedMappings: ProcessedSourceMapSpan[]; - let sourceOrderedMappings: ProcessedSourceMapSpan[]; + let decodedMappings: ProcessedSourceMapPosition[]; + let generatedOrderedMappings: ProcessedSourceMapPosition[]; + let sourceOrderedMappings: ProcessedSourceMapPosition[]; return { getOriginalPosition, @@ -44,7 +44,7 @@ namespace ts.sourcemaps { function getGeneratedPosition(loc: SourceMappableLocation): SourceMappableLocation { const maps = getGeneratedOrderedMappings(); if (!length(maps)) return loc; - let targetIndex = binarySearch(maps, { sourcePath: loc.fileName, sourcePosition: loc.position }, identity, compareProcessedSpanSourcePositions); + let targetIndex = binarySearch(maps, { sourcePath: loc.fileName, sourcePosition: loc.position }, identity, compareProcessedPositionSourcePositions); if (targetIndex < 0 && maps.length > 0) { // if no exact match, closest is 2's compliment of result targetIndex = ~targetIndex; @@ -52,18 +52,18 @@ namespace ts.sourcemaps { if (!maps[targetIndex] || comparePaths(loc.fileName, maps[targetIndex].sourcePath, sourceRoot) !== 0) { return loc; } - return { fileName: toPath(map.file, sourceRoot, host.getCanonicalFileName), position: maps[targetIndex].emittedPosition }; // Closest span + return { fileName: toPath(map.file, sourceRoot, host.getCanonicalFileName), position: maps[targetIndex].emittedPosition }; // Closest pos } function getOriginalPosition(loc: SourceMappableLocation): SourceMappableLocation { const maps = getSourceOrderedMappings(); if (!length(maps)) return loc; - let targetIndex = binarySearch(maps, { emittedPosition: loc.position }, identity, compareProcessedSpanEmittedPositions); + let targetIndex = binarySearch(maps, { emittedPosition: loc.position }, identity, compareProcessedPositionEmittedPositions); if (targetIndex < 0 && maps.length > 0) { // if no exact match, closest is 2's compliment of result targetIndex = ~targetIndex; } - return { fileName: toPath(maps[targetIndex].sourcePath, sourceRoot, host.getCanonicalFileName), position: maps[targetIndex].sourcePosition }; // Closest span + return { fileName: toPath(maps[targetIndex].sourcePath, sourceRoot, host.getCanonicalFileName), position: maps[targetIndex].sourcePosition }; // Closest pos } function getSourceFileLike(fileName: string, location: string): SourceFileLike | undefined { @@ -90,15 +90,15 @@ namespace ts.sourcemaps { } function getSourceOrderedMappings() { - return sourceOrderedMappings || (sourceOrderedMappings = getDecodedMappings().slice().sort(compareProcessedSpanSourcePositions)); + return sourceOrderedMappings || (sourceOrderedMappings = getDecodedMappings().slice().sort(compareProcessedPositionSourcePositions)); } function getGeneratedOrderedMappings() { - return generatedOrderedMappings || (generatedOrderedMappings = getDecodedMappings().slice().sort(compareProcessedSpanEmittedPositions)); + return generatedOrderedMappings || (generatedOrderedMappings = getDecodedMappings().slice().sort(compareProcessedPositionEmittedPositions)); } - function calculateDecodedMappings(): ProcessedSourceMapSpan[] { - const state: DecoderState = { + function calculateDecodedMappings(): ProcessedSourceMapPosition[] { + const state: DecoderState = { encodedText: map.mappings, currentNameIndex: undefined, sourceMapNamesLength: map.names ? map.names.length : undefined, @@ -107,48 +107,48 @@ namespace ts.sourcemaps { currentSourceColumn: 1, currentSourceLine: 1, currentSourceIndex: 0, - spans: [], + positions: [], decodingIndex: 0, - processSpan, + processPosition, }; while (!hasCompletedDecoding(state)) { - decodeSingleSpan(state); + decodeSinglePosition(state); if (state.error) { host.log(`Encountered error while decoding sourcemap found at ${mapPath}: ${state.error}`); return []; } } - return state.spans; + return state.positions; } - function compareProcessedSpanSourcePositions(a: ProcessedSourceMapSpan, b: ProcessedSourceMapSpan) { + function compareProcessedPositionSourcePositions(a: ProcessedSourceMapPosition, b: ProcessedSourceMapPosition) { return comparePaths(a.sourcePath, b.sourcePath, sourceRoot) || compareValues(a.sourcePosition, b.sourcePosition); } - function compareProcessedSpanEmittedPositions(a: ProcessedSourceMapSpan, b: ProcessedSourceMapSpan) { + function compareProcessedPositionEmittedPositions(a: ProcessedSourceMapPosition, b: ProcessedSourceMapPosition) { return compareValues(a.emittedPosition, b.emittedPosition); } - function processSpan(span: RawSourceMapSpan): ProcessedSourceMapSpan { - const sourcePath = map.sources[span.sourceIndex]; + function processPosition(position: RawSourceMapPosition): ProcessedSourceMapPosition { + const sourcePath = map.sources[position.sourceIndex]; return { - emittedPosition: getPositionOfLineAndCharacterUsingName(map.file, currentDirectory, span.emittedLine - 1, span.emittedColumn - 1), - sourcePosition: getPositionOfLineAndCharacterUsingName(sourcePath, sourceRoot, span.sourceLine - 1, span.sourceColumn - 1), + emittedPosition: getPositionOfLineAndCharacterUsingName(map.file, currentDirectory, position.emittedLine - 1, position.emittedColumn - 1), + sourcePosition: getPositionOfLineAndCharacterUsingName(sourcePath, sourceRoot, position.sourceLine - 1, position.sourceColumn - 1), sourcePath, - name: span.nameIndex ? map.names[span.nameIndex] : undefined + name: position.nameIndex ? map.names[position.nameIndex] : undefined }; } } - interface ProcessedSourceMapSpan { + interface ProcessedSourceMapPosition { emittedPosition: number; sourcePosition: number; sourcePath: string; name?: string; } - interface RawSourceMapSpan { + interface RawSourceMapPosition { emittedLine: number; emittedColumn: number; sourceLine: number; @@ -168,15 +168,15 @@ namespace ts.sourcemaps { encodedText: string; sourceMapNamesLength?: number; error?: string; - spans: T[]; - processSpan: (span: RawSourceMapSpan) => T; + positions: T[]; + processPosition: (position: RawSourceMapPosition) => T; } function hasCompletedDecoding(state: DecoderState) { return state.decodingIndex === state.encodedText.length; } - function decodeSingleSpan(state: DecoderState): void { + function decodeSinglePosition(state: DecoderState): void { while (state.decodingIndex < state.encodedText.length) { const char = state.encodedText.charCodeAt(state.decodingIndex); if (char === CharacterCodes.semicolon) { @@ -193,7 +193,7 @@ namespace ts.sourcemaps { continue; } - // Read the current span + // Read the current position // 1. Column offset from prev read jsColumn state.currentEmittedColumn += base64VLQFormatDecode(); // Incorrect emittedColumn dont support this map @@ -211,7 +211,7 @@ namespace ts.sourcemaps { if (createErrorIfCondition(state.currentSourceIndex < 0, "Invalid sourceIndex found")) { return; } - // Dont support reading mappings that dont have information about original source span + // Dont support reading mappings that dont have information about original source position if (createErrorIfCondition(isSourceMappingSegmentEnd(state.encodedText, state.decodingIndex), "Unsupported Error Format: No entries after sourceIndex")) { return; } @@ -250,15 +250,15 @@ namespace ts.sourcemaps { } // Entry should be complete - captureSpan(); + capturePosition(); return; } createErrorIfCondition(/*condition*/ true, "No encoded entry found"); return; - function captureSpan() { - state.spans.push(state.processSpan({ + function capturePosition() { + state.positions.push(state.processPosition({ emittedColumn: state.currentEmittedColumn, emittedLine: state.currentEmittedLine, sourceColumn: state.currentSourceColumn, From 5687e1078fe275e396005d91c872994f0bc9c825 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 18:02:14 -0700 Subject: [PATCH 22/32] Use character codes --- src/services/sourcemaps.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts index 6faf8de1df0f9..3051d33b73479 100644 --- a/src/services/sourcemaps.ts +++ b/src/services/sourcemaps.ts @@ -323,7 +323,7 @@ namespace ts.sourcemaps { function isSourceMappingSegmentEnd(encodedText: string, pos: number) { return (pos === encodedText.length || - encodedText.charAt(pos) === "," || - encodedText.charAt(pos) === ";"); + encodedText.charCodeAt(pos) === CharacterCodes.comma || + encodedText.charCodeAt(pos) === CharacterCodes.semicolon); } } \ No newline at end of file From e64e73d3f616f61c0b2454fe2fb39ff56a9cc64c Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 18:05:28 -0700 Subject: [PATCH 23/32] Dont parse our sourcemap names until we need to start using them --- src/services/sourcemaps.ts | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts index 3051d33b73479..f343b17d21a15 100644 --- a/src/services/sourcemaps.ts +++ b/src/services/sourcemaps.ts @@ -136,7 +136,8 @@ namespace ts.sourcemaps { emittedPosition: getPositionOfLineAndCharacterUsingName(map.file, currentDirectory, position.emittedLine - 1, position.emittedColumn - 1), sourcePosition: getPositionOfLineAndCharacterUsingName(sourcePath, sourceRoot, position.sourceLine - 1, position.sourceColumn - 1), sourcePath, - name: position.nameIndex ? map.names[position.nameIndex] : undefined + // TODO: Consider using `name` field to remap the expected identifier to scan for renames to handle another tool renaming oout output + // name: position.nameIndex ? map.names[position.nameIndex] : undefined }; } } @@ -145,7 +146,6 @@ namespace ts.sourcemaps { emittedPosition: number; sourcePosition: number; sourcePath: string; - name?: string; } interface RawSourceMapPosition { @@ -240,9 +240,10 @@ namespace ts.sourcemaps { } state.currentNameIndex += base64VLQFormatDecode(); // Incorrect nameIndex dont support this map - if (createErrorIfCondition(state.currentNameIndex < 0 || state.currentNameIndex >= state.sourceMapNamesLength, "Invalid name index for the source map entry")) { - return; - } + // TODO: If we start using `name`s, issue errors when they aren't correct in the sourcemap + // if (createErrorIfCondition(state.currentNameIndex < 0 || state.currentNameIndex >= state.sourceMapNamesLength, "Invalid name index for the source map entry")) { + // return; + // } } // Dont support reading mappings that dont have information about original source and its line numbers if (createErrorIfCondition(!isSourceMappingSegmentEnd(state.encodedText, state.decodingIndex), "Unsupported Error Format: There are more entries after " + (state.currentNameIndex === undefined ? "sourceColumn" : "nameIndex"))) { From 47e701ae036eb91106321508b09d260b8b71fc95 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 18:11:49 -0700 Subject: [PATCH 24/32] zero-index parsed positions --- src/services/sourcemaps.ts | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/src/services/sourcemaps.ts b/src/services/sourcemaps.ts index f343b17d21a15..c24ed0f64a8f4 100644 --- a/src/services/sourcemaps.ts +++ b/src/services/sourcemaps.ts @@ -102,10 +102,10 @@ namespace ts.sourcemaps { encodedText: map.mappings, currentNameIndex: undefined, sourceMapNamesLength: map.names ? map.names.length : undefined, - currentEmittedColumn: 1, - currentEmittedLine: 1, - currentSourceColumn: 1, - currentSourceLine: 1, + currentEmittedColumn: 0, + currentEmittedLine: 0, + currentSourceColumn: 0, + currentSourceLine: 0, currentSourceIndex: 0, positions: [], decodingIndex: 0, @@ -133,8 +133,8 @@ namespace ts.sourcemaps { function processPosition(position: RawSourceMapPosition): ProcessedSourceMapPosition { const sourcePath = map.sources[position.sourceIndex]; return { - emittedPosition: getPositionOfLineAndCharacterUsingName(map.file, currentDirectory, position.emittedLine - 1, position.emittedColumn - 1), - sourcePosition: getPositionOfLineAndCharacterUsingName(sourcePath, sourceRoot, position.sourceLine - 1, position.sourceColumn - 1), + emittedPosition: getPositionOfLineAndCharacterUsingName(map.file, currentDirectory, position.emittedLine, position.emittedColumn), + sourcePosition: getPositionOfLineAndCharacterUsingName(sourcePath, sourceRoot, position.sourceLine, position.sourceColumn), sourcePath, // TODO: Consider using `name` field to remap the expected identifier to scan for renames to handle another tool renaming oout output // name: position.nameIndex ? map.names[position.nameIndex] : undefined @@ -182,7 +182,7 @@ namespace ts.sourcemaps { if (char === CharacterCodes.semicolon) { // New line state.currentEmittedLine++; - state.currentEmittedColumn = 1; + state.currentEmittedColumn = 0; state.decodingIndex++; continue; } @@ -197,7 +197,7 @@ namespace ts.sourcemaps { // 1. Column offset from prev read jsColumn state.currentEmittedColumn += base64VLQFormatDecode(); // Incorrect emittedColumn dont support this map - if (createErrorIfCondition(state.currentEmittedColumn < 1, "Invalid emittedColumn found")) { + if (createErrorIfCondition(state.currentEmittedColumn < 0, "Invalid emittedColumn found")) { return; } // Dont support reading mappings that dont have information about original source and its line numbers @@ -219,7 +219,7 @@ namespace ts.sourcemaps { // 3. Relative sourceLine 0 based state.currentSourceLine += base64VLQFormatDecode(); // Incorrect sourceLine dont support this map - if (createErrorIfCondition(state.currentSourceLine < 1, "Invalid sourceLine found")) { + if (createErrorIfCondition(state.currentSourceLine < 0, "Invalid sourceLine found")) { return; } // Dont support reading mappings that dont have information about original source and its line numbers @@ -230,7 +230,7 @@ namespace ts.sourcemaps { // 4. Relative sourceColumn 0 based state.currentSourceColumn += base64VLQFormatDecode(); // Incorrect sourceColumn dont support this map - if (createErrorIfCondition(state.currentSourceColumn < 1, "Invalid sourceLine found")) { + if (createErrorIfCondition(state.currentSourceColumn < 0, "Invalid sourceLine found")) { return; } // 5. Check if there is name: From 6a467ba0415597bd8b7ce993b1ba620f74a077d9 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 19 Mar 2018 19:30:48 -0700 Subject: [PATCH 25/32] Handle sourceMappingURL comments, including base64 encoded ones --- src/compiler/sys.ts | 6 ++ src/compiler/utilities.ts | 36 +++++++++++ src/services/services.ts | 61 +++++++++++++++---- .../reference/api/tsserverlibrary.d.ts | 1 + tests/baselines/reference/api/typescript.d.ts | 1 + 5 files changed, 93 insertions(+), 12 deletions(-) diff --git a/src/compiler/sys.ts b/src/compiler/sys.ts index e3c304afb8f36..edf007335f681 100644 --- a/src/compiler/sys.ts +++ b/src/compiler/sys.ts @@ -457,6 +457,7 @@ namespace ts { clearTimeout?(timeoutId: any): void; clearScreen?(): void; /*@internal*/ setBlocking?(): void; + base64decode?(input: string): string; } export interface FileWatcher { @@ -528,6 +529,8 @@ namespace ts { _crypto = undefined; } + const Buffer: typeof global.Buffer = require("buffer").Buffer; + const nodeVersion = getNodeMajorVersion(); const isNode4OrLater = nodeVersion >= 4; @@ -620,6 +623,9 @@ namespace ts { if (process.stdout && process.stdout._handle && process.stdout._handle.setBlocking) { process.stdout._handle.setBlocking(true); } + }, + base64decode: input => { + return Buffer.from(input, "base64").toString("utf8"); } }; return nodeSystem; diff --git a/src/compiler/utilities.ts b/src/compiler/utilities.ts index e756fa5763baa..b95644a85aea6 100644 --- a/src/compiler/utilities.ts +++ b/src/compiler/utilities.ts @@ -3463,6 +3463,42 @@ namespace ts { return result; } + export function base64decode(host: { base64decode?(input: string): string }, input: string): string { + if (host.base64decode) { + return host.base64decode(input); + } + let result = ""; + const length = input.length; + let i = 0; + while (i < length) { + // Stop decoding once padding characters are present + if (input.charCodeAt(i) === base64Digits.charCodeAt(64)) { + break; + } + // convert 4 input digits into three characters, ignoring padding characters at the end + const ch1 = base64Digits.indexOf(input[i]); + const ch2 = base64Digits.indexOf(input[i + 1]); + const ch3 = base64Digits.indexOf(input[i + 2]); + const ch4 = base64Digits.indexOf(input[i + 3]); + + const code1 = (ch1 & 0B00111111 << 2) | (ch2 & 0B00000011); + const code2 = (ch2 & 0B00001111 << 4) | (ch3 & 0B00001111); + const code3 = (ch3 & 0B00000011 << 6) | (ch4 & 0B00111111); + + if (code2 === 0 && ch3 !== 0) { // code2 decoded to zero, but ch3 was padding - elide code2 and code3 + result += String.fromCharCode(code1); + } + else if (code3 === 0 && ch4 !== 0) { // code3 decoded to zero, but ch4 was padding, elide code3 + result += `${String.fromCharCode(code1)}${String.fromCharCode(code2)}`; + } + else { + result += `${String.fromCharCode(code1)}${String.fromCharCode(code2)}${String.fromCharCode(code3)}`; + } + i += 4; + } + return result; + } + const carriageReturnLineFeed = "\r\n"; const lineFeed = "\n"; export function getNewLineCharacter(options: CompilerOptions | PrinterOptions, getNewLine?: () => string): string { diff --git a/src/services/services.ts b/src/services/services.ts index 2e954675f2ed0..8e69234383cfb 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1573,22 +1573,29 @@ namespace ts { return checker.getSymbolAtLocation(node); } - function getSourceMapper(fileName: string, file: { sourceMapper?: sourcemaps.SourceMapper }) { - if (!host.readFile || !host.fileExists) { - return file.sourceMapper = sourcemaps.identitySourceMapper; - } - if (file.sourceMapper) { - return file.sourceMapper; + + const sourceMapCommentRE = /^\/\/[@#] sourceMappingURL=(.+)$/gm; + const dataURLRE = /^data:/; + const base64URLRE = /^data:application\/json;charset=utf-8;base64,(.+)$/; + function scanForSourcemapURL(fileName: string) { + const mappedFile = sourcemappedFileCache.get(toPath(fileName, currentDirectory, getCanonicalFileName)); + if (!mappedFile) { + return; } - // TODO (weswigham): Read sourcemappingurl from last line of .d.ts if present - const mapFileName = fileName + ".map"; - if (!host.fileExists(mapFileName)) { - return file.sourceMapper = sourcemaps.identitySourceMapper; + const starts = getLineStarts(mappedFile); + for (let index = starts.length - 1; index--; index >= 0) { + sourceMapCommentRE.lastIndex = starts[index]; + const comment = sourceMapCommentRE.exec(mappedFile.text); + if (comment) { + return comment[1]; + } } - const doc = host.readFile(mapFileName); + } + + function convertDocumentToSourceMapper(file: { sourceMapper?: sourcemaps.SourceMapper }, contents: string, mapFileName: string) { let maps: sourcemaps.SourceMapData; try { - maps = JSON.parse(doc); + maps = JSON.parse(contents); } catch { // swallow error @@ -1605,6 +1612,36 @@ namespace ts { }, mapFileName, maps, program, sourcemappedFileCache); } + function getSourceMapper(fileName: string, file: { sourceMapper?: sourcemaps.SourceMapper }) { + if (!host.readFile || !host.fileExists) { + return file.sourceMapper = sourcemaps.identitySourceMapper; + } + if (file.sourceMapper) { + return file.sourceMapper; + } + let mapFileName = scanForSourcemapURL(fileName); + if (mapFileName && dataURLRE.exec(mapFileName)) { + const b64EncodedMatch = base64URLRE.exec(mapFileName); + if (b64EncodedMatch) { + const base64Object = b64EncodedMatch[1]; + return convertDocumentToSourceMapper(file, base64decode(sys, base64Object), fileName); + } + mapFileName = undefined; + } + const possibleMapLocations: string[] = []; + if (mapFileName) { + possibleMapLocations.push(mapFileName); + } + possibleMapLocations.push(fileName + ".map"); + for (const location of possibleMapLocations) { + const mapPath = toPath(location, getDirectoryPath(fileName), getCanonicalFileName); + if (host.fileExists(mapPath)) { + return convertDocumentToSourceMapper(file, host.readFile(mapPath), mapPath); + } + } + return file.sourceMapper = sourcemaps.identitySourceMapper; + } + function makeGetTargetOfMappedPosition( extract: (original: TIn) => sourcemaps.SourceMappableLocation, create: (result: sourcemaps.SourceMappableLocation, original: TIn) => TIn diff --git a/tests/baselines/reference/api/tsserverlibrary.d.ts b/tests/baselines/reference/api/tsserverlibrary.d.ts index afc70224c1aab..7bcd0f933fda3 100644 --- a/tests/baselines/reference/api/tsserverlibrary.d.ts +++ b/tests/baselines/reference/api/tsserverlibrary.d.ts @@ -2893,6 +2893,7 @@ declare namespace ts { setTimeout?(callback: (...args: any[]) => void, ms: number, ...args: any[]): any; clearTimeout?(timeoutId: any): void; clearScreen?(): void; + base64decode?(input: string): string; } interface FileWatcher { close(): void; diff --git a/tests/baselines/reference/api/typescript.d.ts b/tests/baselines/reference/api/typescript.d.ts index 87627051634af..af13d1d0f248c 100644 --- a/tests/baselines/reference/api/typescript.d.ts +++ b/tests/baselines/reference/api/typescript.d.ts @@ -2893,6 +2893,7 @@ declare namespace ts { setTimeout?(callback: (...args: any[]) => void, ms: number, ...args: any[]): any; clearTimeout?(timeoutId: any): void; clearScreen?(): void; + base64decode?(input: string): string; } interface FileWatcher { close(): void; From f2c8d3fa82beb4a248b9431ef89f732e39facd02 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Tue, 20 Mar 2018 13:17:18 -0700 Subject: [PATCH 26/32] Unittest b64 decoder, make mroe robust to handle unicode properly --- Jakefile.js | 1 + src/compiler/utilities.ts | 51 +++++++++++++++++++++++++++------ src/harness/tsconfig.json | 1 + src/harness/unittests/base64.ts | 23 +++++++++++++++ 4 files changed, 67 insertions(+), 9 deletions(-) create mode 100644 src/harness/unittests/base64.ts diff --git a/Jakefile.js b/Jakefile.js index 584a9aa82b3a1..f2821a0168642 100644 --- a/Jakefile.js +++ b/Jakefile.js @@ -115,6 +115,7 @@ var harnessCoreSources = [ }); var harnessSources = harnessCoreSources.concat([ + "base64.ts", "incrementalParser.ts", "jsDocParsing.ts", "services/colorization.ts", diff --git a/src/compiler/utilities.ts b/src/compiler/utilities.ts index b95644a85aea6..4a6abd135de0b 100644 --- a/src/compiler/utilities.ts +++ b/src/compiler/utilities.ts @@ -3398,7 +3398,7 @@ namespace ts { for (let i = 0; i < length; i++) { const charCode = input.charCodeAt(i); - // handel utf8 + // handle utf8 if (charCode < 0x80) { output.push(charCode); } @@ -3463,12 +3463,45 @@ namespace ts { return result; } + function getStringFromExpandedCharCodes(codes: number[]): string { + let output = ""; + let i = 0; + const length = codes.length; + while (i < length) { + const charCode = codes[i]; + + if (charCode < 0x80) { + output += String.fromCharCode(charCode); + i++; + } + else if ((charCode & 0B11000000) === 0B11000000) { + let value = charCode & 0B00111111; + i++; + let nextCode: number = codes[i]; + while ((nextCode & 0B11000000) === 0B10000000) { + value = (value << 6) | (nextCode & 0B00111111); + i++; + nextCode = codes[i]; + } + // `value` may be greater than 10FFFF (the maximum unicode codepoint) - JS will just make this into an invalid character for us + output += String.fromCharCode(value); + } + else { + // We don't want to kill the process when decoding fails (due to a following char byte not + // following a leading char), so we just print the (bad) value + output += String.fromCharCode(charCode); + i++; + } + } + return output; + } + export function base64decode(host: { base64decode?(input: string): string }, input: string): string { if (host.base64decode) { return host.base64decode(input); } - let result = ""; const length = input.length; + let expandedCharCodes: number[] = []; let i = 0; while (i < length) { // Stop decoding once padding characters are present @@ -3481,22 +3514,22 @@ namespace ts { const ch3 = base64Digits.indexOf(input[i + 2]); const ch4 = base64Digits.indexOf(input[i + 3]); - const code1 = (ch1 & 0B00111111 << 2) | (ch2 & 0B00000011); - const code2 = (ch2 & 0B00001111 << 4) | (ch3 & 0B00001111); - const code3 = (ch3 & 0B00000011 << 6) | (ch4 & 0B00111111); + const code1 = ((ch1 & 0B00111111) << 2) | ((ch2 >> 4) & 0B00000011); + const code2 = ((ch2 & 0B00001111) << 4) | ((ch3 >> 2) & 0B00001111); + const code3 = ((ch3 & 0B00000011) << 6) | (ch4 & 0B00111111); if (code2 === 0 && ch3 !== 0) { // code2 decoded to zero, but ch3 was padding - elide code2 and code3 - result += String.fromCharCode(code1); + expandedCharCodes.push(code1); } else if (code3 === 0 && ch4 !== 0) { // code3 decoded to zero, but ch4 was padding, elide code3 - result += `${String.fromCharCode(code1)}${String.fromCharCode(code2)}`; + expandedCharCodes.push(code1, code2); } else { - result += `${String.fromCharCode(code1)}${String.fromCharCode(code2)}${String.fromCharCode(code3)}`; + expandedCharCodes.push(code1, code2, code3); } i += 4; } - return result; + return getStringFromExpandedCharCodes(expandedCharCodes); } const carriageReturnLineFeed = "\r\n"; diff --git a/src/harness/tsconfig.json b/src/harness/tsconfig.json index 6b5a42c2d98e0..54766ed02eca4 100644 --- a/src/harness/tsconfig.json +++ b/src/harness/tsconfig.json @@ -95,6 +95,7 @@ "../server/session.ts", "../server/client.ts", "../server/editorServices.ts", + "./unittests/base64.ts", "./unittests/incrementalParser.ts", "./unittests/jsDocParsing.ts", "./unittests/services/colorization.ts", diff --git a/src/harness/unittests/base64.ts b/src/harness/unittests/base64.ts new file mode 100644 index 0000000000000..b503955ee68d9 --- /dev/null +++ b/src/harness/unittests/base64.ts @@ -0,0 +1,23 @@ +/// +namespace ts { + describe("base64", () => { + describe("base64decode", () => { + it("can decode input strings correctly without needing a host implementation", () => { + const tests = [ + // "a", + // "this is a test", + // " !\"#$ %&'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~", + "日本語", + "🐱", + "\x00\x01", + "\t\n\r\\\"\'\u0062", + "====", + "", + ]; + for (const test of tests) { + assert.equal(base64decode({}, convertToBase64(test)), test); + } + }); + }); + }); +} From bd9cccffe477d5aa6806c43c184297eff474c33a Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Tue, 20 Mar 2018 13:47:06 -0700 Subject: [PATCH 27/32] Fix lint --- src/compiler/utilities.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/compiler/utilities.ts b/src/compiler/utilities.ts index 4a6abd135de0b..336e9081acee4 100644 --- a/src/compiler/utilities.ts +++ b/src/compiler/utilities.ts @@ -3501,7 +3501,7 @@ namespace ts { return host.base64decode(input); } const length = input.length; - let expandedCharCodes: number[] = []; + const expandedCharCodes: number[] = []; let i = 0; while (i < length) { // Stop decoding once padding characters are present From c7a52967e887e9a882c48bcc075d2f3fbb32f164 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Tue, 20 Mar 2018 14:04:07 -0700 Subject: [PATCH 28/32] declarationMaps -> declarationMap --- src/compiler/commandLineParser.ts | 2 +- src/compiler/core.ts | 2 +- src/compiler/emitter.ts | 2 +- src/compiler/program.ts | 8 ++++---- src/compiler/types.ts | 2 +- src/harness/compilerRunner.ts | 2 +- tests/baselines/reference/api/tsserverlibrary.d.ts | 2 +- tests/baselines/reference/api/typescript.d.ts | 2 +- .../declarationMapsWithoutDeclaration.errors.txt | 4 ++-- .../reference/optionsInlineSourceMapMapRoot.errors.txt | 4 ++-- .../amd/mapRootSourceRootWithNoSourceMapOption.errors.txt | 4 ++-- .../mapRootSourceRootWithNoSourceMapOption.errors.txt | 4 ++-- .../amd/mapRootWithNoSourceMapOption.errors.txt | 4 ++-- .../node/mapRootWithNoSourceMapOption.errors.txt | 4 ++-- .../tsConfig/Default initialized TSConfig/tsconfig.json | 2 +- .../tsconfig.json | 2 +- .../tsconfig.json | 2 +- .../Initialized TSConfig with files options/tsconfig.json | 2 +- .../tsconfig.json | 2 +- .../tsconfig.json | 2 +- .../tsconfig.json | 2 +- .../tsconfig.json | 2 +- tests/cases/compiler/declarationMaps.ts | 2 +- tests/cases/compiler/declarationMapsMultifile.ts | 2 +- tests/cases/compiler/declarationMapsOutFile.ts | 2 +- tests/cases/compiler/declarationMapsOutFile2.ts | 2 +- tests/cases/compiler/declarationMapsWithSourceMap.ts | 2 +- tests/cases/compiler/declarationMapsWithoutDeclaration.ts | 2 +- .../fourslash/declarationMapsEnableMapping_NoInline.ts | 2 +- .../declarationMapsEnableMapping_NoInlineSources.ts | 2 +- .../declarationMapsGeneratedMapsEnableMapping.ts | 2 +- .../declarationMapsGeneratedMapsEnableMapping2.ts | 2 +- .../declarationMapsGeneratedMapsEnableMapping3.ts | 2 +- 33 files changed, 42 insertions(+), 42 deletions(-) diff --git a/src/compiler/commandLineParser.ts b/src/compiler/commandLineParser.ts index 9e805d80d3af1..e15cd5df7db0a 100644 --- a/src/compiler/commandLineParser.ts +++ b/src/compiler/commandLineParser.ts @@ -195,7 +195,7 @@ namespace ts { description: Diagnostics.Generates_corresponding_d_ts_file, }, { - name: "declarationMaps", + name: "declarationMap", type: "boolean", showInSimplifiedHelpView: true, category: Diagnostics.Basic_Options, diff --git a/src/compiler/core.ts b/src/compiler/core.ts index b55d3dd79b71c..4e73a07e6e336 100644 --- a/src/compiler/core.ts +++ b/src/compiler/core.ts @@ -2066,7 +2066,7 @@ namespace ts { } export function getAreDeclarationMapsEnabled(options: CompilerOptions) { - return !!(options.declaration && options.declarationMaps); + return !!(options.declaration && options.declarationMap); } export function getAllowSyntheticDefaultImports(compilerOptions: CompilerOptions) { diff --git a/src/compiler/emitter.ts b/src/compiler/emitter.ts index 188c6d450383d..a3fc5b4d42f0e 100644 --- a/src/compiler/emitter.ts +++ b/src/compiler/emitter.ts @@ -97,7 +97,7 @@ namespace ts { const writer = createTextWriter(newLine); const sourceMap = createSourceMapWriter(host, writer); const declarationSourceMap = createSourceMapWriter(host, writer, { - sourceMap: compilerOptions.declarationMaps, + sourceMap: compilerOptions.declarationMap, sourceRoot: compilerOptions.sourceRoot, mapRoot: compilerOptions.mapRoot, extendedDiagnostics: compilerOptions.extendedDiagnostics, diff --git a/src/compiler/program.ts b/src/compiler/program.ts index 34e5d65b7921c..2b844efa7d0df 100755 --- a/src/compiler/program.ts +++ b/src/compiler/program.ts @@ -2111,9 +2111,9 @@ namespace ts { createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_with_option_1, "out", "outFile"); } - if (options.mapRoot && !(options.sourceMap || options.declarationMaps)) { + if (options.mapRoot && !(options.sourceMap || options.declarationMap)) { // Error to specify --mapRoot without --sourcemap - createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1_or_option_2, "mapRoot", "sourceMap", "declarationMaps"); + createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1_or_option_2, "mapRoot", "sourceMap", "declarationMap"); } if (options.declarationDir) { @@ -2125,8 +2125,8 @@ namespace ts { } } - if (options.declarationMaps && !options.declaration) { - createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "declarationMaps", "declaration"); + if (options.declarationMap && !options.declaration) { + createDiagnosticForOptionName(Diagnostics.Option_0_cannot_be_specified_without_specifying_option_1, "declarationMap", "declaration"); } if (options.lib && options.noLib) { diff --git a/src/compiler/types.ts b/src/compiler/types.ts index 64b361b60efc6..2961825d35347 100644 --- a/src/compiler/types.ts +++ b/src/compiler/types.ts @@ -4097,7 +4097,7 @@ namespace ts { /** configFile is set as non enumerable property so as to avoid checking of json source files */ /* @internal */ readonly configFile?: JsonSourceFile; declaration?: boolean; - declarationMaps?: boolean; + declarationMap?: boolean; emitDeclarationOnly?: boolean; declarationDir?: string; /* @internal */ diagnostics?: boolean; diff --git a/src/harness/compilerRunner.ts b/src/harness/compilerRunner.ts index 630116d6a6fb7..4edff00b2887e 100644 --- a/src/harness/compilerRunner.ts +++ b/src/harness/compilerRunner.ts @@ -148,7 +148,7 @@ class CompilerBaselineRunner extends RunnerBase { // Source maps? it("Correct sourcemap content for " + fileName, () => { - if (options.sourceMap || options.inlineSourceMap || options.declarationMaps) { + if (options.sourceMap || options.inlineSourceMap || options.declarationMap) { Harness.Baseline.runBaseline(justName.replace(/\.tsx?$/, ".sourcemap.txt"), () => { const record = result.getSourceMapRecord(); if ((options.noEmitOnError && result.errors.length !== 0) || record === undefined) { diff --git a/tests/baselines/reference/api/tsserverlibrary.d.ts b/tests/baselines/reference/api/tsserverlibrary.d.ts index 7bcd0f933fda3..ca810039b4d63 100644 --- a/tests/baselines/reference/api/tsserverlibrary.d.ts +++ b/tests/baselines/reference/api/tsserverlibrary.d.ts @@ -2302,7 +2302,7 @@ declare namespace ts { charset?: string; checkJs?: boolean; declaration?: boolean; - declarationMaps?: boolean; + declarationMap?: boolean; emitDeclarationOnly?: boolean; declarationDir?: string; disableSizeLimit?: boolean; diff --git a/tests/baselines/reference/api/typescript.d.ts b/tests/baselines/reference/api/typescript.d.ts index af13d1d0f248c..b114b5a6a9625 100644 --- a/tests/baselines/reference/api/typescript.d.ts +++ b/tests/baselines/reference/api/typescript.d.ts @@ -2302,7 +2302,7 @@ declare namespace ts { charset?: string; checkJs?: boolean; declaration?: boolean; - declarationMaps?: boolean; + declarationMap?: boolean; emitDeclarationOnly?: boolean; declarationDir?: string; disableSizeLimit?: boolean; diff --git a/tests/baselines/reference/declarationMapsWithoutDeclaration.errors.txt b/tests/baselines/reference/declarationMapsWithoutDeclaration.errors.txt index 15afcf6622e31..c432f44513058 100644 --- a/tests/baselines/reference/declarationMapsWithoutDeclaration.errors.txt +++ b/tests/baselines/reference/declarationMapsWithoutDeclaration.errors.txt @@ -1,7 +1,7 @@ -error TS5052: Option 'declarationMaps' cannot be specified without specifying option 'declaration'. +error TS5052: Option 'declarationMap' cannot be specified without specifying option 'declaration'. -!!! error TS5052: Option 'declarationMaps' cannot be specified without specifying option 'declaration'. +!!! error TS5052: Option 'declarationMap' cannot be specified without specifying option 'declaration'. ==== tests/cases/compiler/declarationMapsWithoutDeclaration.ts (0 errors) ==== module m2 { export interface connectModule { diff --git a/tests/baselines/reference/optionsInlineSourceMapMapRoot.errors.txt b/tests/baselines/reference/optionsInlineSourceMapMapRoot.errors.txt index 0520441c0cfbd..09c29b6ea65b9 100644 --- a/tests/baselines/reference/optionsInlineSourceMapMapRoot.errors.txt +++ b/tests/baselines/reference/optionsInlineSourceMapMapRoot.errors.txt @@ -1,8 +1,8 @@ error TS5053: Option 'mapRoot' cannot be specified with option 'inlineSourceMap'. -error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. +error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMap'. !!! error TS5053: Option 'mapRoot' cannot be specified with option 'inlineSourceMap'. -!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. +!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMap'. ==== tests/cases/compiler/optionsInlineSourceMapMapRoot.ts (0 errors) ==== var a = 10; \ No newline at end of file diff --git a/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/amd/mapRootSourceRootWithNoSourceMapOption.errors.txt b/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/amd/mapRootSourceRootWithNoSourceMapOption.errors.txt index a47daea28c7e8..aa02b659d5de9 100644 --- a/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/amd/mapRootSourceRootWithNoSourceMapOption.errors.txt +++ b/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/amd/mapRootSourceRootWithNoSourceMapOption.errors.txt @@ -1,9 +1,9 @@ error TS5051: Option 'sourceRoot can only be used when either option '--inlineSourceMap' or option '--sourceMap' is provided. -error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. +error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMap'. !!! error TS5051: Option 'sourceRoot can only be used when either option '--inlineSourceMap' or option '--sourceMap' is provided. -!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. +!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMap'. ==== m1.ts (0 errors) ==== var m1_a1 = 10; class m1_c1 { diff --git a/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/node/mapRootSourceRootWithNoSourceMapOption.errors.txt b/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/node/mapRootSourceRootWithNoSourceMapOption.errors.txt index a47daea28c7e8..aa02b659d5de9 100644 --- a/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/node/mapRootSourceRootWithNoSourceMapOption.errors.txt +++ b/tests/baselines/reference/project/mapRootSourceRootWithNoSourceMapOption/node/mapRootSourceRootWithNoSourceMapOption.errors.txt @@ -1,9 +1,9 @@ error TS5051: Option 'sourceRoot can only be used when either option '--inlineSourceMap' or option '--sourceMap' is provided. -error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. +error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMap'. !!! error TS5051: Option 'sourceRoot can only be used when either option '--inlineSourceMap' or option '--sourceMap' is provided. -!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. +!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMap'. ==== m1.ts (0 errors) ==== var m1_a1 = 10; class m1_c1 { diff --git a/tests/baselines/reference/project/mapRootWithNoSourceMapOption/amd/mapRootWithNoSourceMapOption.errors.txt b/tests/baselines/reference/project/mapRootWithNoSourceMapOption/amd/mapRootWithNoSourceMapOption.errors.txt index 983c75ac5d3bd..7ad42e1a26d37 100644 --- a/tests/baselines/reference/project/mapRootWithNoSourceMapOption/amd/mapRootWithNoSourceMapOption.errors.txt +++ b/tests/baselines/reference/project/mapRootWithNoSourceMapOption/amd/mapRootWithNoSourceMapOption.errors.txt @@ -1,7 +1,7 @@ -error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. +error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMap'. -!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. +!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMap'. ==== m1.ts (0 errors) ==== var m1_a1 = 10; class m1_c1 { diff --git a/tests/baselines/reference/project/mapRootWithNoSourceMapOption/node/mapRootWithNoSourceMapOption.errors.txt b/tests/baselines/reference/project/mapRootWithNoSourceMapOption/node/mapRootWithNoSourceMapOption.errors.txt index 983c75ac5d3bd..7ad42e1a26d37 100644 --- a/tests/baselines/reference/project/mapRootWithNoSourceMapOption/node/mapRootWithNoSourceMapOption.errors.txt +++ b/tests/baselines/reference/project/mapRootWithNoSourceMapOption/node/mapRootWithNoSourceMapOption.errors.txt @@ -1,7 +1,7 @@ -error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. +error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMap'. -!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMaps'. +!!! error TS5069: Option 'mapRoot' cannot be specified without specifying option 'sourceMap' or option 'declarationMap'. ==== m1.ts (0 errors) ==== var m1_a1 = 10; class m1_c1 { diff --git a/tests/baselines/reference/tsConfig/Default initialized TSConfig/tsconfig.json b/tests/baselines/reference/tsConfig/Default initialized TSConfig/tsconfig.json index e4c98cac5676b..28ccc78540c7a 100644 --- a/tests/baselines/reference/tsConfig/Default initialized TSConfig/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Default initialized TSConfig/tsconfig.json @@ -8,7 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ - // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ + // "declarationMap": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with boolean value compiler options/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with boolean value compiler options/tsconfig.json index 3d3c62fa83773..25c75a4b55699 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with boolean value compiler options/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with boolean value compiler options/tsconfig.json @@ -8,7 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ - // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ + // "declarationMap": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with enum value compiler options/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with enum value compiler options/tsconfig.json index e413105cfa00f..168fcc6dfe637 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with enum value compiler options/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with enum value compiler options/tsconfig.json @@ -8,7 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ "jsx": "react", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ - // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ + // "declarationMap": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with files options/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with files options/tsconfig.json index 9e941d4bd36c2..e87936a34fc8d 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with files options/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with files options/tsconfig.json @@ -8,7 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ - // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ + // "declarationMap": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option value/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option value/tsconfig.json index 18abba4d3200b..c00426a373b6a 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option value/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option value/tsconfig.json @@ -8,7 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ - // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ + // "declarationMap": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option/tsconfig.json index e4c98cac5676b..28ccc78540c7a 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with incorrect compiler option/tsconfig.json @@ -8,7 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ - // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ + // "declarationMap": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options with enum value/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options with enum value/tsconfig.json index dcb1a15965b7f..a4d7d4ffc2c90 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options with enum value/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options with enum value/tsconfig.json @@ -8,7 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ - // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ + // "declarationMap": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options/tsconfig.json b/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options/tsconfig.json index df2966ad60534..d01ba76d82237 100644 --- a/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options/tsconfig.json +++ b/tests/baselines/reference/tsConfig/Initialized TSConfig with list compiler options/tsconfig.json @@ -8,7 +8,7 @@ // "checkJs": true, /* Report errors in .js files. */ // "jsx": "preserve", /* Specify JSX code generation: 'preserve', 'react-native', or 'react'. */ // "declaration": true, /* Generates corresponding '.d.ts' file. */ - // "declarationMaps": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ + // "declarationMap": true, /* Generates a sourcemap for each corresponding '.d.ts' file. */ // "sourceMap": true, /* Generates corresponding '.map' file. */ // "outFile": "./", /* Concatenate and emit output to single file. */ // "outDir": "./", /* Redirect output structure to the directory. */ diff --git a/tests/cases/compiler/declarationMaps.ts b/tests/cases/compiler/declarationMaps.ts index 218f1c9394207..cb7883359432b 100644 --- a/tests/cases/compiler/declarationMaps.ts +++ b/tests/cases/compiler/declarationMaps.ts @@ -1,5 +1,5 @@ // @declaration: true -// @declarationMaps: true +// @declarationMap: true module m2 { export interface connectModule { (res, req, next): void; diff --git a/tests/cases/compiler/declarationMapsMultifile.ts b/tests/cases/compiler/declarationMapsMultifile.ts index f63ac88355f2c..43d77c4c49e1c 100644 --- a/tests/cases/compiler/declarationMapsMultifile.ts +++ b/tests/cases/compiler/declarationMapsMultifile.ts @@ -1,5 +1,5 @@ // @declaration: true -// @declarationMaps: true +// @declarationMap: true // @filename: a.ts export class Foo { doThing(x: {a: number}) { diff --git a/tests/cases/compiler/declarationMapsOutFile.ts b/tests/cases/compiler/declarationMapsOutFile.ts index 4e047388bdacc..b388d567ef335 100644 --- a/tests/cases/compiler/declarationMapsOutFile.ts +++ b/tests/cases/compiler/declarationMapsOutFile.ts @@ -1,5 +1,5 @@ // @declaration: true -// @declarationMaps: true +// @declarationMap: true // @module: amd // @outFile: bundle.js // @filename: a.ts diff --git a/tests/cases/compiler/declarationMapsOutFile2.ts b/tests/cases/compiler/declarationMapsOutFile2.ts index 6876db0e6ff88..713f6f5391044 100644 --- a/tests/cases/compiler/declarationMapsOutFile2.ts +++ b/tests/cases/compiler/declarationMapsOutFile2.ts @@ -1,5 +1,5 @@ // @declaration: true -// @declarationMaps: true +// @declarationMap: true // @outFile: bundle.js // @filename: a.ts class Foo { diff --git a/tests/cases/compiler/declarationMapsWithSourceMap.ts b/tests/cases/compiler/declarationMapsWithSourceMap.ts index 498bcb41a973e..332ccb04fce36 100644 --- a/tests/cases/compiler/declarationMapsWithSourceMap.ts +++ b/tests/cases/compiler/declarationMapsWithSourceMap.ts @@ -1,5 +1,5 @@ // @declaration: true -// @declarationMaps: true +// @declarationMap: true // @outFile: bundle.js // @sourceMap: true // @filename: a.ts diff --git a/tests/cases/compiler/declarationMapsWithoutDeclaration.ts b/tests/cases/compiler/declarationMapsWithoutDeclaration.ts index c1911eb7b7efa..266b0760bb126 100644 --- a/tests/cases/compiler/declarationMapsWithoutDeclaration.ts +++ b/tests/cases/compiler/declarationMapsWithoutDeclaration.ts @@ -1,4 +1,4 @@ -// @declarationMaps: true +// @declarationMap: true module m2 { export interface connectModule { (res, req, next): void; diff --git a/tests/cases/fourslash/declarationMapsEnableMapping_NoInline.ts b/tests/cases/fourslash/declarationMapsEnableMapping_NoInline.ts index 1422a098b5d88..a17f7afcdd884 100644 --- a/tests/cases/fourslash/declarationMapsEnableMapping_NoInline.ts +++ b/tests/cases/fourslash/declarationMapsEnableMapping_NoInline.ts @@ -3,7 +3,7 @@ // @outDir: ./dist // @inlineSourceMap: true // @declaration: true -// @declarationMaps: true +// @declarationMap: true // @Filename: index.ts // @emitThisFile: true ////export class Foo { diff --git a/tests/cases/fourslash/declarationMapsEnableMapping_NoInlineSources.ts b/tests/cases/fourslash/declarationMapsEnableMapping_NoInlineSources.ts index 10ad967e35e62..4760bbc628d2d 100644 --- a/tests/cases/fourslash/declarationMapsEnableMapping_NoInlineSources.ts +++ b/tests/cases/fourslash/declarationMapsEnableMapping_NoInlineSources.ts @@ -4,7 +4,7 @@ // @inlineSourceMap: true // @inlineSources: true // @declaration: true -// @declarationMaps: true +// @declarationMap: true // @Filename: index.ts // @emitThisFile: true ////export class Foo { diff --git a/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping.ts b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping.ts index 23185af0e6555..42e08f8a27f31 100644 --- a/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping.ts +++ b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping.ts @@ -2,7 +2,7 @@ // @BaselineFile: declarationMapsGeneratedMapsEnableMapping.baseline // @outDir: ./dist // @declaration: true -// @declarationMaps: true +// @declarationMap: true // @Filename: index.ts // @emitThisFile: true ////export class Foo { diff --git a/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping2.ts b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping2.ts index 6dfd7a997d07c..261122c1e8e7b 100644 --- a/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping2.ts +++ b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping2.ts @@ -4,7 +4,7 @@ // @sourceMap: true // @sourceRoot: /tests/cases/fourslash/ // @declaration: true -// @declarationMaps: true +// @declarationMap: true // @Filename: index.ts // @emitThisFile: true ////export class Foo { diff --git a/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping3.ts b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping3.ts index 883f806e09475..cd90793358e5c 100644 --- a/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping3.ts +++ b/tests/cases/fourslash/declarationMapsGeneratedMapsEnableMapping3.ts @@ -3,7 +3,7 @@ // @outDir: ./dist // @sourceRoot: /tests/cases/fourslash/ // @declaration: true -// @declarationMaps: true +// @declarationMap: true // @Filename: index.ts // @emitThisFile: true ////export class Foo { From d0a5e2863bf2d21b4c11ca19765a1501086c964d Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Fri, 23 Mar 2018 17:20:31 -0700 Subject: [PATCH 29/32] Even more feedback --- src/compiler/emitter.ts | 4 ++-- src/compiler/sourcemap.ts | 2 +- src/compiler/sys.ts | 15 +++++++++++++-- src/compiler/utilities.ts | 7 +++++++ src/services/services.ts | 16 ++++++++-------- .../baselines/reference/api/tsserverlibrary.d.ts | 1 + tests/baselines/reference/api/typescript.d.ts | 1 + 7 files changed, 33 insertions(+), 13 deletions(-) diff --git a/src/compiler/emitter.ts b/src/compiler/emitter.ts index c4448de506bc4..b46ca85213557 100644 --- a/src/compiler/emitter.ts +++ b/src/compiler/emitter.ts @@ -199,7 +199,7 @@ namespace ts { onEmitSourceMapOfNode: declarationSourceMap.emitNodeWithSourceMap, onEmitSourceMapOfToken: declarationSourceMap.emitTokenWithSourceMap, onEmitSourceMapOfPosition: declarationSourceMap.emitPos, - onSetSourceFile: setSourceFileDeclarationMaps, + onSetSourceFile: setSourceFileForDeclarationSourceMaps, // transform hooks onEmitNode: declarationTransform.emitNodeWithNotification, @@ -258,7 +258,7 @@ namespace ts { sourceMap.setSourceFile(node); } - function setSourceFileDeclarationMaps(node: SourceFile) { + function setSourceFileForDeclarationSourceMaps(node: SourceFile) { currentSourceFile = node; declarationSourceMap.setSourceFile(node); } diff --git a/src/compiler/sourcemap.ts b/src/compiler/sourcemap.ts index 83c95dd645926..2e81da35cfd62 100644 --- a/src/compiler/sourcemap.ts +++ b/src/compiler/sourcemap.ts @@ -475,7 +475,7 @@ namespace ts { if (compilerOptions.inlineSourceMap) { // Encode the sourceMap into the sourceMap url - const base64SourceMapText = convertToBase64(getText()); + const base64SourceMapText = base64encode(sys, getText()); return sourceMapData.jsSourceMappingURL = `data:application/json;base64,${base64SourceMapText}`; } else { diff --git a/src/compiler/sys.ts b/src/compiler/sys.ts index edf007335f681..f459515301cac 100644 --- a/src/compiler/sys.ts +++ b/src/compiler/sys.ts @@ -458,6 +458,7 @@ namespace ts { clearScreen?(): void; /*@internal*/ setBlocking?(): void; base64decode?(input: string): string; + base64encode?(input: string): string; } export interface FileWatcher { @@ -529,7 +530,10 @@ namespace ts { _crypto = undefined; } - const Buffer: typeof global.Buffer = require("buffer").Buffer; + const Buffer: { + new (input: string, encoding?: string): any; + from?(input: string, encoding?: string): any; + } = require("buffer").Buffer; const nodeVersion = getNodeMajorVersion(); const isNode4OrLater = nodeVersion >= 4; @@ -624,8 +628,15 @@ namespace ts { process.stdout._handle.setBlocking(true); } }, - base64decode: input => { + base64decode: Buffer.from ? input => { return Buffer.from(input, "base64").toString("utf8"); + } : input => { + return new Buffer(input, "base64").toString("utf8"); + }, + base64encode: Buffer.from ? input => { + return Buffer.from(input).toString("base64"); + } : input => { + return new Buffer(input).toString("base64"); } }; return nodeSystem; diff --git a/src/compiler/utilities.ts b/src/compiler/utilities.ts index 609bc0175517e..c9d8b6af5c04d 100644 --- a/src/compiler/utilities.ts +++ b/src/compiler/utilities.ts @@ -3512,6 +3512,13 @@ namespace ts { return output; } + export function base64encode(host: { base64encode?(input: string): string }, input: string): string { + if (host.base64encode) { + return host.base64encode(input); + } + return convertToBase64(input); + } + export function base64decode(host: { base64decode?(input: string): string }, input: string): string { if (host.base64decode) { return host.base64decode(input); diff --git a/src/services/services.ts b/src/services/services.ts index 434421bbb88a4..b0e1e72712777 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1581,18 +1581,18 @@ namespace ts { } - const sourceMapCommentRE = /^\/\/[@#] sourceMappingURL=(.+)$/gm; - const dataURLRE = /^data:/; - const base64URLRE = /^data:application\/json;charset=utf-8;base64,(.+)$/; + const sourceMapCommentRegExp = /^\/\/[@#] sourceMappingURL=(.+)$/gm; + const dataURLRegExp = /^data:/; + const base64UrlRegExp = /^data:application\/json;charset=utf-8;base64,(.+)$/; function scanForSourcemapURL(fileName: string) { const mappedFile = sourcemappedFileCache.get(toPath(fileName, currentDirectory, getCanonicalFileName)); if (!mappedFile) { return; } const starts = getLineStarts(mappedFile); - for (let index = starts.length - 1; index--; index >= 0) { - sourceMapCommentRE.lastIndex = starts[index]; - const comment = sourceMapCommentRE.exec(mappedFile.text); + for (let index = starts.length - 1; index >= 0; index--) { + sourceMapCommentRegExp.lastIndex = starts[index]; + const comment = sourceMapCommentRegExp.exec(mappedFile.text); if (comment) { return comment[1]; } @@ -1627,8 +1627,8 @@ namespace ts { return file.sourceMapper; } let mapFileName = scanForSourcemapURL(fileName); - if (mapFileName && dataURLRE.exec(mapFileName)) { - const b64EncodedMatch = base64URLRE.exec(mapFileName); + if (mapFileName && dataURLRegExp.exec(mapFileName)) { + const b64EncodedMatch = base64UrlRegExp.exec(mapFileName); if (b64EncodedMatch) { const base64Object = b64EncodedMatch[1]; return convertDocumentToSourceMapper(file, base64decode(sys, base64Object), fileName); diff --git a/tests/baselines/reference/api/tsserverlibrary.d.ts b/tests/baselines/reference/api/tsserverlibrary.d.ts index 6b2126d389646..c344b31b1fc13 100644 --- a/tests/baselines/reference/api/tsserverlibrary.d.ts +++ b/tests/baselines/reference/api/tsserverlibrary.d.ts @@ -2899,6 +2899,7 @@ declare namespace ts { clearTimeout?(timeoutId: any): void; clearScreen?(): void; base64decode?(input: string): string; + base64encode?(input: string): string; } interface FileWatcher { close(): void; diff --git a/tests/baselines/reference/api/typescript.d.ts b/tests/baselines/reference/api/typescript.d.ts index 9d044da4dde22..f849dc7f664e6 100644 --- a/tests/baselines/reference/api/typescript.d.ts +++ b/tests/baselines/reference/api/typescript.d.ts @@ -2899,6 +2899,7 @@ declare namespace ts { clearTimeout?(timeoutId: any): void; clearScreen?(): void; base64decode?(input: string): string; + base64encode?(input: string): string; } interface FileWatcher { close(): void; From 943dc12d6456b8236acad19456b095ef74c15a60 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 26 Mar 2018 11:25:18 -0700 Subject: [PATCH 30/32] USE Mroe lenient combined regexp --- src/services/services.ts | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/src/services/services.ts b/src/services/services.ts index b0e1e72712777..a7b8e3ead5907 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1582,8 +1582,7 @@ namespace ts { const sourceMapCommentRegExp = /^\/\/[@#] sourceMappingURL=(.+)$/gm; - const dataURLRegExp = /^data:/; - const base64UrlRegExp = /^data:application\/json;charset=utf-8;base64,(.+)$/; + const base64UrlRegExp = /^data:(?:application\/json(?:;charset=[uU][tT][fF]-8);base64,(.+)$)?/; function scanForSourcemapURL(fileName: string) { const mappedFile = sourcemappedFileCache.get(toPath(fileName, currentDirectory, getCanonicalFileName)); if (!mappedFile) { @@ -1627,12 +1626,13 @@ namespace ts { return file.sourceMapper; } let mapFileName = scanForSourcemapURL(fileName); - if (mapFileName && dataURLRegExp.exec(mapFileName)) { - const b64EncodedMatch = base64UrlRegExp.exec(mapFileName); - if (b64EncodedMatch) { - const base64Object = b64EncodedMatch[1]; + let match: RegExpExecArray; + if (mapFileName && (match = base64UrlRegExp.exec(mapFileName))) { + if (match[1]) { + const base64Object = match[1]; return convertDocumentToSourceMapper(file, base64decode(sys, base64Object), fileName); } + // Not a data URL we can parse, skip it mapFileName = undefined; } const possibleMapLocations: string[] = []; From 265cc1a876f328f8784c0b14aa5623153efc38a7 Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 26 Mar 2018 11:26:44 -0700 Subject: [PATCH 31/32] only match base64 characters --- src/services/services.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/services/services.ts b/src/services/services.ts index a7b8e3ead5907..5f0c740f8167a 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1582,7 +1582,7 @@ namespace ts { const sourceMapCommentRegExp = /^\/\/[@#] sourceMappingURL=(.+)$/gm; - const base64UrlRegExp = /^data:(?:application\/json(?:;charset=[uU][tT][fF]-8);base64,(.+)$)?/; + const base64UrlRegExp = /^data:(?:application\/json(?:;charset=[uU][tT][fF]-8);base64,([A-Za-z0-9+\/=]+)$)?/; function scanForSourcemapURL(fileName: string) { const mappedFile = sourcemappedFileCache.get(toPath(fileName, currentDirectory, getCanonicalFileName)); if (!mappedFile) { From e34a6bd89341f5bec62b79463626ef0481b2857f Mon Sep 17 00:00:00 2001 From: Wesley Wigham Date: Mon, 26 Mar 2018 11:30:03 -0700 Subject: [PATCH 32/32] Fix nit --- src/services/services.ts | 16 +++++++++------- 1 file changed, 9 insertions(+), 7 deletions(-) diff --git a/src/services/services.ts b/src/services/services.ts index 5f0c740f8167a..cd35e0375a38b 100644 --- a/src/services/services.ts +++ b/src/services/services.ts @@ -1626,14 +1626,16 @@ namespace ts { return file.sourceMapper; } let mapFileName = scanForSourcemapURL(fileName); - let match: RegExpExecArray; - if (mapFileName && (match = base64UrlRegExp.exec(mapFileName))) { - if (match[1]) { - const base64Object = match[1]; - return convertDocumentToSourceMapper(file, base64decode(sys, base64Object), fileName); + if (mapFileName) { + const match = base64UrlRegExp.exec(mapFileName); + if (match) { + if (match[1]) { + const base64Object = match[1]; + return convertDocumentToSourceMapper(file, base64decode(sys, base64Object), fileName); + } + // Not a data URL we can parse, skip it + mapFileName = undefined; } - // Not a data URL we can parse, skip it - mapFileName = undefined; } const possibleMapLocations: string[] = []; if (mapFileName) {