1 | /**
|
---|
2 | * @license
|
---|
3 | * Copyright Google LLC All Rights Reserved.
|
---|
4 | *
|
---|
5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
6 | * found in the LICENSE file at https://angular.io/license
|
---|
7 | */
|
---|
8 | (function (factory) {
|
---|
9 | if (typeof module === "object" && typeof module.exports === "object") {
|
---|
10 | var v = factory(require, exports);
|
---|
11 | if (v !== undefined) module.exports = v;
|
---|
12 | }
|
---|
13 | else if (typeof define === "function" && define.amd) {
|
---|
14 | define("@angular/compiler-cli/src/transformers/patch_alias_reference_resolution", ["require", "exports", "tslib", "typescript"], factory);
|
---|
15 | }
|
---|
16 | })(function (require, exports) {
|
---|
17 | "use strict";
|
---|
18 | Object.defineProperty(exports, "__esModule", { value: true });
|
---|
19 | exports.isAliasImportDeclaration = exports.loadIsReferencedAliasDeclarationPatch = void 0;
|
---|
20 | var tslib_1 = require("tslib");
|
---|
21 | var ts = require("typescript");
|
---|
22 | var patchedReferencedAliasesSymbol = Symbol('patchedReferencedAliases');
|
---|
23 | /**
|
---|
24 | * Patches the alias declaration reference resolution for a given transformation context
|
---|
25 | * so that TypeScript knows about the specified alias declarations being referenced.
|
---|
26 | *
|
---|
27 | * This exists because TypeScript performs analysis of import usage before transformers
|
---|
28 | * run and doesn't refresh its state after transformations. This means that imports
|
---|
29 | * for symbols used as constructor types are elided due to their original type-only usage.
|
---|
30 | *
|
---|
31 | * In reality though, since we downlevel decorators and constructor parameters, we want
|
---|
32 | * these symbols to be retained in the JavaScript output as they will be used as values
|
---|
33 | * at runtime. We can instruct TypeScript to preserve imports for such identifiers by
|
---|
34 | * creating a mutable clone of a given import specifier/clause or namespace, but that
|
---|
35 | * has the downside of preserving the full import in the JS output. See:
|
---|
36 | * https://github.com/microsoft/TypeScript/blob/3eaa7c65f6f076a08a5f7f1946fd0df7c7430259/src/compiler/transformers/ts.ts#L242-L250.
|
---|
37 | *
|
---|
38 | * This is a trick the CLI used in the past for constructor parameter downleveling in JIT:
|
---|
39 | * https://github.com/angular/angular-cli/blob/b3f84cc5184337666ce61c07b7b9df418030106f/packages/ngtools/webpack/src/transformers/ctor-parameters.ts#L323-L325
|
---|
40 | * The trick is not ideal though as it preserves the full import (as outlined before), and it
|
---|
41 | * results in a slow-down due to the type checker being involved multiple times. The CLI worked
|
---|
42 | * around this import preserving issue by having another complex post-process step that detects and
|
---|
43 | * elides unused imports. Note that these unused imports could cause unused chunks being generated
|
---|
44 | * by Webpack if the application or library is not marked as side-effect free.
|
---|
45 | *
|
---|
46 | * This is not ideal though, as we basically re-implement the complex import usage resolution
|
---|
47 | * from TypeScript. We can do better by letting TypeScript do the import eliding, but providing
|
---|
48 | * information about the alias declarations (e.g. import specifiers) that should not be elided
|
---|
49 | * because they are actually referenced (as they will now appear in static properties).
|
---|
50 | *
|
---|
51 | * More information about these limitations with transformers can be found in:
|
---|
52 | * 1. https://github.com/Microsoft/TypeScript/issues/17552.
|
---|
53 | * 2. https://github.com/microsoft/TypeScript/issues/17516.
|
---|
54 | * 3. https://github.com/angular/tsickle/issues/635.
|
---|
55 | *
|
---|
56 | * The patch we apply to tell TypeScript about actual referenced aliases (i.e. imported symbols),
|
---|
57 | * matches conceptually with the logic that runs internally in TypeScript when the
|
---|
58 | * `emitDecoratorMetadata` flag is enabled. TypeScript basically surfaces the same problem and
|
---|
59 | * solves it conceptually the same way, but obviously doesn't need to access an `@internal` API.
|
---|
60 | *
|
---|
61 | * The set that is returned by this function is meant to be filled with import declaration nodes
|
---|
62 | * that have been referenced in a value-position by the transform, such the installed patch can
|
---|
63 | * ensure that those import declarations are not elided.
|
---|
64 | *
|
---|
65 | * See below. Note that this uses sourcegraph as the TypeScript checker file doesn't display on
|
---|
66 | * Github.
|
---|
67 | * https://sourcegraph.com/github.com/microsoft/TypeScript@3eaa7c65f6f076a08a5f7f1946fd0df7c7430259/-/blob/src/compiler/checker.ts#L31219-31257
|
---|
68 | */
|
---|
69 | function loadIsReferencedAliasDeclarationPatch(context) {
|
---|
70 | // If the `getEmitResolver` method is not available, TS most likely changed the
|
---|
71 | // internal structure of the transformation context. We will abort gracefully.
|
---|
72 | if (!isTransformationContextWithEmitResolver(context)) {
|
---|
73 | throwIncompatibleTransformationContextError();
|
---|
74 | }
|
---|
75 | var emitResolver = context.getEmitResolver();
|
---|
76 | // The emit resolver may have been patched already, in which case we return the set of referenced
|
---|
77 | // aliases that was created when the patch was first applied.
|
---|
78 | // See https://github.com/angular/angular/issues/40276.
|
---|
79 | var existingReferencedAliases = emitResolver[patchedReferencedAliasesSymbol];
|
---|
80 | if (existingReferencedAliases !== undefined) {
|
---|
81 | return existingReferencedAliases;
|
---|
82 | }
|
---|
83 | var originalIsReferencedAliasDeclaration = emitResolver.isReferencedAliasDeclaration;
|
---|
84 | // If the emit resolver does not have a function called `isReferencedAliasDeclaration`, then
|
---|
85 | // we abort gracefully as most likely TS changed the internal structure of the emit resolver.
|
---|
86 | if (originalIsReferencedAliasDeclaration === undefined) {
|
---|
87 | throwIncompatibleTransformationContextError();
|
---|
88 | }
|
---|
89 | var referencedAliases = new Set();
|
---|
90 | emitResolver.isReferencedAliasDeclaration = function (node) {
|
---|
91 | var args = [];
|
---|
92 | for (var _i = 1; _i < arguments.length; _i++) {
|
---|
93 | args[_i - 1] = arguments[_i];
|
---|
94 | }
|
---|
95 | if (isAliasImportDeclaration(node) && referencedAliases.has(node)) {
|
---|
96 | return true;
|
---|
97 | }
|
---|
98 | return originalIsReferencedAliasDeclaration.call.apply(originalIsReferencedAliasDeclaration, tslib_1.__spreadArray([emitResolver, node], tslib_1.__read(args)));
|
---|
99 | };
|
---|
100 | return emitResolver[patchedReferencedAliasesSymbol] = referencedAliases;
|
---|
101 | }
|
---|
102 | exports.loadIsReferencedAliasDeclarationPatch = loadIsReferencedAliasDeclarationPatch;
|
---|
103 | /**
|
---|
104 | * Gets whether a given node corresponds to an import alias declaration. Alias
|
---|
105 | * declarations can be import specifiers, namespace imports or import clauses
|
---|
106 | * as these do not declare an actual symbol but just point to a target declaration.
|
---|
107 | */
|
---|
108 | function isAliasImportDeclaration(node) {
|
---|
109 | return ts.isImportSpecifier(node) || ts.isNamespaceImport(node) || ts.isImportClause(node);
|
---|
110 | }
|
---|
111 | exports.isAliasImportDeclaration = isAliasImportDeclaration;
|
---|
112 | /** Whether the transformation context exposes its emit resolver. */
|
---|
113 | function isTransformationContextWithEmitResolver(context) {
|
---|
114 | return context.getEmitResolver !== undefined;
|
---|
115 | }
|
---|
116 | /**
|
---|
117 | * Throws an error about an incompatible TypeScript version for which the alias
|
---|
118 | * declaration reference resolution could not be monkey-patched. The error will
|
---|
119 | * also propose potential solutions that can be applied by developers.
|
---|
120 | */
|
---|
121 | function throwIncompatibleTransformationContextError() {
|
---|
122 | throw Error('Unable to downlevel Angular decorators due to an incompatible TypeScript ' +
|
---|
123 | 'version.\nIf you recently updated TypeScript and this issue surfaces now, consider ' +
|
---|
124 | 'downgrading.\n\n' +
|
---|
125 | 'Please report an issue on the Angular repositories when this issue ' +
|
---|
126 | 'surfaces and you are using a supposedly compatible TypeScript version.');
|
---|
127 | }
|
---|
128 | });
|
---|
129 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"patch_alias_reference_resolution.js","sourceRoot":"","sources":["../../../../../../../packages/compiler-cli/src/transformers/patch_alias_reference_resolution.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;;;;;;;;;;;;;;IAEH,+BAAiC;IAWjC,IAAM,8BAA8B,GAAG,MAAM,CAAC,0BAA0B,CAAC,CAAC;IAQ1E;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OA6CG;IACH,SAAgB,qCAAqC,CAAC,OAAiC;QAErF,+EAA+E;QAC/E,8EAA8E;QAC9E,IAAI,CAAC,uCAAuC,CAAC,OAAO,CAAC,EAAE;YACrD,2CAA2C,EAAE,CAAC;SAC/C;QACD,IAAM,YAAY,GAAG,OAAO,CAAC,eAAe,EAAE,CAAC;QAE/C,iGAAiG;QACjG,6DAA6D;QAC7D,uDAAuD;QACvD,IAAM,yBAAyB,GAAG,YAAY,CAAC,8BAA8B,CAAC,CAAC;QAC/E,IAAI,yBAAyB,KAAK,SAAS,EAAE;YAC3C,OAAO,yBAAyB,CAAC;SAClC;QAED,IAAM,oCAAoC,GAAG,YAAY,CAAC,4BAA4B,CAAC;QACvF,4FAA4F;QAC5F,6FAA6F;QAC7F,IAAI,oCAAoC,KAAK,SAAS,EAAE;YACtD,2CAA2C,EAAE,CAAC;SAC/C;QAED,IAAM,iBAAiB,GAAG,IAAI,GAAG,EAAkB,CAAC;QACpD,YAAY,CAAC,4BAA4B,GAAG,UAAS,IAAI;YAAE,cAAO;iBAAP,UAAO,EAAP,qBAAO,EAAP,IAAO;gBAAP,6BAAO;;YAChE,IAAI,wBAAwB,CAAC,IAAI,CAAC,IAAI,iBAAiB,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE;gBACjE,OAAO,IAAI,CAAC;aACb;YACD,OAAO,oCAAoC,CAAC,IAAI,OAAzC,oCAAoC,yBAAM,YAAY,EAAE,IAAI,kBAAK,IAAI,IAAE;QAChF,CAAC,CAAC;QACF,OAAO,YAAY,CAAC,8BAA8B,CAAC,GAAG,iBAAiB,CAAC;IAC1E,CAAC;IAhCD,sFAgCC;IAED;;;;OAIG;IACH,SAAgB,wBAAwB,CAAC,IAAa;QAEpD,OAAO,EAAE,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,EAAE,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;IAC7F,CAAC;IAHD,4DAGC;IAED,oEAAoE;IACpE,SAAS,uCAAuC,CAAC,OAAiC;QAEhF,OAAQ,OAAsD,CAAC,eAAe,KAAK,SAAS,CAAC;IAC/F,CAAC;IAGD;;;;OAIG;IACH,SAAS,2CAA2C;QAClD,MAAM,KAAK,CACP,2EAA2E;YAC3E,qFAAqF;YACrF,kBAAkB;YAClB,qEAAqE;YACrE,wEAAwE,CAAC,CAAC;IAChF,CAAC","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\n/**\n * Describes a TypeScript transformation context with the internal emit\n * resolver exposed. There are requests upstream in TypeScript to expose\n * that as public API: https://github.com/microsoft/TypeScript/issues/17516..\n */\ninterface TransformationContextWithResolver extends ts.TransformationContext {\n  getEmitResolver: () => EmitResolver;\n}\n\nconst patchedReferencedAliasesSymbol = Symbol('patchedReferencedAliases');\n\n/** Describes a subset of the TypeScript internal emit resolver. */\ninterface EmitResolver {\n  isReferencedAliasDeclaration?(node: ts.Node, ...args: unknown[]): void;\n  [patchedReferencedAliasesSymbol]?: Set<ts.Declaration>;\n}\n\n/**\n * Patches the alias declaration reference resolution for a given transformation context\n * so that TypeScript knows about the specified alias declarations being referenced.\n *\n * This exists because TypeScript performs analysis of import usage before transformers\n * run and doesn't refresh its state after transformations. This means that imports\n * for symbols used as constructor types are elided due to their original type-only usage.\n *\n * In reality though, since we downlevel decorators and constructor parameters, we want\n * these symbols to be retained in the JavaScript output as they will be used as values\n * at runtime. We can instruct TypeScript to preserve imports for such identifiers by\n * creating a mutable clone of a given import specifier/clause or namespace, but that\n * has the downside of preserving the full import in the JS output. See:\n * https://github.com/microsoft/TypeScript/blob/3eaa7c65f6f076a08a5f7f1946fd0df7c7430259/src/compiler/transformers/ts.ts#L242-L250.\n *\n * This is a trick the CLI used in the past  for constructor parameter downleveling in JIT:\n * https://github.com/angular/angular-cli/blob/b3f84cc5184337666ce61c07b7b9df418030106f/packages/ngtools/webpack/src/transformers/ctor-parameters.ts#L323-L325\n * The trick is not ideal though as it preserves the full import (as outlined before), and it\n * results in a slow-down due to the type checker being involved multiple times. The CLI worked\n * around this import preserving issue by having another complex post-process step that detects and\n * elides unused imports. Note that these unused imports could cause unused chunks being generated\n * by Webpack if the application or library is not marked as side-effect free.\n *\n * This is not ideal though, as we basically re-implement the complex import usage resolution\n * from TypeScript. We can do better by letting TypeScript do the import eliding, but providing\n * information about the alias declarations (e.g. import specifiers) that should not be elided\n * because they are actually referenced (as they will now appear in static properties).\n *\n * More information about these limitations with transformers can be found in:\n *   1. https://github.com/Microsoft/TypeScript/issues/17552.\n *   2. https://github.com/microsoft/TypeScript/issues/17516.\n *   3. https://github.com/angular/tsickle/issues/635.\n *\n * The patch we apply to tell TypeScript about actual referenced aliases (i.e. imported symbols),\n * matches conceptually with the logic that runs internally in TypeScript when the\n * `emitDecoratorMetadata` flag is enabled. TypeScript basically surfaces the same problem and\n * solves it conceptually the same way, but obviously doesn't need to access an `@internal` API.\n *\n * The set that is returned by this function is meant to be filled with import declaration nodes\n * that have been referenced in a value-position by the transform, such the installed patch can\n * ensure that those import declarations are not elided.\n *\n * See below. Note that this uses sourcegraph as the TypeScript checker file doesn't display on\n * Github.\n * https://sourcegraph.com/github.com/microsoft/TypeScript@3eaa7c65f6f076a08a5f7f1946fd0df7c7430259/-/blob/src/compiler/checker.ts#L31219-31257\n */\nexport function loadIsReferencedAliasDeclarationPatch(context: ts.TransformationContext):\n    Set<ts.Declaration> {\n  // If the `getEmitResolver` method is not available, TS most likely changed the\n  // internal structure of the transformation context. We will abort gracefully.\n  if (!isTransformationContextWithEmitResolver(context)) {\n    throwIncompatibleTransformationContextError();\n  }\n  const emitResolver = context.getEmitResolver();\n\n  // The emit resolver may have been patched already, in which case we return the set of referenced\n  // aliases that was created when the patch was first applied.\n  // See https://github.com/angular/angular/issues/40276.\n  const existingReferencedAliases = emitResolver[patchedReferencedAliasesSymbol];\n  if (existingReferencedAliases !== undefined) {\n    return existingReferencedAliases;\n  }\n\n  const originalIsReferencedAliasDeclaration = emitResolver.isReferencedAliasDeclaration;\n  // If the emit resolver does not have a function called `isReferencedAliasDeclaration`, then\n  // we abort gracefully as most likely TS changed the internal structure of the emit resolver.\n  if (originalIsReferencedAliasDeclaration === undefined) {\n    throwIncompatibleTransformationContextError();\n  }\n\n  const referencedAliases = new Set<ts.Declaration>();\n  emitResolver.isReferencedAliasDeclaration = function(node, ...args) {\n    if (isAliasImportDeclaration(node) && referencedAliases.has(node)) {\n      return true;\n    }\n    return originalIsReferencedAliasDeclaration.call(emitResolver, node, ...args);\n  };\n  return emitResolver[patchedReferencedAliasesSymbol] = referencedAliases;\n}\n\n/**\n * Gets whether a given node corresponds to an import alias declaration. Alias\n * declarations can be import specifiers, namespace imports or import clauses\n * as these do not declare an actual symbol but just point to a target declaration.\n */\nexport function isAliasImportDeclaration(node: ts.Node): node is ts.ImportSpecifier|\n    ts.NamespaceImport|ts.ImportClause {\n  return ts.isImportSpecifier(node) || ts.isNamespaceImport(node) || ts.isImportClause(node);\n}\n\n/** Whether the transformation context exposes its emit resolver. */\nfunction isTransformationContextWithEmitResolver(context: ts.TransformationContext):\n    context is TransformationContextWithResolver {\n  return (context as Partial<TransformationContextWithResolver>).getEmitResolver !== undefined;\n}\n\n\n/**\n * Throws an error about an incompatible TypeScript version for which the alias\n * declaration reference resolution could not be monkey-patched. The error will\n * also propose potential solutions that can be applied by developers.\n */\nfunction throwIncompatibleTransformationContextError(): never {\n  throw Error(\n      'Unable to downlevel Angular decorators due to an incompatible TypeScript ' +\n      'version.\\nIf you recently updated TypeScript and this issue surfaces now, consider ' +\n      'downgrading.\\n\\n' +\n      'Please report an issue on the Angular repositories when this issue ' +\n      'surfaces and you are using a supposedly compatible TypeScript version.');\n}\n"]} |
---|