source: trip-planner-front/node_modules/@angular/compiler-cli/ngcc/src/analysis/module_with_providers_analyzer.js@ 6a3a178

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

initial commit

  • Property mode set to 100644
File size: 31.1 KB
Line 
1(function (factory) {
2 if (typeof module === "object" && typeof module.exports === "object") {
3 var v = factory(require, exports);
4 if (v !== undefined) module.exports = v;
5 }
6 else if (typeof define === "function" && define.amd) {
7 define("@angular/compiler-cli/ngcc/src/analysis/module_with_providers_analyzer", ["require", "exports", "typescript", "@angular/compiler-cli/src/ngtsc/imports", "@angular/compiler-cli/src/ngtsc/partial_evaluator", "@angular/compiler-cli/src/ngtsc/reflection", "@angular/compiler-cli/ngcc/src/utils"], factory);
8 }
9})(function (require, exports) {
10 "use strict";
11 Object.defineProperty(exports, "__esModule", { value: true });
12 exports.ModuleWithProvidersAnalyzer = exports.ModuleWithProvidersAnalyses = void 0;
13 /**
14 * @license
15 * Copyright Google LLC All Rights Reserved.
16 *
17 * Use of this source code is governed by an MIT-style license that can be
18 * found in the LICENSE file at https://angular.io/license
19 */
20 var ts = require("typescript");
21 var imports_1 = require("@angular/compiler-cli/src/ngtsc/imports");
22 var partial_evaluator_1 = require("@angular/compiler-cli/src/ngtsc/partial_evaluator");
23 var reflection_1 = require("@angular/compiler-cli/src/ngtsc/reflection");
24 var utils_1 = require("@angular/compiler-cli/ngcc/src/utils");
25 exports.ModuleWithProvidersAnalyses = Map;
26 var ModuleWithProvidersAnalyzer = /** @class */ (function () {
27 function ModuleWithProvidersAnalyzer(host, typeChecker, referencesRegistry, processDts) {
28 this.host = host;
29 this.typeChecker = typeChecker;
30 this.referencesRegistry = referencesRegistry;
31 this.processDts = processDts;
32 this.evaluator = new partial_evaluator_1.PartialEvaluator(this.host, this.typeChecker, null);
33 }
34 ModuleWithProvidersAnalyzer.prototype.analyzeProgram = function (program) {
35 var _this = this;
36 var analyses = new exports.ModuleWithProvidersAnalyses();
37 var rootFiles = this.getRootFiles(program);
38 rootFiles.forEach(function (f) {
39 var fns = _this.getModuleWithProvidersFunctions(f);
40 fns && fns.forEach(function (fn) {
41 if (fn.ngModule.bestGuessOwningModule === null) {
42 // Record the usage of an internal module as it needs to become an exported symbol
43 _this.referencesRegistry.add(fn.ngModule.node, new imports_1.Reference(fn.ngModule.node));
44 }
45 // Only when processing the dts files do we need to determine which declaration to update.
46 if (_this.processDts) {
47 var dtsFn = _this.getDtsModuleWithProvidersFunction(fn);
48 var dtsFnType = dtsFn.declaration.type;
49 var typeParam = dtsFnType && ts.isTypeReferenceNode(dtsFnType) &&
50 dtsFnType.typeArguments && dtsFnType.typeArguments[0] ||
51 null;
52 if (!typeParam || isAnyKeyword(typeParam)) {
53 var dtsFile = dtsFn.declaration.getSourceFile();
54 var analysis = analyses.has(dtsFile) ? analyses.get(dtsFile) : [];
55 analysis.push(dtsFn);
56 analyses.set(dtsFile, analysis);
57 }
58 }
59 });
60 });
61 return analyses;
62 };
63 ModuleWithProvidersAnalyzer.prototype.getRootFiles = function (program) {
64 return program.getRootFileNames().map(function (f) { return program.getSourceFile(f); }).filter(utils_1.isDefined);
65 };
66 ModuleWithProvidersAnalyzer.prototype.getModuleWithProvidersFunctions = function (f) {
67 var _this = this;
68 var exports = this.host.getExportsOfModule(f);
69 if (!exports)
70 return [];
71 var infos = [];
72 exports.forEach(function (declaration) {
73 if (declaration.node === null) {
74 return;
75 }
76 if (_this.host.isClass(declaration.node)) {
77 _this.host.getMembersOfClass(declaration.node).forEach(function (member) {
78 if (member.isStatic) {
79 var info = _this.parseForModuleWithProviders(member.name, member.node, member.implementation, declaration.node);
80 if (info) {
81 infos.push(info);
82 }
83 }
84 });
85 }
86 else {
87 if (utils_1.hasNameIdentifier(declaration.node)) {
88 var info = _this.parseForModuleWithProviders(declaration.node.name.text, declaration.node);
89 if (info) {
90 infos.push(info);
91 }
92 }
93 }
94 });
95 return infos;
96 };
97 /**
98 * Parse a function/method node (or its implementation), to see if it returns a
99 * `ModuleWithProviders` object.
100 * @param name The name of the function.
101 * @param node the node to check - this could be a function, a method or a variable declaration.
102 * @param implementation the actual function expression if `node` is a variable declaration.
103 * @param container the class that contains the function, if it is a method.
104 * @returns info about the function if it does return a `ModuleWithProviders` object; `null`
105 * otherwise.
106 */
107 ModuleWithProvidersAnalyzer.prototype.parseForModuleWithProviders = function (name, node, implementation, container) {
108 if (implementation === void 0) { implementation = node; }
109 if (container === void 0) { container = null; }
110 if (implementation === null ||
111 (!ts.isFunctionDeclaration(implementation) && !ts.isMethodDeclaration(implementation) &&
112 !ts.isFunctionExpression(implementation))) {
113 return null;
114 }
115 var declaration = implementation;
116 var definition = this.host.getDefinitionOfFunction(declaration);
117 if (definition === null) {
118 return null;
119 }
120 var body = definition.body;
121 if (body === null || body.length === 0) {
122 return null;
123 }
124 // Get hold of the return statement expression for the function
125 var lastStatement = body[body.length - 1];
126 if (!ts.isReturnStatement(lastStatement) || lastStatement.expression === undefined) {
127 return null;
128 }
129 // Evaluate this expression and extract the `ngModule` reference
130 var result = this.evaluator.evaluate(lastStatement.expression);
131 if (!(result instanceof Map) || !result.has('ngModule')) {
132 return null;
133 }
134 var ngModuleRef = result.get('ngModule');
135 if (!(ngModuleRef instanceof imports_1.Reference)) {
136 return null;
137 }
138 if (!reflection_1.isNamedClassDeclaration(ngModuleRef.node) &&
139 !reflection_1.isNamedVariableDeclaration(ngModuleRef.node)) {
140 throw new Error("The identity given by " + ngModuleRef.debugName + " referenced in \"" + declaration.getText() + "\" doesn't appear to be a \"class\" declaration.");
141 }
142 var ngModule = ngModuleRef;
143 return { name: name, ngModule: ngModule, declaration: declaration, container: container };
144 };
145 ModuleWithProvidersAnalyzer.prototype.getDtsModuleWithProvidersFunction = function (fn) {
146 var dtsFn = null;
147 var containerClass = fn.container && this.host.getClassSymbol(fn.container);
148 if (containerClass) {
149 var dtsClass = this.host.getDtsDeclaration(containerClass.declaration.valueDeclaration);
150 // Get the declaration of the matching static method
151 dtsFn = dtsClass && ts.isClassDeclaration(dtsClass) ?
152 dtsClass.members.find(function (member) { return ts.isMethodDeclaration(member) && ts.isIdentifier(member.name) &&
153 member.name.text === fn.name; }) :
154 null;
155 }
156 else {
157 dtsFn = this.host.getDtsDeclaration(fn.declaration);
158 }
159 if (!dtsFn) {
160 throw new Error("Matching type declaration for " + fn.declaration.getText() + " is missing");
161 }
162 if (!isFunctionOrMethod(dtsFn)) {
163 throw new Error("Matching type declaration for " + fn.declaration.getText() + " is not a function: " + dtsFn.getText());
164 }
165 var container = containerClass ? containerClass.declaration.valueDeclaration : null;
166 var ngModule = this.resolveNgModuleReference(fn);
167 return { name: fn.name, container: container, declaration: dtsFn, ngModule: ngModule };
168 };
169 ModuleWithProvidersAnalyzer.prototype.resolveNgModuleReference = function (fn) {
170 var ngModule = fn.ngModule;
171 // For external module references, use the declaration as is.
172 if (ngModule.bestGuessOwningModule !== null) {
173 return ngModule;
174 }
175 // For internal (non-library) module references, redirect the module's value declaration
176 // to its type declaration.
177 var dtsNgModule = this.host.getDtsDeclaration(ngModule.node);
178 if (!dtsNgModule) {
179 throw new Error("No typings declaration can be found for the referenced NgModule class in " + fn.declaration.getText() + ".");
180 }
181 if (!reflection_1.isNamedClassDeclaration(dtsNgModule)) {
182 throw new Error("The referenced NgModule in " + fn.declaration
183 .getText() + " is not a named class declaration in the typings program; instead we get " + dtsNgModule.getText());
184 }
185 return new imports_1.Reference(dtsNgModule, null);
186 };
187 return ModuleWithProvidersAnalyzer;
188 }());
189 exports.ModuleWithProvidersAnalyzer = ModuleWithProvidersAnalyzer;
190 function isFunctionOrMethod(declaration) {
191 return ts.isFunctionDeclaration(declaration) || ts.isMethodDeclaration(declaration);
192 }
193 function isAnyKeyword(typeParam) {
194 return typeParam.kind === ts.SyntaxKind.AnyKeyword;
195 }
196});
197//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"module_with_providers_analyzer.js","sourceRoot":"","sources":["../../../../../../../../packages/compiler-cli/ngcc/src/analysis/module_with_providers_analyzer.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 */\nimport * as ts from 'typescript';\n\nimport {ReferencesRegistry} from '../../../src/ngtsc/annotations';\nimport {Reference} from '../../../src/ngtsc/imports';\nimport {PartialEvaluator} from '../../../src/ngtsc/partial_evaluator';\nimport {ClassDeclaration, DeclarationNode, isNamedClassDeclaration, isNamedVariableDeclaration} from '../../../src/ngtsc/reflection';\nimport {NgccReflectionHost} from '../host/ngcc_host';\nimport {hasNameIdentifier, isDefined} from '../utils';\n\n/**\n * A structure returned from `getModuleWithProvidersFunctions()` that describes functions\n * that return ModuleWithProviders objects.\n */\nexport interface ModuleWithProvidersInfo {\n  /**\n   * The name of the declared function.\n   */\n  name: string;\n  /**\n   * The declaration of the function that returns the `ModuleWithProviders` object.\n   */\n  declaration: ts.SignatureDeclaration;\n  /**\n   * Declaration of the containing class (if this is a method)\n   */\n  container: DeclarationNode|null;\n  /**\n   * The declaration of the class that the `ngModule` property on the `ModuleWithProviders` object\n   * refers to.\n   */\n  ngModule: Reference<ClassDeclaration>;\n}\n\nexport type ModuleWithProvidersAnalyses = Map<ts.SourceFile, ModuleWithProvidersInfo[]>;\nexport const ModuleWithProvidersAnalyses = Map;\n\nexport class ModuleWithProvidersAnalyzer {\n  private evaluator = new PartialEvaluator(this.host, this.typeChecker, null);\n\n  constructor(\n      private host: NgccReflectionHost, private typeChecker: ts.TypeChecker,\n      private referencesRegistry: ReferencesRegistry, private processDts: boolean) {}\n\n  analyzeProgram(program: ts.Program): ModuleWithProvidersAnalyses {\n    const analyses: ModuleWithProvidersAnalyses = new ModuleWithProvidersAnalyses();\n    const rootFiles = this.getRootFiles(program);\n    rootFiles.forEach(f => {\n      const fns = this.getModuleWithProvidersFunctions(f);\n      fns && fns.forEach(fn => {\n        if (fn.ngModule.bestGuessOwningModule === null) {\n          // Record the usage of an internal module as it needs to become an exported symbol\n          this.referencesRegistry.add(fn.ngModule.node, new Reference(fn.ngModule.node));\n        }\n\n        // Only when processing the dts files do we need to determine which declaration to update.\n        if (this.processDts) {\n          const dtsFn = this.getDtsModuleWithProvidersFunction(fn);\n          const dtsFnType = dtsFn.declaration.type;\n          const typeParam = dtsFnType && ts.isTypeReferenceNode(dtsFnType) &&\n                  dtsFnType.typeArguments && dtsFnType.typeArguments[0] ||\n              null;\n          if (!typeParam || isAnyKeyword(typeParam)) {\n            const dtsFile = dtsFn.declaration.getSourceFile();\n            const analysis = analyses.has(dtsFile) ? analyses.get(dtsFile)! : [];\n            analysis.push(dtsFn);\n            analyses.set(dtsFile, analysis);\n          }\n        }\n      });\n    });\n    return analyses;\n  }\n\n  private getRootFiles(program: ts.Program): ts.SourceFile[] {\n    return program.getRootFileNames().map(f => program.getSourceFile(f)).filter(isDefined);\n  }\n\n  private getModuleWithProvidersFunctions(f: ts.SourceFile): ModuleWithProvidersInfo[] {\n    const exports = this.host.getExportsOfModule(f);\n    if (!exports) return [];\n    const infos: ModuleWithProvidersInfo[] = [];\n    exports.forEach((declaration) => {\n      if (declaration.node === null) {\n        return;\n      }\n      if (this.host.isClass(declaration.node)) {\n        this.host.getMembersOfClass(declaration.node).forEach(member => {\n          if (member.isStatic) {\n            const info = this.parseForModuleWithProviders(\n                member.name, member.node, member.implementation, declaration.node);\n            if (info) {\n              infos.push(info);\n            }\n          }\n        });\n      } else {\n        if (hasNameIdentifier(declaration.node)) {\n          const info =\n              this.parseForModuleWithProviders(declaration.node.name.text, declaration.node);\n          if (info) {\n            infos.push(info);\n          }\n        }\n      }\n    });\n    return infos;\n  }\n\n  /**\n   * Parse a function/method node (or its implementation), to see if it returns a\n   * `ModuleWithProviders` object.\n   * @param name The name of the function.\n   * @param node the node to check - this could be a function, a method or a variable declaration.\n   * @param implementation the actual function expression if `node` is a variable declaration.\n   * @param container the class that contains the function, if it is a method.\n   * @returns info about the function if it does return a `ModuleWithProviders` object; `null`\n   * otherwise.\n   */\n  private parseForModuleWithProviders(\n      name: string, node: ts.Node|null, implementation: ts.Node|null = node,\n      container: DeclarationNode|null = null): ModuleWithProvidersInfo|null {\n    if (implementation === null ||\n        (!ts.isFunctionDeclaration(implementation) && !ts.isMethodDeclaration(implementation) &&\n         !ts.isFunctionExpression(implementation))) {\n      return null;\n    }\n    const declaration = implementation;\n    const definition = this.host.getDefinitionOfFunction(declaration);\n    if (definition === null) {\n      return null;\n    }\n\n    const body = definition.body;\n    if (body === null || body.length === 0) {\n      return null;\n    }\n\n    // Get hold of the return statement expression for the function\n    const lastStatement = body[body.length - 1];\n    if (!ts.isReturnStatement(lastStatement) || lastStatement.expression === undefined) {\n      return null;\n    }\n\n    // Evaluate this expression and extract the `ngModule` reference\n    const result = this.evaluator.evaluate(lastStatement.expression);\n    if (!(result instanceof Map) || !result.has('ngModule')) {\n      return null;\n    }\n\n    const ngModuleRef = result.get('ngModule')!;\n    if (!(ngModuleRef instanceof Reference)) {\n      return null;\n    }\n\n    if (!isNamedClassDeclaration(ngModuleRef.node) &&\n        !isNamedVariableDeclaration(ngModuleRef.node)) {\n      throw new Error(`The identity given by ${ngModuleRef.debugName} referenced in \"${\n          declaration!.getText()}\" doesn't appear to be a \"class\" declaration.`);\n    }\n\n    const ngModule = ngModuleRef as Reference<ClassDeclaration>;\n    return {name, ngModule, declaration, container};\n  }\n\n  private getDtsModuleWithProvidersFunction(fn: ModuleWithProvidersInfo): ModuleWithProvidersInfo {\n    let dtsFn: ts.Declaration|null = null;\n    const containerClass = fn.container && this.host.getClassSymbol(fn.container);\n    if (containerClass) {\n      const dtsClass = this.host.getDtsDeclaration(containerClass.declaration.valueDeclaration);\n      // Get the declaration of the matching static method\n      dtsFn = dtsClass && ts.isClassDeclaration(dtsClass) ?\n          dtsClass.members.find(\n              member => ts.isMethodDeclaration(member) && ts.isIdentifier(member.name) &&\n                  member.name.text === fn.name) as ts.Declaration :\n          null;\n    } else {\n      dtsFn = this.host.getDtsDeclaration(fn.declaration);\n    }\n    if (!dtsFn) {\n      throw new Error(`Matching type declaration for ${fn.declaration.getText()} is missing`);\n    }\n    if (!isFunctionOrMethod(dtsFn)) {\n      throw new Error(`Matching type declaration for ${\n          fn.declaration.getText()} is not a function: ${dtsFn.getText()}`);\n    }\n    const container = containerClass ? containerClass.declaration.valueDeclaration : null;\n    const ngModule = this.resolveNgModuleReference(fn);\n    return {name: fn.name, container, declaration: dtsFn, ngModule};\n  }\n\n  private resolveNgModuleReference(fn: ModuleWithProvidersInfo): Reference<ClassDeclaration> {\n    const ngModule = fn.ngModule;\n\n    // For external module references, use the declaration as is.\n    if (ngModule.bestGuessOwningModule !== null) {\n      return ngModule;\n    }\n\n    // For internal (non-library) module references, redirect the module's value declaration\n    // to its type declaration.\n    const dtsNgModule = this.host.getDtsDeclaration(ngModule.node);\n    if (!dtsNgModule) {\n      throw new Error(`No typings declaration can be found for the referenced NgModule class in ${\n          fn.declaration.getText()}.`);\n    }\n    if (!isNamedClassDeclaration(dtsNgModule)) {\n      throw new Error(`The referenced NgModule in ${\n          fn.declaration\n              .getText()} is not a named class declaration in the typings program; instead we get ${\n          dtsNgModule.getText()}`);\n    }\n    return new Reference(dtsNgModule, null);\n  }\n}\n\n\nfunction isFunctionOrMethod(declaration: ts.Declaration): declaration is ts.FunctionDeclaration|\n    ts.MethodDeclaration {\n  return ts.isFunctionDeclaration(declaration) || ts.isMethodDeclaration(declaration);\n}\n\nfunction isAnyKeyword(typeParam: ts.TypeNode): typeParam is ts.KeywordTypeNode {\n  return typeParam.kind === ts.SyntaxKind.AnyKeyword;\n}\n"]}
Note: See TracBrowser for help on using the repository browser.