/** * @license * Copyright Google LLC All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ (function (factory) { if (typeof module === "object" && typeof module.exports === "object") { var v = factory(require, exports); if (v !== undefined) module.exports = v; } else if (typeof define === "function" && define.amd) { define("@angular/compiler-cli/src/ngtsc/tsc_plugin", ["require", "exports", "tslib", "@angular/compiler-cli/src/ngtsc/core", "@angular/compiler-cli/src/ngtsc/file_system", "@angular/compiler-cli/src/ngtsc/incremental", "@angular/compiler-cli/src/ngtsc/perf", "@angular/compiler-cli/src/ngtsc/program_driver", "@angular/compiler-cli/src/ngtsc/shims", "@angular/compiler-cli/src/ngtsc/typecheck/api"], factory); } })(function (require, exports) { "use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.NgTscPlugin = void 0; var tslib_1 = require("tslib"); var core_1 = require("@angular/compiler-cli/src/ngtsc/core"); var file_system_1 = require("@angular/compiler-cli/src/ngtsc/file_system"); var incremental_1 = require("@angular/compiler-cli/src/ngtsc/incremental"); var perf_1 = require("@angular/compiler-cli/src/ngtsc/perf"); var program_driver_1 = require("@angular/compiler-cli/src/ngtsc/program_driver"); var shims_1 = require("@angular/compiler-cli/src/ngtsc/shims"); var api_1 = require("@angular/compiler-cli/src/ngtsc/typecheck/api"); /** * A plugin for `tsc_wrapped` which allows Angular compilation from a plain `ts_library`. */ var NgTscPlugin = /** @class */ (function () { function NgTscPlugin(ngOptions) { this.ngOptions = ngOptions; this.name = 'ngtsc'; this.options = null; this.host = null; this._compiler = null; file_system_1.setFileSystem(new file_system_1.NodeJSFileSystem()); } Object.defineProperty(NgTscPlugin.prototype, "compiler", { get: function () { if (this._compiler === null) { throw new Error('Lifecycle error: setupCompilation() must be called first.'); } return this._compiler; }, enumerable: false, configurable: true }); NgTscPlugin.prototype.wrapHost = function (host, inputFiles, options) { // TODO(alxhub): Eventually the `wrapHost()` API will accept the old `ts.Program` (if one is // available). When it does, its `ts.SourceFile`s need to be re-tagged to enable proper // incremental compilation. this.options = tslib_1.__assign(tslib_1.__assign({}, this.ngOptions), options); this.host = core_1.NgCompilerHost.wrap(host, inputFiles, this.options, /* oldProgram */ null); return this.host; }; NgTscPlugin.prototype.setupCompilation = function (program, oldProgram) { var e_1, _a; var _b; // TODO(alxhub): we provide a `PerfRecorder` to the compiler, but because we're not driving the // compilation, the information captured within it is incomplete, and may not include timings // for phases such as emit. // // Additionally, nothing actually captures the perf results here, so recording stats at all is // somewhat moot for now :) var perfRecorder = perf_1.ActivePerfRecorder.zeroedToNow(); if (this.host === null || this.options === null) { throw new Error('Lifecycle error: setupCompilation() before wrapHost().'); } this.host.postProgramCreationCleanup(); shims_1.untagAllTsFiles(program); var programDriver = new program_driver_1.TsCreateProgramDriver(program, this.host, this.options, this.host.shimExtensionPrefixes); var strategy = new incremental_1.PatchedProgramIncrementalBuildStrategy(); var oldState = oldProgram !== undefined ? strategy.getIncrementalState(oldProgram) : null; var ticket; var modifiedResourceFiles = new Set(); if (this.host.getModifiedResourceFiles !== undefined) { try { for (var _c = tslib_1.__values((_b = this.host.getModifiedResourceFiles()) !== null && _b !== void 0 ? _b : []), _d = _c.next(); !_d.done; _d = _c.next()) { var resourceFile = _d.value; modifiedResourceFiles.add(file_system_1.resolve(resourceFile)); } } catch (e_1_1) { e_1 = { error: e_1_1 }; } finally { try { if (_d && !_d.done && (_a = _c.return)) _a.call(_c); } finally { if (e_1) throw e_1.error; } } } if (oldProgram === undefined || oldState === null) { ticket = core_1.freshCompilationTicket(program, this.options, strategy, programDriver, perfRecorder, /* enableTemplateTypeChecker */ false, /* usePoisonedData */ false); } else { strategy.toNextBuildStrategy().getIncrementalState(oldProgram); ticket = core_1.incrementalFromStateTicket(oldProgram, oldState, program, this.options, strategy, programDriver, modifiedResourceFiles, perfRecorder, false, false); } this._compiler = core_1.NgCompiler.fromTicket(ticket, this.host); return { ignoreForDiagnostics: this._compiler.ignoreForDiagnostics, ignoreForEmit: this._compiler.ignoreForEmit, }; }; NgTscPlugin.prototype.getDiagnostics = function (file) { if (file === undefined) { return this.compiler.getDiagnostics(); } return this.compiler.getDiagnosticsForFile(file, api_1.OptimizeFor.WholeProgram); }; NgTscPlugin.prototype.getOptionDiagnostics = function () { return this.compiler.getOptionDiagnostics(); }; NgTscPlugin.prototype.getNextProgram = function () { return this.compiler.getCurrentProgram(); }; NgTscPlugin.prototype.createTransformers = function () { // The plugin consumer doesn't know about our perf tracing system, so we consider the emit phase // as beginning now. this.compiler.perfRecorder.phase(perf_1.PerfPhase.TypeScriptEmit); return this.compiler.prepareEmit().transformers; }; return NgTscPlugin; }()); exports.NgTscPlugin = NgTscPlugin; }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tsc_plugin.js","sourceRoot":"","sources":["../../../../../../../packages/compiler-cli/src/ngtsc/tsc_plugin.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport * as ts from 'typescript';\n\nimport {CompilationTicket, freshCompilationTicket, incrementalFromStateTicket, NgCompiler, NgCompilerHost} from './core';\nimport {NgCompilerOptions, UnifiedModulesHost} from './core/api';\nimport {AbsoluteFsPath, NodeJSFileSystem, resolve, setFileSystem} from './file_system';\nimport {PatchedProgramIncrementalBuildStrategy} from './incremental';\nimport {ActivePerfRecorder, PerfPhase} from './perf';\nimport {TsCreateProgramDriver} from './program_driver';\nimport {untagAllTsFiles} from './shims';\nimport {OptimizeFor} from './typecheck/api';\n\n// The following is needed to fix a the chicken-and-egg issue where the sync (into g3) script will\n// refuse to accept this file unless the following string appears:\n// import * as plugin from '@bazel/typescript/internal/tsc_wrapped/plugin_api';\n\n/**\n * A `ts.CompilerHost` which also returns a list of input files, out of which the `ts.Program`\n * should be created.\n *\n * Currently mirrored from @bazel/typescript/internal/tsc_wrapped/plugin_api (with the naming of\n * `fileNameToModuleName` corrected).\n */\ninterface PluginCompilerHost extends ts.CompilerHost, Partial<UnifiedModulesHost> {\n  readonly inputFiles: ReadonlyArray<string>;\n}\n\n/**\n * Mirrors the plugin interface from tsc_wrapped which is currently under active development. To\n * enable progress to be made in parallel, the upstream interface isn't implemented directly.\n * Instead, `TscPlugin` here is structurally assignable to what tsc_wrapped expects.\n */\ninterface TscPlugin {\n  readonly name: string;\n\n  wrapHost(\n      host: ts.CompilerHost&Partial<UnifiedModulesHost>, inputFiles: ReadonlyArray<string>,\n      options: ts.CompilerOptions): PluginCompilerHost;\n\n  setupCompilation(program: ts.Program, oldProgram?: ts.Program): {\n    ignoreForDiagnostics: Set<ts.SourceFile>,\n    ignoreForEmit: Set<ts.SourceFile>,\n  };\n\n  getDiagnostics(file?: ts.SourceFile): ts.Diagnostic[];\n\n  getOptionDiagnostics(): ts.Diagnostic[];\n\n  getNextProgram(): ts.Program;\n\n  createTransformers(): ts.CustomTransformers;\n}\n\n/**\n * A plugin for `tsc_wrapped` which allows Angular compilation from a plain `ts_library`.\n */\nexport class NgTscPlugin implements TscPlugin {\n  name = 'ngtsc';\n\n  private options: NgCompilerOptions|null = null;\n  private host: NgCompilerHost|null = null;\n  private _compiler: NgCompiler|null = null;\n\n  get compiler(): NgCompiler {\n    if (this._compiler === null) {\n      throw new Error('Lifecycle error: setupCompilation() must be called first.');\n    }\n    return this._compiler;\n  }\n\n  constructor(private ngOptions: {}) {\n    setFileSystem(new NodeJSFileSystem());\n  }\n\n  wrapHost(\n      host: ts.CompilerHost&Partial<UnifiedModulesHost>, inputFiles: readonly string[],\n      options: ts.CompilerOptions): PluginCompilerHost {\n    // TODO(alxhub): Eventually the `wrapHost()` API will accept the old `ts.Program` (if one is\n    // available). When it does, its `ts.SourceFile`s need to be re-tagged to enable proper\n    // incremental compilation.\n    this.options = {...this.ngOptions, ...options} as NgCompilerOptions;\n    this.host = NgCompilerHost.wrap(host, inputFiles, this.options, /* oldProgram */ null);\n    return this.host;\n  }\n\n  setupCompilation(program: ts.Program, oldProgram?: ts.Program): {\n    ignoreForDiagnostics: Set<ts.SourceFile>,\n    ignoreForEmit: Set<ts.SourceFile>,\n  } {\n    // TODO(alxhub): we provide a `PerfRecorder` to the compiler, but because we're not driving the\n    // compilation, the information captured within it is incomplete, and may not include timings\n    // for phases such as emit.\n    //\n    // Additionally, nothing actually captures the perf results here, so recording stats at all is\n    // somewhat moot for now :)\n    const perfRecorder = ActivePerfRecorder.zeroedToNow();\n    if (this.host === null || this.options === null) {\n      throw new Error('Lifecycle error: setupCompilation() before wrapHost().');\n    }\n    this.host.postProgramCreationCleanup();\n    untagAllTsFiles(program);\n    const programDriver = new TsCreateProgramDriver(\n        program, this.host, this.options, this.host.shimExtensionPrefixes);\n    const strategy = new PatchedProgramIncrementalBuildStrategy();\n    const oldState = oldProgram !== undefined ? strategy.getIncrementalState(oldProgram) : null;\n    let ticket: CompilationTicket;\n\n    const modifiedResourceFiles = new Set<AbsoluteFsPath>();\n    if (this.host.getModifiedResourceFiles !== undefined) {\n      for (const resourceFile of this.host.getModifiedResourceFiles() ?? []) {\n        modifiedResourceFiles.add(resolve(resourceFile));\n      }\n    }\n\n    if (oldProgram === undefined || oldState === null) {\n      ticket = freshCompilationTicket(\n          program, this.options, strategy, programDriver, perfRecorder,\n          /* enableTemplateTypeChecker */ false, /* usePoisonedData */ false);\n    } else {\n      strategy.toNextBuildStrategy().getIncrementalState(oldProgram);\n      ticket = incrementalFromStateTicket(\n          oldProgram, oldState, program, this.options, strategy, programDriver,\n          modifiedResourceFiles, perfRecorder, false, false);\n    }\n    this._compiler = NgCompiler.fromTicket(ticket, this.host);\n    return {\n      ignoreForDiagnostics: this._compiler.ignoreForDiagnostics,\n      ignoreForEmit: this._compiler.ignoreForEmit,\n    };\n  }\n\n  getDiagnostics(file?: ts.SourceFile): ts.Diagnostic[] {\n    if (file === undefined) {\n      return this.compiler.getDiagnostics();\n    }\n    return this.compiler.getDiagnosticsForFile(file, OptimizeFor.WholeProgram);\n  }\n\n  getOptionDiagnostics(): ts.Diagnostic[] {\n    return this.compiler.getOptionDiagnostics();\n  }\n\n  getNextProgram(): ts.Program {\n    return this.compiler.getCurrentProgram();\n  }\n\n  createTransformers(): ts.CustomTransformers {\n    // The plugin consumer doesn't know about our perf tracing system, so we consider the emit phase\n    // as beginning now.\n    this.compiler.perfRecorder.phase(PerfPhase.TypeScriptEmit);\n    return this.compiler.prepareEmit().transformers;\n  }\n}\n"]}