| (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/core/schematics/migrations/undecorated-classes-with-di/decorator_rewrite/decorator_rewriter", ["require", "exports", "typescript", "@angular/core/schematics/utils/typescript/functions", "@angular/core/schematics/migrations/undecorated-classes-with-di/decorator_rewrite/import_rewrite_visitor"], factory); |
| } |
| })(function (require, exports) { |
| "use strict"; |
| Object.defineProperty(exports, "__esModule", { value: true }); |
| exports.DecoratorRewriter = void 0; |
| const ts = require("typescript"); |
| const functions_1 = require("@angular/core/schematics/utils/typescript/functions"); |
| const import_rewrite_visitor_1 = require("@angular/core/schematics/migrations/undecorated-classes-with-di/decorator_rewrite/import_rewrite_visitor"); |
| /** |
| * Class that can be used to copy decorators to a new location. The rewriter ensures that |
| * identifiers and imports are rewritten to work in the new file location. Fields in a |
| * decorator that cannot be cleanly copied will be copied with a comment explaining that |
| * imports and identifiers need to be adjusted manually. |
| */ |
| class DecoratorRewriter { |
| constructor(importManager, typeChecker, evaluator, compiler) { |
| this.importManager = importManager; |
| this.typeChecker = typeChecker; |
| this.evaluator = evaluator; |
| this.compiler = compiler; |
| this.previousSourceFile = null; |
| this.newSourceFile = null; |
| this.newProperties = []; |
| this.nonCopyableProperties = []; |
| this.importRewriterFactory = new import_rewrite_visitor_1.ImportRewriteTransformerFactory(this.importManager, this.typeChecker, this.compiler['_host']); |
| } |
| rewrite(ngDecorator, newSourceFile) { |
| const decorator = ngDecorator.node; |
| // Reset the previous state of the decorator rewriter. |
| this.newProperties = []; |
| this.nonCopyableProperties = []; |
| this.newSourceFile = newSourceFile; |
| this.previousSourceFile = decorator.getSourceFile(); |
| // If the decorator will be added to the same source file it currently |
| // exists in, we don't need to rewrite any paths or add new imports. |
| if (this.previousSourceFile === newSourceFile) { |
| return this._createDecorator(decorator.expression); |
| } |
| const oldCallExpr = decorator.expression; |
| if (!oldCallExpr.arguments.length) { |
| // Re-use the original decorator if there are no arguments and nothing needs |
| // to be sanitized or rewritten. |
| return this._createDecorator(decorator.expression); |
| } |
| const metadata = functions_1.unwrapExpression(oldCallExpr.arguments[0]); |
| if (!ts.isObjectLiteralExpression(metadata)) { |
| // Re-use the original decorator as there is no metadata that can be sanitized. |
| return this._createDecorator(decorator.expression); |
| } |
| metadata.properties.forEach(prop => { |
| // We don't handle spread assignments, accessors or method declarations automatically |
| // as it involves more advanced static analysis and these type of properties are not |
| // picked up by ngc either. |
| if (ts.isSpreadAssignment(prop) || ts.isAccessor(prop) || ts.isMethodDeclaration(prop)) { |
| this.nonCopyableProperties.push(prop); |
| return; |
| } |
| const sanitizedProp = this._sanitizeMetadataProperty(prop); |
| if (sanitizedProp !== null) { |
| this.newProperties.push(sanitizedProp); |
| } |
| else { |
| this.nonCopyableProperties.push(prop); |
| } |
| }); |
| // In case there is at least one non-copyable property, we add a leading comment to |
| // the first property assignment in order to ask the developer to manually manage |
| // imports and do path rewriting for these properties. |
| if (this.nonCopyableProperties.length !== 0) { |
| ['The following fields were copied from the base class,', |
| 'but could not be updated automatically to work in the', |
| 'new file location. Please add any required imports for', 'the properties below:'] |
| .forEach(text => ts.addSyntheticLeadingComment(this.nonCopyableProperties[0], ts.SyntaxKind.SingleLineCommentTrivia, ` ${text}`, true)); |
| } |
| // Note that we don't update the decorator as we don't want to copy potential leading |
| // comments of the decorator. This is necessary because otherwise comments from the |
| // copied decorator end up describing the new class (which is not always correct). |
| return this._createDecorator(ts.createCall(this.importManager.addImportToSourceFile(newSourceFile, ngDecorator.name, ngDecorator.moduleName), undefined, [ts.updateObjectLiteral(metadata, [...this.newProperties, ...this.nonCopyableProperties])])); |
| } |
| /** Creates a new decorator with the given expression. */ |
| _createDecorator(expr) { |
| // Note that we don't update the decorator as we don't want to copy potential leading |
| // comments of the decorator. This is necessary because otherwise comments from the |
| // copied decorator end up describing the new class (which is not always correct). |
| return ts.createDecorator(expr); |
| } |
| /** |
| * Sanitizes a metadata property by ensuring that all contained identifiers |
| * are imported in the target source file. |
| */ |
| _sanitizeMetadataProperty(prop) { |
| try { |
| return ts |
| .transform(prop, [ctx => this.importRewriterFactory.create(ctx, this.newSourceFile)]) |
| .transformed[0]; |
| } |
| catch (e) { |
| // If the error is for an unresolved identifier, we want to return "null" because |
| // such object literal elements could be added to the non-copyable properties. |
| if (e instanceof import_rewrite_visitor_1.UnresolvedIdentifierError) { |
| return null; |
| } |
| throw e; |
| } |
| } |
| } |
| exports.DecoratorRewriter = DecoratorRewriter; |
| }); |
| //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"decorator_rewriter.js","sourceRoot":"","sources":["../../../../../../../../../packages/core/schematics/migrations/undecorated-classes-with-di/decorator_rewrite/decorator_rewriter.ts"],"names":[],"mappings":"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","sourcesContent":["\n/**\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 {AotCompiler} from '@angular/compiler';\nimport {PartialEvaluator} from '@angular/compiler-cli/src/ngtsc/partial_evaluator';\nimport * as ts from 'typescript';\n\nimport {ImportManager} from '../../../utils/import_manager';\nimport {NgDecorator} from '../../../utils/ng_decorators';\nimport {unwrapExpression} from '../../../utils/typescript/functions';\n\nimport {ImportRewriteTransformerFactory, UnresolvedIdentifierError} from './import_rewrite_visitor';\n\n\n/**\n * Class that can be used to copy decorators to a new location. The rewriter ensures that\n * identifiers and imports are rewritten to work in the new file location. Fields in a\n * decorator that cannot be cleanly copied will be copied with a comment explaining that\n * imports and identifiers need to be adjusted manually.\n */\nexport class DecoratorRewriter {\n  previousSourceFile: ts.SourceFile|null = null;\n  newSourceFile: ts.SourceFile|null = null;\n\n  newProperties: ts.ObjectLiteralElementLike[] = [];\n  nonCopyableProperties: ts.ObjectLiteralElementLike[] = [];\n\n  private importRewriterFactory = new ImportRewriteTransformerFactory(\n      this.importManager, this.typeChecker, this.compiler['_host']);\n\n  constructor(\n      private importManager: ImportManager, private typeChecker: ts.TypeChecker,\n      private evaluator: PartialEvaluator, private compiler: AotCompiler) {}\n\n  rewrite(ngDecorator: NgDecorator, newSourceFile: ts.SourceFile): ts.Decorator {\n    const decorator = ngDecorator.node;\n\n    // Reset the previous state of the decorator rewriter.\n    this.newProperties = [];\n    this.nonCopyableProperties = [];\n    this.newSourceFile = newSourceFile;\n    this.previousSourceFile = decorator.getSourceFile();\n\n    // If the decorator will be added to the same source file it currently\n    // exists in, we don't need to rewrite any paths or add new imports.\n    if (this.previousSourceFile === newSourceFile) {\n      return this._createDecorator(decorator.expression);\n    }\n\n    const oldCallExpr = decorator.expression;\n\n    if (!oldCallExpr.arguments.length) {\n      // Re-use the original decorator if there are no arguments and nothing needs\n      // to be sanitized or rewritten.\n      return this._createDecorator(decorator.expression);\n    }\n\n    const metadata = unwrapExpression(oldCallExpr.arguments[0]);\n    if (!ts.isObjectLiteralExpression(metadata)) {\n      // Re-use the original decorator as there is no metadata that can be sanitized.\n      return this._createDecorator(decorator.expression);\n    }\n\n    metadata.properties.forEach(prop => {\n      // We don't handle spread assignments, accessors or method declarations automatically\n      // as it involves more advanced static analysis and these type of properties are not\n      // picked up by ngc either.\n      if (ts.isSpreadAssignment(prop) || ts.isAccessor(prop) || ts.isMethodDeclaration(prop)) {\n        this.nonCopyableProperties.push(prop);\n        return;\n      }\n\n      const sanitizedProp = this._sanitizeMetadataProperty(prop);\n      if (sanitizedProp !== null) {\n        this.newProperties.push(sanitizedProp);\n      } else {\n        this.nonCopyableProperties.push(prop);\n      }\n    });\n\n    // In case there is at least one non-copyable property, we add a leading comment to\n    // the first property assignment in order to ask the developer to manually manage\n    // imports and do path rewriting for these properties.\n    if (this.nonCopyableProperties.length !== 0) {\n      ['The following fields were copied from the base class,',\n       'but could not be updated automatically to work in the',\n       'new file location. Please add any required imports for', 'the properties below:']\n          .forEach(\n              text => ts.addSyntheticLeadingComment(\n                  this.nonCopyableProperties[0], ts.SyntaxKind.SingleLineCommentTrivia, ` ${text}`,\n                  true));\n    }\n\n    // Note that we don't update the decorator as we don't want to copy potential leading\n    // comments of the decorator. This is necessary because otherwise comments from the\n    // copied decorator end up describing the new class (which is not always correct).\n    return this._createDecorator(ts.createCall(\n        this.importManager.addImportToSourceFile(\n            newSourceFile, ngDecorator.name, ngDecorator.moduleName),\n        undefined, [ts.updateObjectLiteral(\n                       metadata, [...this.newProperties, ...this.nonCopyableProperties])]));\n  }\n\n  /** Creates a new decorator with the given expression. */\n  private _createDecorator(expr: ts.Expression): ts.Decorator {\n    // Note that we don't update the decorator as we don't want to copy potential leading\n    // comments of the decorator. This is necessary because otherwise comments from the\n    // copied decorator end up describing the new class (which is not always correct).\n    return ts.createDecorator(expr);\n  }\n\n  /**\n   * Sanitizes a metadata property by ensuring that all contained identifiers\n   * are imported in the target source file.\n   */\n  private _sanitizeMetadataProperty(prop: ts.ObjectLiteralElementLike): ts.ObjectLiteralElementLike\n      |null {\n    try {\n      return ts\n          .transform(prop, [ctx => this.importRewriterFactory.create(ctx, this.newSourceFile!)])\n          .transformed[0];\n    } catch (e) {\n      // If the error is for an unresolved identifier, we want to return \"null\" because\n      // such object literal elements could be added to the non-copyable properties.\n      if (e instanceof UnresolvedIdentifierError) {\n        return null;\n      }\n      throw e;\n    }\n  }\n}\n"]} |