source: trip-planner-front/node_modules/@angular/compiler-cli/src/transformers/patch_alias_reference_resolution.js@ 59329aa

Last change on this file since 59329aa was 6a3a178, checked in by Ema <ema_spirova@…>, 3 years ago

initial commit

  • Property mode set to 100644
File size: 19.5 KB
Line 
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"]}
Note: See TracBrowser for help on using the repository browser.