@angular/core 12.1.0-next.6 → 12.1.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (164) hide show
  1. package/bundles/core-testing.umd.js +239 -46
  2. package/bundles/core-testing.umd.js.map +1 -1
  3. package/bundles/core.umd.js +126 -49
  4. package/bundles/core.umd.js.map +1 -1
  5. package/core.d.ts +45 -19
  6. package/core.metadata.json +1 -1
  7. package/esm2015/src/application_init.js +5 -5
  8. package/esm2015/src/application_ref.js +34 -9
  9. package/esm2015/src/change_detection/differs/iterable_differs.js +1 -1
  10. package/esm2015/src/compiler/compiler_facade.js +31 -8
  11. package/esm2015/src/compiler/compiler_facade_interface.js +1 -1
  12. package/esm2015/src/di/jit/injectable.js +4 -3
  13. package/esm2015/src/di/r3_injector.js +5 -2
  14. package/esm2015/src/error_handler.js +2 -6
  15. package/esm2015/src/event_emitter.js +1 -1
  16. package/esm2015/src/linker/component_factory_resolver.js +1 -1
  17. package/esm2015/src/linker/template_ref.js +1 -1
  18. package/esm2015/src/linker/view_container_ref.js +1 -1
  19. package/esm2015/src/metadata/di.js +1 -1
  20. package/esm2015/src/metadata/do_boostrap.js +1 -1
  21. package/esm2015/src/render3/component_ref.js +1 -1
  22. package/esm2015/src/render3/jit/directive.js +6 -5
  23. package/esm2015/src/render3/jit/module.js +7 -4
  24. package/esm2015/src/render3/jit/partial.js +27 -9
  25. package/esm2015/src/render3/jit/pipe.js +5 -3
  26. package/esm2015/src/render3/ng_module_ref.js +1 -1
  27. package/esm2015/src/render3/view_ref.js +1 -1
  28. package/esm2015/src/sanitization/bypass.js +1 -1
  29. package/esm2015/src/version.js +1 -1
  30. package/esm2015/src/view/entrypoint.js +1 -1
  31. package/esm2015/src/view/refs.js +1 -1
  32. package/esm2015/testing/src/ng_zone_mock.js +1 -1
  33. package/esm2015/testing/src/r3_test_bed.js +84 -7
  34. package/esm2015/testing/src/r3_test_bed_compiler.js +4 -1
  35. package/esm2015/testing/src/resolvers.js +1 -1
  36. package/esm2015/testing/src/test_bed.js +110 -21
  37. package/esm2015/testing/src/test_bed_common.js +7 -1
  38. package/esm2015/testing/src/test_hooks.js +45 -0
  39. package/esm2015/testing/src/testing.js +3 -3
  40. package/fesm2015/core.js +116 -45
  41. package/fesm2015/core.js.map +1 -1
  42. package/fesm2015/testing.js +216 -29
  43. package/fesm2015/testing.js.map +1 -1
  44. package/package.json +1 -1
  45. package/schematics/migrations/missing-injectable/providers_evaluator.js +1 -1
  46. package/schematics/migrations/static-queries/strategies/usage_strategy/template_usage_visitor.js +1 -1
  47. package/schematics/migrations/template-var-assignment/angular/html_variable_assignment_visitor.js +1 -1
  48. package/schematics/migrations.json +1 -1
  49. package/src/r3_symbols.d.ts +2 -3
  50. package/testing/testing.d.ts +64 -5
  51. package/testing/testing.metadata.json +1 -1
  52. package/testing.d.ts +1 -1
  53. package/esm2015/testing/src/before_each.js +0 -33
  54. package/schematics/migrations/abstract-control-parent/abstract-control-parent.externs.js +0 -0
  55. package/schematics/migrations/abstract-control-parent/index.mjs +0 -47
  56. package/schematics/migrations/abstract-control-parent/util.mjs +0 -50
  57. package/schematics/migrations/activated-route-snapshot-fragment/activated-route-snapshot-fragment.externs.js +0 -0
  58. package/schematics/migrations/activated-route-snapshot-fragment/index.mjs +0 -48
  59. package/schematics/migrations/activated-route-snapshot-fragment/util.mjs +0 -33
  60. package/schematics/migrations/can-activate-with-redirect-to/can-activate-with-redirect-to.externs.js +0 -0
  61. package/schematics/migrations/can-activate-with-redirect-to/index.mjs +0 -44
  62. package/schematics/migrations/can-activate-with-redirect-to/util.mjs +0 -55
  63. package/schematics/migrations/deep-shadow-piercing-selector/deep-shadow-piercing-selector.externs.js +0 -0
  64. package/schematics/migrations/deep-shadow-piercing-selector/index.mjs +0 -36
  65. package/schematics/migrations/dynamic-queries/dynamic-queries.externs.js +0 -0
  66. package/schematics/migrations/dynamic-queries/index.mjs +0 -51
  67. package/schematics/migrations/dynamic-queries/util.mjs +0 -65
  68. package/schematics/migrations/initial-navigation/collector.mjs +0 -105
  69. package/schematics/migrations/initial-navigation/index.mjs +0 -61
  70. package/schematics/migrations/initial-navigation/initial-navigation.externs.js +0 -0
  71. package/schematics/migrations/initial-navigation/transform.mjs +0 -54
  72. package/schematics/migrations/initial-navigation/update_recorder.mjs +0 -9
  73. package/schematics/migrations/initial-navigation/util.mjs +0 -28
  74. package/schematics/migrations/missing-injectable/definition_collector.mjs +0 -75
  75. package/schematics/migrations/missing-injectable/index.mjs +0 -100
  76. package/schematics/migrations/missing-injectable/missing-injectable.externs.js +0 -0
  77. package/schematics/migrations/missing-injectable/providers_evaluator.mjs +0 -50
  78. package/schematics/migrations/missing-injectable/transform.mjs +0 -187
  79. package/schematics/migrations/missing-injectable/update_recorder.mjs +0 -9
  80. package/schematics/migrations/module-with-providers/collector.mjs +0 -59
  81. package/schematics/migrations/module-with-providers/index.mjs +0 -71
  82. package/schematics/migrations/module-with-providers/module-with-providers.externs.js +0 -0
  83. package/schematics/migrations/module-with-providers/transform.mjs +0 -134
  84. package/schematics/migrations/module-with-providers/util.mjs +0 -25
  85. package/schematics/migrations/move-document/document_import_visitor.mjs +0 -60
  86. package/schematics/migrations/move-document/index.mjs +0 -71
  87. package/schematics/migrations/move-document/move-document.externs.js +0 -0
  88. package/schematics/migrations/move-document/move-import.mjs +0 -45
  89. package/schematics/migrations/native-view-encapsulation/index.mjs +0 -41
  90. package/schematics/migrations/native-view-encapsulation/native-view-encapsulation.externs.js +0 -0
  91. package/schematics/migrations/native-view-encapsulation/util.mjs +0 -34
  92. package/schematics/migrations/navigation-extras-omissions/index.mjs +0 -47
  93. package/schematics/migrations/navigation-extras-omissions/navigation-extras-omissions.externs.js +0 -0
  94. package/schematics/migrations/navigation-extras-omissions/util.mjs +0 -111
  95. package/schematics/migrations/relative-link-resolution/collector.mjs +0 -81
  96. package/schematics/migrations/relative-link-resolution/index.mjs +0 -62
  97. package/schematics/migrations/relative-link-resolution/relative-link-resolution.externs.js +0 -0
  98. package/schematics/migrations/relative-link-resolution/transform.mjs +0 -51
  99. package/schematics/migrations/relative-link-resolution/update_recorder.mjs +0 -9
  100. package/schematics/migrations/relative-link-resolution/util.mjs +0 -28
  101. package/schematics/migrations/renderer-to-renderer2/helpers.mjs +0 -224
  102. package/schematics/migrations/renderer-to-renderer2/index.mjs +0 -113
  103. package/schematics/migrations/renderer-to-renderer2/migration.mjs +0 -211
  104. package/schematics/migrations/renderer-to-renderer2/renderer-to-renderer2.externs.js +0 -0
  105. package/schematics/migrations/renderer-to-renderer2/util.mjs +0 -69
  106. package/schematics/migrations/router-preserve-query-params/index.mjs +0 -50
  107. package/schematics/migrations/router-preserve-query-params/router-preserve-query-params.externs.js +0 -0
  108. package/schematics/migrations/router-preserve-query-params/util.mjs +0 -88
  109. package/schematics/migrations/static-queries/angular/directive_inputs.mjs +0 -70
  110. package/schematics/migrations/static-queries/angular/ng_query_visitor.mjs +0 -113
  111. package/schematics/migrations/static-queries/angular/query-definition.mjs +0 -20
  112. package/schematics/migrations/static-queries/angular/super_class.mjs +0 -21
  113. package/schematics/migrations/static-queries/index.mjs +0 -206
  114. package/schematics/migrations/static-queries/static-queries.externs.js +0 -0
  115. package/schematics/migrations/static-queries/strategies/template_strategy/template_strategy.mjs +0 -200
  116. package/schematics/migrations/static-queries/strategies/test_strategy/test_strategy.mjs +0 -24
  117. package/schematics/migrations/static-queries/strategies/timing-strategy.mjs +0 -9
  118. package/schematics/migrations/static-queries/strategies/usage_strategy/declaration_usage_visitor.mjs +0 -355
  119. package/schematics/migrations/static-queries/strategies/usage_strategy/super_class_context.mjs +0 -38
  120. package/schematics/migrations/static-queries/strategies/usage_strategy/template_usage_visitor.mjs +0 -80
  121. package/schematics/migrations/static-queries/strategies/usage_strategy/usage_strategy.mjs +0 -154
  122. package/schematics/migrations/static-queries/transform.mjs +0 -88
  123. package/schematics/migrations/template-var-assignment/analyze_template.mjs +0 -25
  124. package/schematics/migrations/template-var-assignment/angular/html_variable_assignment_visitor.mjs +0 -65
  125. package/schematics/migrations/template-var-assignment/index.mjs +0 -68
  126. package/schematics/migrations/template-var-assignment/template-var-assignment.externs.js +0 -0
  127. package/schematics/migrations/undecorated-classes-with-decorated-fields/index.mjs +0 -92
  128. package/schematics/migrations/undecorated-classes-with-decorated-fields/transform.mjs +0 -286
  129. package/schematics/migrations/undecorated-classes-with-decorated-fields/undecorated-classes-with-decorated-fields.externs.js +0 -0
  130. package/schematics/migrations/undecorated-classes-with-decorated-fields/update_recorder.mjs +0 -9
  131. package/schematics/migrations/undecorated-classes-with-di/create_ngc_program.mjs +0 -43
  132. package/schematics/migrations/undecorated-classes-with-di/decorator_rewrite/convert_directive_metadata.mjs +0 -82
  133. package/schematics/migrations/undecorated-classes-with-di/decorator_rewrite/decorator_rewriter.mjs +0 -102
  134. package/schematics/migrations/undecorated-classes-with-di/decorator_rewrite/import_rewrite_visitor.mjs +0 -111
  135. package/schematics/migrations/undecorated-classes-with-di/decorator_rewrite/path_format.mjs +0 -17
  136. package/schematics/migrations/undecorated-classes-with-di/decorator_rewrite/source_file_exports.mjs +0 -51
  137. package/schematics/migrations/undecorated-classes-with-di/index.mjs +0 -162
  138. package/schematics/migrations/undecorated-classes-with-di/ng_declaration_collector.mjs +0 -124
  139. package/schematics/migrations/undecorated-classes-with-di/transform.mjs +0 -356
  140. package/schematics/migrations/undecorated-classes-with-di/undecorated-classes-with-di.externs.js +0 -0
  141. package/schematics/migrations/undecorated-classes-with-di/update_recorder.mjs +0 -9
  142. package/schematics/migrations/wait-for-async/index.mjs +0 -76
  143. package/schematics/migrations/wait-for-async/util.mjs +0 -23
  144. package/schematics/migrations/wait-for-async/wait-for-async.externs.js +0 -0
  145. package/schematics/migrations/xhr-factory/index.mjs +0 -91
  146. package/schematics/migrations/xhr-factory/xhr-factory.externs.js +0 -0
  147. package/schematics/utils/import_manager.mjs +0 -197
  148. package/schematics/utils/line_mappings.mjs +0 -60
  149. package/schematics/utils/ng_component_template.mjs +0 -96
  150. package/schematics/utils/ng_decorators.mjs +0 -23
  151. package/schematics/utils/parse_html.mjs +0 -24
  152. package/schematics/utils/project_tsconfig_paths.mjs +0 -74
  153. package/schematics/utils/schematics_prompt.mjs +0 -30
  154. package/schematics/utils/typescript/class_declaration.mjs +0 -33
  155. package/schematics/utils/typescript/compiler_host.mjs +0 -66
  156. package/schematics/utils/typescript/decorators.mjs +0 -20
  157. package/schematics/utils/typescript/find_base_classes.mjs +0 -30
  158. package/schematics/utils/typescript/functions.mjs +0 -28
  159. package/schematics/utils/typescript/imports.mjs +0 -91
  160. package/schematics/utils/typescript/nodes.mjs +0 -57
  161. package/schematics/utils/typescript/parse_tsconfig.mjs +0 -26
  162. package/schematics/utils/typescript/property_name.mjs +0 -23
  163. package/schematics/utils/typescript/symbol.mjs +0 -72
  164. package/schematics/utils/utils.externs.js +0 -0
@@ -1,113 +0,0 @@
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
- import * as ts from 'typescript';
9
- import { getAngularDecorators } from '../../../utils/ng_decorators';
10
- import { findParentClassDeclaration, getBaseTypeIdentifiers } from '../../../utils/typescript/class_declaration';
11
- import { getPropertyNameText } from '../../../utils/typescript/property_name';
12
- import { getInputNamesOfClass } from './directive_inputs';
13
- import { QueryType } from './query-definition';
14
- /**
15
- * Visitor that can be used to determine Angular queries within given TypeScript nodes.
16
- * Besides resolving queries, the visitor also records class relations and searches for
17
- * Angular input setters which can be used to analyze the timing usage of a given query.
18
- */
19
- export class NgQueryResolveVisitor {
20
- constructor(typeChecker) {
21
- this.typeChecker = typeChecker;
22
- /** Resolved Angular query definitions. */
23
- this.resolvedQueries = new Map();
24
- /** Maps a class declaration to its class metadata. */
25
- this.classMetadata = new Map();
26
- }
27
- visitNode(node) {
28
- switch (node.kind) {
29
- case ts.SyntaxKind.PropertyDeclaration:
30
- this.visitPropertyDeclaration(node);
31
- break;
32
- case ts.SyntaxKind.ClassDeclaration:
33
- this.visitClassDeclaration(node);
34
- break;
35
- case ts.SyntaxKind.GetAccessor:
36
- case ts.SyntaxKind.SetAccessor:
37
- this.visitAccessorDeclaration(node);
38
- break;
39
- }
40
- ts.forEachChild(node, n => this.visitNode(n));
41
- }
42
- visitPropertyDeclaration(node) {
43
- this._recordQueryDeclaration(node, node, getPropertyNameText(node.name));
44
- }
45
- visitAccessorDeclaration(node) {
46
- this._recordQueryDeclaration(node, null, getPropertyNameText(node.name));
47
- }
48
- visitClassDeclaration(node) {
49
- this._recordClassInputSetters(node);
50
- this._recordClassInheritances(node);
51
- }
52
- _recordQueryDeclaration(node, property, queryName) {
53
- if (!node.decorators || !node.decorators.length) {
54
- return;
55
- }
56
- const ngDecorators = getAngularDecorators(this.typeChecker, node.decorators);
57
- const queryDecorator = ngDecorators.find(({ name }) => name === 'ViewChild' || name === 'ContentChild');
58
- // Ensure that the current property declaration is defining a query.
59
- if (!queryDecorator) {
60
- return;
61
- }
62
- const queryContainer = findParentClassDeclaration(node);
63
- // If the query is not located within a class declaration, skip this node.
64
- if (!queryContainer) {
65
- return;
66
- }
67
- const sourceFile = node.getSourceFile();
68
- const newQueries = this.resolvedQueries.get(sourceFile) || [];
69
- this.resolvedQueries.set(sourceFile, newQueries.concat({
70
- name: queryName,
71
- type: queryDecorator.name === 'ViewChild' ? QueryType.ViewChild : QueryType.ContentChild,
72
- node,
73
- property,
74
- decorator: queryDecorator,
75
- container: queryContainer,
76
- }));
77
- }
78
- _recordClassInputSetters(node) {
79
- const resolvedInputNames = getInputNamesOfClass(node, this.typeChecker);
80
- if (resolvedInputNames) {
81
- const classMetadata = this._getClassMetadata(node);
82
- classMetadata.ngInputNames = resolvedInputNames;
83
- this.classMetadata.set(node, classMetadata);
84
- }
85
- }
86
- _recordClassInheritances(node) {
87
- const baseTypes = getBaseTypeIdentifiers(node);
88
- if (!baseTypes || baseTypes.length !== 1) {
89
- return;
90
- }
91
- const superClass = baseTypes[0];
92
- const baseClassMetadata = this._getClassMetadata(node);
93
- // We need to resolve the value declaration through the resolved type as the base
94
- // class could be declared in different source files and the local symbol won't
95
- // contain a value declaration as the value is not declared locally.
96
- const symbol = this.typeChecker.getTypeAtLocation(superClass).getSymbol();
97
- if (symbol && symbol.valueDeclaration && ts.isClassDeclaration(symbol.valueDeclaration)) {
98
- const extendedClass = symbol.valueDeclaration;
99
- const classMetadataExtended = this._getClassMetadata(extendedClass);
100
- // Record all classes that derive from the given class. This makes it easy to
101
- // determine all classes that could potentially use inherited queries statically.
102
- classMetadataExtended.derivedClasses.push(node);
103
- this.classMetadata.set(extendedClass, classMetadataExtended);
104
- // Record the super class of the current class.
105
- baseClassMetadata.superClass = extendedClass;
106
- this.classMetadata.set(node, baseClassMetadata);
107
- }
108
- }
109
- _getClassMetadata(node) {
110
- return this.classMetadata.get(node) || { derivedClasses: [], superClass: null, ngInputNames: [] };
111
- }
112
- }
113
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng_query_visitor.js","sourceRoot":"","sources":["../../../../../../../../../packages/core/schematics/migrations/static-queries/angular/ng_query_visitor.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,KAAK,EAAE,MAAM,YAAY,CAAC;AAGjC,OAAO,EAAC,oBAAoB,EAAC,MAAM,8BAA8B,CAAC;AAClE,OAAO,EAAC,0BAA0B,EAAE,sBAAsB,EAAC,MAAM,6CAA6C,CAAC;AAC/G,OAAO,EAAC,mBAAmB,EAAC,MAAM,yCAAyC,CAAC;AAE5E,OAAO,EAAC,oBAAoB,EAAC,MAAM,oBAAoB,CAAC;AACxD,OAAO,EAAoB,SAAS,EAAC,MAAM,oBAAoB,CAAC;AAkBhE;;;;GAIG;AACH,MAAM,OAAO,qBAAqB;IAOhC,YAAmB,WAA2B;QAA3B,gBAAW,GAAX,WAAW,CAAgB;QAN9C,0CAA0C;QAC1C,oBAAe,GAAG,IAAI,GAAG,EAAsC,CAAC;QAEhE,sDAAsD;QACtD,kBAAa,GAAqB,IAAI,GAAG,EAAE,CAAC;IAEK,CAAC;IAElD,SAAS,CAAC,IAAa;QACrB,QAAQ,IAAI,CAAC,IAAI,EAAE;YACjB,KAAK,EAAE,CAAC,UAAU,CAAC,mBAAmB;gBACpC,IAAI,CAAC,wBAAwB,CAAC,IAA8B,CAAC,CAAC;gBAC9D,MAAM;YACR,KAAK,EAAE,CAAC,UAAU,CAAC,gBAAgB;gBACjC,IAAI,CAAC,qBAAqB,CAAC,IAA2B,CAAC,CAAC;gBACxD,MAAM;YACR,KAAK,EAAE,CAAC,UAAU,CAAC,WAAW,CAAC;YAC/B,KAAK,EAAE,CAAC,UAAU,CAAC,WAAW;gBAC5B,IAAI,CAAC,wBAAwB,CAAC,IAA8B,CAAC,CAAC;gBAC9D,MAAM;SACT;QAED,EAAE,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;IAChD,CAAC;IAEO,wBAAwB,CAAC,IAA4B;QAC3D,IAAI,CAAC,uBAAuB,CAAC,IAAI,EAAE,IAAI,EAAE,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;IAC3E,CAAC;IAEO,wBAAwB,CAAC,IAA4B;QAC3D,IAAI,CAAC,uBAAuB,CAAC,IAAI,EAAE,IAAI,EAAE,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;IAC3E,CAAC;IAEO,qBAAqB,CAAC,IAAyB;QACrD,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,CAAC;QACpC,IAAI,CAAC,wBAAwB,CAAC,IAAI,CAAC,CAAC;IACtC,CAAC;IAEO,uBAAuB,CAC3B,IAAa,EAAE,QAAqC,EAAE,SAAsB;QAC9E,IAAI,CAAC,IAAI,CAAC,UAAU,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,MAAM,EAAE;YAC/C,OAAO;SACR;QAED,MAAM,YAAY,GAAG,oBAAoB,CAAC,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;QAC7E,MAAM,cAAc,GAChB,YAAY,CAAC,IAAI,CAAC,CAAC,EAAC,IAAI,EAAC,EAAE,EAAE,CAAC,IAAI,KAAK,WAAW,IAAI,IAAI,KAAK,cAAc,CAAC,CAAC;QAEnF,oEAAoE;QACpE,IAAI,CAAC,cAAc,EAAE;YACnB,OAAO;SACR;QAED,MAAM,cAAc,GAAG,0BAA0B,CAAC,IAAI,CAAC,CAAC;QAExD,0EAA0E;QAC1E,IAAI,CAAC,cAAc,EAAE;YACnB,OAAO;SACR;QAED,MAAM,UAAU,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QACxC,MAAM,UAAU,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;QAE9D,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,UAAU,EAAE,UAAU,CAAC,MAAM,CAAC;YACrD,IAAI,EAAE,SAAS;YACf,IAAI,EAAE,cAAc,CAAC,IAAI,KAAK,WAAW,CAAC,CAAC,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,YAAY;YACxF,IAAI;YACJ,QAAQ;YACR,SAAS,EAAE,cAAc;YACzB,SAAS,EAAE,cAAc;SAC1B,CAAC,CAAC,CAAC;IACN,CAAC;IAEO,wBAAwB,CAAC,IAAyB;QACxD,MAAM,kBAAkB,GAAG,oBAAoB,CAAC,IAAI,EAAE,IAAI,CAAC,WAAW,CAAC,CAAC;QAExE,IAAI,kBAAkB,EAAE;YACtB,MAAM,aAAa,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;YAEnD,aAAa,CAAC,YAAY,GAAG,kBAAkB,CAAC;YAChD,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,IAAI,EAAE,aAAa,CAAC,CAAC;SAC7C;IACH,CAAC;IAEO,wBAAwB,CAAC,IAAyB;QACxD,MAAM,SAAS,GAAG,sBAAsB,CAAC,IAAI,CAAC,CAAC;QAE/C,IAAI,CAAC,SAAS,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE;YACxC,OAAO;SACR;QAED,MAAM,UAAU,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC;QAChC,MAAM,iBAAiB,GAAG,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC;QAEvD,iFAAiF;QACjF,+EAA+E;QAC/E,oEAAoE;QACpE,MAAM,MAAM,GAAG,IAAI,CAAC,WAAW,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC,SAAS,EAAE,CAAC;QAE1E,IAAI,MAAM,IAAI,MAAM,CAAC,gBAAgB,IAAI,EAAE,CAAC,kBAAkB,CAAC,MAAM,CAAC,gBAAgB,CAAC,EAAE;YACvF,MAAM,aAAa,GAAG,MAAM,CAAC,gBAAgB,CAAC;YAC9C,MAAM,qBAAqB,GAAG,IAAI,CAAC,iBAAiB,CAAC,aAAa,CAAC,CAAC;YAEpE,6EAA6E;YAC7E,iFAAiF;YACjF,qBAAqB,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAChD,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,aAAa,EAAE,qBAAqB,CAAC,CAAC;YAE7D,+CAA+C;YAC/C,iBAAiB,CAAC,UAAU,GAAG,aAAa,CAAC;YAC7C,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,IAAI,EAAE,iBAAiB,CAAC,CAAC;SACjD;IACH,CAAC;IAEO,iBAAiB,CAAC,IAAyB;QACjD,OAAO,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,EAAC,cAAc,EAAE,EAAE,EAAE,UAAU,EAAE,IAAI,EAAE,YAAY,EAAE,EAAE,EAAC,CAAC;IAClG,CAAC;CACF","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 {ResolvedTemplate} from '../../../utils/ng_component_template';\nimport {getAngularDecorators} from '../../../utils/ng_decorators';\nimport {findParentClassDeclaration, getBaseTypeIdentifiers} from '../../../utils/typescript/class_declaration';\nimport {getPropertyNameText} from '../../../utils/typescript/property_name';\n\nimport {getInputNamesOfClass} from './directive_inputs';\nimport {NgQueryDefinition, QueryType} from './query-definition';\n\n\n/** Resolved metadata of a given class. */\nexport interface ClassMetadata {\n  /** List of class declarations that derive from the given class. */\n  derivedClasses: ts.ClassDeclaration[];\n  /** Super class of the given class. */\n  superClass: ts.ClassDeclaration|null;\n  /** List of property names that declare an Angular input within the given class. */\n  ngInputNames: string[];\n  /** Component template that belongs to that class if present. */\n  template?: ResolvedTemplate;\n}\n\n/** Type that describes a map which can be used to get a class declaration's metadata. */\nexport type ClassMetadataMap = Map<ts.ClassDeclaration, ClassMetadata>;\n\n/**\n * Visitor that can be used to determine Angular queries within given TypeScript nodes.\n * Besides resolving queries, the visitor also records class relations and searches for\n * Angular input setters which can be used to analyze the timing usage of a given query.\n */\nexport class NgQueryResolveVisitor {\n  /** Resolved Angular query definitions. */\n  resolvedQueries = new Map<ts.SourceFile, NgQueryDefinition[]>();\n\n  /** Maps a class declaration to its class metadata. */\n  classMetadata: ClassMetadataMap = new Map();\n\n  constructor(public typeChecker: ts.TypeChecker) {}\n\n  visitNode(node: ts.Node) {\n    switch (node.kind) {\n      case ts.SyntaxKind.PropertyDeclaration:\n        this.visitPropertyDeclaration(node as ts.PropertyDeclaration);\n        break;\n      case ts.SyntaxKind.ClassDeclaration:\n        this.visitClassDeclaration(node as ts.ClassDeclaration);\n        break;\n      case ts.SyntaxKind.GetAccessor:\n      case ts.SyntaxKind.SetAccessor:\n        this.visitAccessorDeclaration(node as ts.AccessorDeclaration);\n        break;\n    }\n\n    ts.forEachChild(node, n => this.visitNode(n));\n  }\n\n  private visitPropertyDeclaration(node: ts.PropertyDeclaration) {\n    this._recordQueryDeclaration(node, node, getPropertyNameText(node.name));\n  }\n\n  private visitAccessorDeclaration(node: ts.AccessorDeclaration) {\n    this._recordQueryDeclaration(node, null, getPropertyNameText(node.name));\n  }\n\n  private visitClassDeclaration(node: ts.ClassDeclaration) {\n    this._recordClassInputSetters(node);\n    this._recordClassInheritances(node);\n  }\n\n  private _recordQueryDeclaration(\n      node: ts.Node, property: ts.PropertyDeclaration|null, queryName: string|null) {\n    if (!node.decorators || !node.decorators.length) {\n      return;\n    }\n\n    const ngDecorators = getAngularDecorators(this.typeChecker, node.decorators);\n    const queryDecorator =\n        ngDecorators.find(({name}) => name === 'ViewChild' || name === 'ContentChild');\n\n    // Ensure that the current property declaration is defining a query.\n    if (!queryDecorator) {\n      return;\n    }\n\n    const queryContainer = findParentClassDeclaration(node);\n\n    // If the query is not located within a class declaration, skip this node.\n    if (!queryContainer) {\n      return;\n    }\n\n    const sourceFile = node.getSourceFile();\n    const newQueries = this.resolvedQueries.get(sourceFile) || [];\n\n    this.resolvedQueries.set(sourceFile, newQueries.concat({\n      name: queryName,\n      type: queryDecorator.name === 'ViewChild' ? QueryType.ViewChild : QueryType.ContentChild,\n      node,\n      property,\n      decorator: queryDecorator,\n      container: queryContainer,\n    }));\n  }\n\n  private _recordClassInputSetters(node: ts.ClassDeclaration) {\n    const resolvedInputNames = getInputNamesOfClass(node, this.typeChecker);\n\n    if (resolvedInputNames) {\n      const classMetadata = this._getClassMetadata(node);\n\n      classMetadata.ngInputNames = resolvedInputNames;\n      this.classMetadata.set(node, classMetadata);\n    }\n  }\n\n  private _recordClassInheritances(node: ts.ClassDeclaration) {\n    const baseTypes = getBaseTypeIdentifiers(node);\n\n    if (!baseTypes || baseTypes.length !== 1) {\n      return;\n    }\n\n    const superClass = baseTypes[0];\n    const baseClassMetadata = this._getClassMetadata(node);\n\n    // We need to resolve the value declaration through the resolved type as the base\n    // class could be declared in different source files and the local symbol won't\n    // contain a value declaration as the value is not declared locally.\n    const symbol = this.typeChecker.getTypeAtLocation(superClass).getSymbol();\n\n    if (symbol && symbol.valueDeclaration && ts.isClassDeclaration(symbol.valueDeclaration)) {\n      const extendedClass = symbol.valueDeclaration;\n      const classMetadataExtended = this._getClassMetadata(extendedClass);\n\n      // Record all classes that derive from the given class. This makes it easy to\n      // determine all classes that could potentially use inherited queries statically.\n      classMetadataExtended.derivedClasses.push(node);\n      this.classMetadata.set(extendedClass, classMetadataExtended);\n\n      // Record the super class of the current class.\n      baseClassMetadata.superClass = extendedClass;\n      this.classMetadata.set(node, baseClassMetadata);\n    }\n  }\n\n  private _getClassMetadata(node: ts.ClassDeclaration): ClassMetadata {\n    return this.classMetadata.get(node) || {derivedClasses: [], superClass: null, ngInputNames: []};\n  }\n}\n"]}
@@ -1,20 +0,0 @@
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
- /** Timing of a given query. Either static or dynamic. */
9
- export var QueryTiming;
10
- (function (QueryTiming) {
11
- QueryTiming[QueryTiming["STATIC"] = 0] = "STATIC";
12
- QueryTiming[QueryTiming["DYNAMIC"] = 1] = "DYNAMIC";
13
- })(QueryTiming || (QueryTiming = {}));
14
- /** Type of a given query. */
15
- export var QueryType;
16
- (function (QueryType) {
17
- QueryType[QueryType["ViewChild"] = 0] = "ViewChild";
18
- QueryType[QueryType["ContentChild"] = 1] = "ContentChild";
19
- })(QueryType || (QueryType = {}));
20
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicXVlcnktZGVmaW5pdGlvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2NvcmUvc2NoZW1hdGljcy9taWdyYXRpb25zL3N0YXRpYy1xdWVyaWVzL2FuZ3VsYXIvcXVlcnktZGVmaW5pdGlvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7O0dBTUc7QUFLSCx5REFBeUQ7QUFDekQsTUFBTSxDQUFOLElBQVksV0FHWDtBQUhELFdBQVksV0FBVztJQUNyQixpREFBTSxDQUFBO0lBQ04sbURBQU8sQ0FBQTtBQUNULENBQUMsRUFIVyxXQUFXLEtBQVgsV0FBVyxRQUd0QjtBQUVELDZCQUE2QjtBQUM3QixNQUFNLENBQU4sSUFBWSxTQUdYO0FBSEQsV0FBWSxTQUFTO0lBQ25CLG1EQUFTLENBQUE7SUFDVCx5REFBWSxDQUFBO0FBQ2QsQ0FBQyxFQUhXLFNBQVMsS0FBVCxTQUFTLFFBR3BCIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBAbGljZW5zZVxuICogQ29weXJpZ2h0IEdvb2dsZSBMTEMgQWxsIFJpZ2h0cyBSZXNlcnZlZC5cbiAqXG4gKiBVc2Ugb2YgdGhpcyBzb3VyY2UgY29kZSBpcyBnb3Zlcm5lZCBieSBhbiBNSVQtc3R5bGUgbGljZW5zZSB0aGF0IGNhbiBiZVxuICogZm91bmQgaW4gdGhlIExJQ0VOU0UgZmlsZSBhdCBodHRwczovL2FuZ3VsYXIuaW8vbGljZW5zZVxuICovXG5cbmltcG9ydCAqIGFzIHRzIGZyb20gJ3R5cGVzY3JpcHQnO1xuaW1wb3J0IHtOZ0RlY29yYXRvcn0gZnJvbSAnLi4vLi4vLi4vdXRpbHMvbmdfZGVjb3JhdG9ycyc7XG5cbi8qKiBUaW1pbmcgb2YgYSBnaXZlbiBxdWVyeS4gRWl0aGVyIHN0YXRpYyBvciBkeW5hbWljLiAqL1xuZXhwb3J0IGVudW0gUXVlcnlUaW1pbmcge1xuICBTVEFUSUMsXG4gIERZTkFNSUMsXG59XG5cbi8qKiBUeXBlIG9mIGEgZ2l2ZW4gcXVlcnkuICovXG5leHBvcnQgZW51bSBRdWVyeVR5cGUge1xuICBWaWV3Q2hpbGQsXG4gIENvbnRlbnRDaGlsZFxufVxuXG5leHBvcnQgaW50ZXJmYWNlIE5nUXVlcnlEZWZpbml0aW9uIHtcbiAgLyoqIE5hbWUgb2YgdGhlIHF1ZXJ5LiBTZXQgdG8gXCJudWxsXCIgaW4gY2FzZSB0aGUgcXVlcnkgbmFtZSBpcyBub3Qgc3RhdGljYWxseSBhbmFseXphYmxlLiAqL1xuICBuYW1lOiBzdHJpbmd8bnVsbDtcbiAgLyoqIFR5cGUgb2YgdGhlIHF1ZXJ5IGRlZmluaXRpb24uICovXG4gIHR5cGU6IFF1ZXJ5VHlwZTtcbiAgLyoqIE5vZGUgdGhhdCBkZWNsYXJlcyB0aGlzIHF1ZXJ5LiAqL1xuICBub2RlOiB0cy5Ob2RlO1xuICAvKipcbiAgICogUHJvcGVydHkgZGVjbGFyYXRpb24gdGhhdCByZWZlcnMgdG8gdGhlIHF1ZXJ5IHZhbHVlLiBGb3IgYWNjZXNzb3JzIHRoZXJlXG4gICAqIGlzIG5vIHByb3BlcnR5IHRoYXQgaXMgZ3VhcmFudGVlZCB0byBhY2Nlc3MgdGhlIHF1ZXJ5IHZhbHVlLlxuICAgKi9cbiAgcHJvcGVydHk6IHRzLlByb3BlcnR5RGVjbGFyYXRpb258bnVsbDtcbiAgLyoqIERlY29yYXRvciB0aGF0IGRlY2xhcmVzIHRoaXMgYXMgYSBxdWVyeS4gKi9cbiAgZGVjb3JhdG9yOiBOZ0RlY29yYXRvcjtcbiAgLyoqIENsYXNzIGRlY2xhcmF0aW9uIHRoYXQgaG9sZHMgdGhpcyBxdWVyeS4gKi9cbiAgY29udGFpbmVyOiB0cy5DbGFzc0RlY2xhcmF0aW9uO1xufVxuIl19
@@ -1,21 +0,0 @@
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
- /**
9
- * Gets all chained super-class TypeScript declarations for the given class
10
- * by using the specified class metadata map.
11
- */
12
- export function getSuperClassDeclarations(classDecl, classMetadataMap) {
13
- const declarations = [];
14
- let current = classMetadataMap.get(classDecl);
15
- while (current && current.superClass) {
16
- declarations.push(current.superClass);
17
- current = classMetadataMap.get(current.superClass);
18
- }
19
- return declarations;
20
- }
21
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3VwZXJfY2xhc3MuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9jb3JlL3NjaGVtYXRpY3MvbWlncmF0aW9ucy9zdGF0aWMtcXVlcmllcy9hbmd1bGFyL3N1cGVyX2NsYXNzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7R0FNRztBQUtIOzs7R0FHRztBQUNILE1BQU0sVUFBVSx5QkFBeUIsQ0FDckMsU0FBOEIsRUFBRSxnQkFBa0M7SUFDcEUsTUFBTSxZQUFZLEdBQTBCLEVBQUUsQ0FBQztJQUUvQyxJQUFJLE9BQU8sR0FBRyxnQkFBZ0IsQ0FBQyxHQUFHLENBQUMsU0FBUyxDQUFDLENBQUM7SUFDOUMsT0FBTyxPQUFPLElBQUksT0FBTyxDQUFDLFVBQVUsRUFBRTtRQUNwQyxZQUFZLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsQ0FBQztRQUN0QyxPQUFPLEdBQUcsZ0JBQWdCLENBQUMsR0FBRyxDQUFDLE9BQU8sQ0FBQyxVQUFVLENBQUMsQ0FBQztLQUNwRDtJQUVELE9BQU8sWUFBWSxDQUFDO0FBQ3RCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuaW1wb3J0ICogYXMgdHMgZnJvbSAndHlwZXNjcmlwdCc7XG5pbXBvcnQge0NsYXNzTWV0YWRhdGFNYXB9IGZyb20gJy4vbmdfcXVlcnlfdmlzaXRvcic7XG5cbi8qKlxuICogR2V0cyBhbGwgY2hhaW5lZCBzdXBlci1jbGFzcyBUeXBlU2NyaXB0IGRlY2xhcmF0aW9ucyBmb3IgdGhlIGdpdmVuIGNsYXNzXG4gKiBieSB1c2luZyB0aGUgc3BlY2lmaWVkIGNsYXNzIG1ldGFkYXRhIG1hcC5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGdldFN1cGVyQ2xhc3NEZWNsYXJhdGlvbnMoXG4gICAgY2xhc3NEZWNsOiB0cy5DbGFzc0RlY2xhcmF0aW9uLCBjbGFzc01ldGFkYXRhTWFwOiBDbGFzc01ldGFkYXRhTWFwKSB7XG4gIGNvbnN0IGRlY2xhcmF0aW9uczogdHMuQ2xhc3NEZWNsYXJhdGlvbltdID0gW107XG5cbiAgbGV0IGN1cnJlbnQgPSBjbGFzc01ldGFkYXRhTWFwLmdldChjbGFzc0RlY2wpO1xuICB3aGlsZSAoY3VycmVudCAmJiBjdXJyZW50LnN1cGVyQ2xhc3MpIHtcbiAgICBkZWNsYXJhdGlvbnMucHVzaChjdXJyZW50LnN1cGVyQ2xhc3MpO1xuICAgIGN1cnJlbnQgPSBjbGFzc01ldGFkYXRhTWFwLmdldChjdXJyZW50LnN1cGVyQ2xhc3MpO1xuICB9XG5cbiAgcmV0dXJuIGRlY2xhcmF0aW9ucztcbn1cbiJdfQ==
@@ -1,206 +0,0 @@
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
- var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
9
- function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
10
- return new (P || (P = Promise))(function (resolve, reject) {
11
- function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
12
- function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
13
- function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
14
- step((generator = generator.apply(thisArg, _arguments || [])).next());
15
- });
16
- };
17
- import { SchematicsException } from '@angular-devkit/schematics';
18
- import { relative } from 'path';
19
- import * as ts from 'typescript';
20
- import { NgComponentTemplateVisitor } from '../../utils/ng_component_template';
21
- import { getProjectTsConfigPaths } from '../../utils/project_tsconfig_paths';
22
- import { canMigrateFile, createMigrationProgram } from '../../utils/typescript/compiler_host';
23
- import { NgQueryResolveVisitor } from './angular/ng_query_visitor';
24
- import { QueryTemplateStrategy } from './strategies/template_strategy/template_strategy';
25
- import { QueryTestStrategy } from './strategies/test_strategy/test_strategy';
26
- import { QueryUsageStrategy } from './strategies/usage_strategy/usage_strategy';
27
- import { getTransformedQueryCallExpr } from './transform';
28
- var SELECTED_STRATEGY;
29
- (function (SELECTED_STRATEGY) {
30
- SELECTED_STRATEGY[SELECTED_STRATEGY["TEMPLATE"] = 0] = "TEMPLATE";
31
- SELECTED_STRATEGY[SELECTED_STRATEGY["USAGE"] = 1] = "USAGE";
32
- SELECTED_STRATEGY[SELECTED_STRATEGY["TESTS"] = 2] = "TESTS";
33
- })(SELECTED_STRATEGY || (SELECTED_STRATEGY = {}));
34
- /** Entry point for the V8 static-query migration. */
35
- export default function () {
36
- return runMigration;
37
- }
38
- /** Runs the V8 migration static-query migration for all determined TypeScript projects. */
39
- function runMigration(tree, context) {
40
- return __awaiter(this, void 0, void 0, function* () {
41
- const { buildPaths, testPaths } = getProjectTsConfigPaths(tree);
42
- const basePath = process.cwd();
43
- const logger = context.logger;
44
- if (!buildPaths.length && !testPaths.length) {
45
- throw new SchematicsException('Could not find any tsconfig file. Cannot migrate queries ' +
46
- 'to add static flag.');
47
- }
48
- const analyzedFiles = new Set();
49
- const buildProjects = new Set();
50
- const failures = [];
51
- const strategy = process.env['NG_STATIC_QUERY_USAGE_STRATEGY'] === 'true' ?
52
- SELECTED_STRATEGY.USAGE :
53
- SELECTED_STRATEGY.TEMPLATE;
54
- for (const tsconfigPath of buildPaths) {
55
- const project = analyzeProject(tree, tsconfigPath, basePath, analyzedFiles, logger);
56
- if (project) {
57
- buildProjects.add(project);
58
- }
59
- }
60
- if (buildProjects.size) {
61
- for (let project of Array.from(buildProjects.values())) {
62
- failures.push(...yield runStaticQueryMigration(tree, project, strategy, logger));
63
- }
64
- }
65
- // For the "test" tsconfig projects we always want to use the test strategy as
66
- // we can't detect the proper timing within spec files.
67
- for (const tsconfigPath of testPaths) {
68
- const project = yield analyzeProject(tree, tsconfigPath, basePath, analyzedFiles, logger);
69
- if (project) {
70
- failures.push(...yield runStaticQueryMigration(tree, project, SELECTED_STRATEGY.TESTS, logger));
71
- }
72
- }
73
- if (failures.length) {
74
- logger.info('');
75
- logger.info('Some queries could not be migrated automatically. Please go');
76
- logger.info('through these manually and apply the appropriate timing.');
77
- logger.info('For more info on how to choose a flag, please see: ');
78
- logger.info('https://v8.angular.io/guide/static-query-migration');
79
- failures.forEach(failure => logger.warn(`⮑ ${failure}`));
80
- }
81
- });
82
- }
83
- /**
84
- * Analyzes the given TypeScript project by looking for queries that need to be
85
- * migrated. In case there are no queries that can be migrated, null is returned.
86
- */
87
- function analyzeProject(tree, tsconfigPath, basePath, analyzedFiles, logger) {
88
- const { program, host } = createMigrationProgram(tree, tsconfigPath, basePath);
89
- const syntacticDiagnostics = program.getSyntacticDiagnostics();
90
- // Syntactic TypeScript errors can throw off the query analysis and therefore we want
91
- // to notify the developer that we couldn't analyze parts of the project. Developers
92
- // can just re-run the migration after fixing these failures.
93
- if (syntacticDiagnostics.length) {
94
- logger.warn(`\nTypeScript project "${tsconfigPath}" has syntactical errors which could cause ` +
95
- `an incomplete migration. Please fix the following failures and rerun the migration:`);
96
- logger.error(ts.formatDiagnostics(syntacticDiagnostics, host));
97
- logger.info('Migration can be rerun with: "ng update @angular/core --from 7 --to 8 --migrate-only"\n');
98
- }
99
- const typeChecker = program.getTypeChecker();
100
- const sourceFiles = program.getSourceFiles().filter(sourceFile => canMigrateFile(basePath, sourceFile, program));
101
- const queryVisitor = new NgQueryResolveVisitor(typeChecker);
102
- // Analyze all project source-files and collect all queries that
103
- // need to be migrated.
104
- sourceFiles.forEach(sourceFile => {
105
- const relativePath = relative(basePath, sourceFile.fileName);
106
- // Only look for queries within the current source files if the
107
- // file has not been analyzed before.
108
- if (!analyzedFiles.has(relativePath)) {
109
- analyzedFiles.add(relativePath);
110
- queryVisitor.visitNode(sourceFile);
111
- }
112
- });
113
- if (queryVisitor.resolvedQueries.size === 0) {
114
- return null;
115
- }
116
- return { program, host, tsconfigPath, typeChecker, basePath, queryVisitor, sourceFiles };
117
- }
118
- /**
119
- * Runs the static query migration for the given project. The schematic analyzes all
120
- * queries within the project and sets up the query timing based on the current usage
121
- * of the query property. e.g. a view query that is not used in any lifecycle hook does
122
- * not need to be static and can be set up with "static: false".
123
- */
124
- function runStaticQueryMigration(tree, project, selectedStrategy, logger) {
125
- return __awaiter(this, void 0, void 0, function* () {
126
- const { sourceFiles, typeChecker, host, queryVisitor, tsconfigPath, basePath } = project;
127
- const printer = ts.createPrinter();
128
- const failureMessages = [];
129
- const templateVisitor = new NgComponentTemplateVisitor(typeChecker);
130
- // If the "usage" strategy is selected, we also need to add the query visitor
131
- // to the analysis visitors so that query usage in templates can be also checked.
132
- if (selectedStrategy === SELECTED_STRATEGY.USAGE) {
133
- sourceFiles.forEach(s => templateVisitor.visitNode(s));
134
- }
135
- const { resolvedQueries, classMetadata } = queryVisitor;
136
- const { resolvedTemplates } = templateVisitor;
137
- if (selectedStrategy === SELECTED_STRATEGY.USAGE) {
138
- // Add all resolved templates to the class metadata if the usage strategy is used. This
139
- // is necessary in order to be able to check component templates for static query usage.
140
- resolvedTemplates.forEach(template => {
141
- if (classMetadata.has(template.container)) {
142
- classMetadata.get(template.container).template = template;
143
- }
144
- });
145
- }
146
- let strategy;
147
- if (selectedStrategy === SELECTED_STRATEGY.USAGE) {
148
- strategy = new QueryUsageStrategy(classMetadata, typeChecker);
149
- }
150
- else if (selectedStrategy === SELECTED_STRATEGY.TESTS) {
151
- strategy = new QueryTestStrategy();
152
- }
153
- else {
154
- strategy = new QueryTemplateStrategy(tsconfigPath, classMetadata, host);
155
- }
156
- try {
157
- strategy.setup();
158
- }
159
- catch (e) {
160
- if (selectedStrategy === SELECTED_STRATEGY.TEMPLATE) {
161
- logger.warn(`\nThe template migration strategy uses the Angular compiler ` +
162
- `internally and therefore projects that no longer build successfully after ` +
163
- `the update cannot use the template migration strategy. Please ensure ` +
164
- `there are no AOT compilation errors.\n`);
165
- }
166
- // In case the strategy could not be set up properly, we just exit the
167
- // migration. We don't want to throw an exception as this could mean
168
- // that other migrations are interrupted.
169
- logger.warn(`Could not setup migration strategy for "${project.tsconfigPath}". The ` +
170
- `following error has been reported:\n`);
171
- logger.error(`${e.toString()}\n`);
172
- logger.info('Migration can be rerun with: "ng update @angular/core --from 7 --to 8 --migrate-only"\n');
173
- return [];
174
- }
175
- // Walk through all source files that contain resolved queries and update
176
- // the source files if needed. Note that we need to update multiple queries
177
- // within a source file within the same recorder in order to not throw off
178
- // the TypeScript node offsets.
179
- resolvedQueries.forEach((queries, sourceFile) => {
180
- const relativePath = relative(basePath, sourceFile.fileName);
181
- const update = tree.beginUpdate(relativePath);
182
- // Compute the query timing for all resolved queries and update the
183
- // query definitions to explicitly set the determined query timing.
184
- queries.forEach(q => {
185
- const queryExpr = q.decorator.node.expression;
186
- const { timing, message } = strategy.detectTiming(q);
187
- const result = getTransformedQueryCallExpr(q, timing, !!message);
188
- if (!result) {
189
- return;
190
- }
191
- const newText = printer.printNode(ts.EmitHint.Unspecified, result.node, sourceFile);
192
- // Replace the existing query decorator call expression with the updated
193
- // call expression node.
194
- update.remove(queryExpr.getStart(), queryExpr.getWidth());
195
- update.insertRight(queryExpr.getStart(), newText);
196
- if (result.failureMessage || message) {
197
- const { line, character } = ts.getLineAndCharacterOfPosition(sourceFile, q.decorator.node.getStart());
198
- failureMessages.push(`${relativePath}@${line + 1}:${character + 1}: ${result.failureMessage || message}`);
199
- }
200
- });
201
- tree.commitUpdate(update);
202
- });
203
- return failureMessages;
204
- });
205
- }
206
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../../../../../packages/core/schematics/migrations/static-queries/index.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;;;;;;;;;;AAGH,OAAO,EAAyB,mBAAmB,EAAO,MAAM,4BAA4B,CAAC;AAC7F,OAAO,EAAC,QAAQ,EAAC,MAAM,MAAM,CAAC;AAC9B,OAAO,KAAK,EAAE,MAAM,YAAY,CAAC;AAEjC,OAAO,EAAC,0BAA0B,EAAC,MAAM,mCAAmC,CAAC;AAC7E,OAAO,EAAC,uBAAuB,EAAC,MAAM,oCAAoC,CAAC;AAC3E,OAAO,EAAC,cAAc,EAAE,sBAAsB,EAAC,MAAM,sCAAsC,CAAC;AAE5F,OAAO,EAAC,qBAAqB,EAAC,MAAM,4BAA4B,CAAC;AACjE,OAAO,EAAC,qBAAqB,EAAC,MAAM,kDAAkD,CAAC;AACvF,OAAO,EAAC,iBAAiB,EAAC,MAAM,0CAA0C,CAAC;AAE3E,OAAO,EAAC,kBAAkB,EAAC,MAAM,4CAA4C,CAAC;AAC9E,OAAO,EAAC,2BAA2B,EAAC,MAAM,aAAa,CAAC;AAExD,IAAK,iBAIJ;AAJD,WAAK,iBAAiB;IACpB,iEAAQ,CAAA;IACR,2DAAK,CAAA;IACL,2DAAK,CAAA;AACP,CAAC,EAJI,iBAAiB,KAAjB,iBAAiB,QAIrB;AAYD,qDAAqD;AACrD,MAAM,CAAC,OAAO;IACZ,OAAO,YAAY,CAAC;AACtB,CAAC;AAED,2FAA2F;AAC3F,SAAe,YAAY,CAAC,IAAU,EAAE,OAAyB;;QAC/D,MAAM,EAAC,UAAU,EAAE,SAAS,EAAC,GAAG,uBAAuB,CAAC,IAAI,CAAC,CAAC;QAC9D,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;QAC/B,MAAM,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;QAE9B,IAAI,CAAC,UAAU,CAAC,MAAM,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE;YAC3C,MAAM,IAAI,mBAAmB,CACzB,2DAA2D;gBAC3D,qBAAqB,CAAC,CAAC;SAC5B;QAED,MAAM,aAAa,GAAG,IAAI,GAAG,EAAU,CAAC;QACxC,MAAM,aAAa,GAAG,IAAI,GAAG,EAAmB,CAAC;QACjD,MAAM,QAAQ,GAAG,EAAE,CAAC;QACpB,MAAM,QAAQ,GAAG,OAAO,CAAC,GAAG,CAAC,gCAAgC,CAAC,KAAK,MAAM,CAAC,CAAC;YACvE,iBAAiB,CAAC,KAAK,CAAC,CAAC;YACzB,iBAAiB,CAAC,QAAQ,CAAC;QAE/B,KAAK,MAAM,YAAY,IAAI,UAAU,EAAE;YACrC,MAAM,OAAO,GAAG,cAAc,CAAC,IAAI,EAAE,YAAY,EAAE,QAAQ,EAAE,aAAa,EAAE,MAAM,CAAC,CAAC;YACpF,IAAI,OAAO,EAAE;gBACX,aAAa,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;aAC5B;SACF;QAED,IAAI,aAAa,CAAC,IAAI,EAAE;YACtB,KAAK,IAAI,OAAO,IAAI,KAAK,CAAC,IAAI,CAAC,aAAa,CAAC,MAAM,EAAE,CAAC,EAAE;gBACtD,QAAQ,CAAC,IAAI,CAAC,GAAG,MAAM,uBAAuB,CAAC,IAAI,EAAE,OAAO,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC,CAAC;aAClF;SACF;QAED,8EAA8E;QAC9E,uDAAuD;QACvD,KAAK,MAAM,YAAY,IAAI,SAAS,EAAE;YACpC,MAAM,OAAO,GAAG,MAAM,cAAc,CAAC,IAAI,EAAE,YAAY,EAAE,QAAQ,EAAE,aAAa,EAAE,MAAM,CAAC,CAAC;YAC1F,IAAI,OAAO,EAAE;gBACX,QAAQ,CAAC,IAAI,CACT,GAAG,MAAM,uBAAuB,CAAC,IAAI,EAAE,OAAO,EAAE,iBAAiB,CAAC,KAAK,EAAE,MAAM,CAAC,CAAC,CAAC;aACvF;SACF;QAED,IAAI,QAAQ,CAAC,MAAM,EAAE;YACnB,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAChB,MAAM,CAAC,IAAI,CAAC,6DAA6D,CAAC,CAAC;YAC3E,MAAM,CAAC,IAAI,CAAC,0DAA0D,CAAC,CAAC;YACxE,MAAM,CAAC,IAAI,CAAC,qDAAqD,CAAC,CAAC;YACnE,MAAM,CAAC,IAAI,CAAC,oDAAoD,CAAC,CAAC;YAClE,QAAQ,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,OAAO,EAAE,CAAC,CAAC,CAAC;SAC5D;IACH,CAAC;CAAA;AAED;;;GAGG;AACH,SAAS,cAAc,CACnB,IAAU,EAAE,YAAoB,EAAE,QAAgB,EAAE,aAA0B,EAC9E,MAAyB;IAC3B,MAAM,EAAC,OAAO,EAAE,IAAI,EAAC,GAAG,sBAAsB,CAAC,IAAI,EAAE,YAAY,EAAE,QAAQ,CAAC,CAAC;IAC7E,MAAM,oBAAoB,GAAG,OAAO,CAAC,uBAAuB,EAAE,CAAC;IAE/D,qFAAqF;IACrF,oFAAoF;IACpF,6DAA6D;IAC7D,IAAI,oBAAoB,CAAC,MAAM,EAAE;QAC/B,MAAM,CAAC,IAAI,CACP,yBAAyB,YAAY,6CAA6C;YAClF,qFAAqF,CAAC,CAAC;QAC3F,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,iBAAiB,CAAC,oBAAoB,EAAE,IAAI,CAAC,CAAC,CAAC;QAC/D,MAAM,CAAC,IAAI,CACP,yFAAyF,CAAC,CAAC;KAChG;IAED,MAAM,WAAW,GAAG,OAAO,CAAC,cAAc,EAAE,CAAC;IAC7C,MAAM,WAAW,GACb,OAAO,CAAC,cAAc,EAAE,CAAC,MAAM,CAAC,UAAU,CAAC,EAAE,CAAC,cAAc,CAAC,QAAQ,EAAE,UAAU,EAAE,OAAO,CAAC,CAAC,CAAC;IACjG,MAAM,YAAY,GAAG,IAAI,qBAAqB,CAAC,WAAW,CAAC,CAAC;IAE5D,gEAAgE;IAChE,uBAAuB;IACvB,WAAW,CAAC,OAAO,CAAC,UAAU,CAAC,EAAE;QAC/B,MAAM,YAAY,GAAG,QAAQ,CAAC,QAAQ,EAAE,UAAU,CAAC,QAAQ,CAAC,CAAC;QAE7D,+DAA+D;QAC/D,qCAAqC;QACrC,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,YAAY,CAAC,EAAE;YACpC,aAAa,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;YAChC,YAAY,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;SACpC;IACH,CAAC,CAAC,CAAC;IAEH,IAAI,YAAY,CAAC,eAAe,CAAC,IAAI,KAAK,CAAC,EAAE;QAC3C,OAAO,IAAI,CAAC;KACb;IAED,OAAO,EAAC,OAAO,EAAE,IAAI,EAAE,YAAY,EAAE,WAAW,EAAE,QAAQ,EAAE,YAAY,EAAE,WAAW,EAAC,CAAC;AACzF,CAAC;AAED;;;;;GAKG;AACH,SAAe,uBAAuB,CAClC,IAAU,EAAE,OAAwB,EAAE,gBAAmC,EACzE,MAAyB;;QAC3B,MAAM,EAAC,WAAW,EAAE,WAAW,EAAE,IAAI,EAAE,YAAY,EAAE,YAAY,EAAE,QAAQ,EAAC,GAAG,OAAO,CAAC;QACvF,MAAM,OAAO,GAAG,EAAE,CAAC,aAAa,EAAE,CAAC;QACnC,MAAM,eAAe,GAAa,EAAE,CAAC;QACrC,MAAM,eAAe,GAAG,IAAI,0BAA0B,CAAC,WAAW,CAAC,CAAC;QAEpE,6EAA6E;QAC7E,iFAAiF;QACjF,IAAI,gBAAgB,KAAK,iBAAiB,CAAC,KAAK,EAAE;YAChD,WAAW,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,eAAe,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,CAAC;SACxD;QAED,MAAM,EAAC,eAAe,EAAE,aAAa,EAAC,GAAG,YAAY,CAAC;QACtD,MAAM,EAAC,iBAAiB,EAAC,GAAG,eAAe,CAAC;QAE5C,IAAI,gBAAgB,KAAK,iBAAiB,CAAC,KAAK,EAAE;YAChD,uFAAuF;YACvF,wFAAwF;YACxF,iBAAiB,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE;gBACnC,IAAI,aAAa,CAAC,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE;oBACzC,aAAa,CAAC,GAAG,CAAC,QAAQ,CAAC,SAAS,CAAE,CAAC,QAAQ,GAAG,QAAQ,CAAC;iBAC5D;YACH,CAAC,CAAC,CAAC;SACJ;QAED,IAAI,QAAwB,CAAC;QAC7B,IAAI,gBAAgB,KAAK,iBAAiB,CAAC,KAAK,EAAE;YAChD,QAAQ,GAAG,IAAI,kBAAkB,CAAC,aAAa,EAAE,WAAW,CAAC,CAAC;SAC/D;aAAM,IAAI,gBAAgB,KAAK,iBAAiB,CAAC,KAAK,EAAE;YACvD,QAAQ,GAAG,IAAI,iBAAiB,EAAE,CAAC;SACpC;aAAM;YACL,QAAQ,GAAG,IAAI,qBAAqB,CAAC,YAAY,EAAE,aAAa,EAAE,IAAI,CAAC,CAAC;SACzE;QAED,IAAI;YACF,QAAQ,CAAC,KAAK,EAAE,CAAC;SAClB;QAAC,OAAO,CAAC,EAAE;YACV,IAAI,gBAAgB,KAAK,iBAAiB,CAAC,QAAQ,EAAE;gBACnD,MAAM,CAAC,IAAI,CACP,8DAA8D;oBAC9D,4EAA4E;oBAC5E,uEAAuE;oBACvE,wCAAwC,CAAC,CAAC;aAC/C;YACD,sEAAsE;YACtE,oEAAoE;YACpE,yCAAyC;YACzC,MAAM,CAAC,IAAI,CACP,2CAA2C,OAAO,CAAC,YAAY,SAAS;gBACxE,sCAAsC,CAAC,CAAC;YAC5C,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,QAAQ,EAAE,IAAI,CAAC,CAAC;YAClC,MAAM,CAAC,IAAI,CACP,yFAAyF,CAAC,CAAC;YAC/F,OAAO,EAAE,CAAC;SACX;QAED,yEAAyE;QACzE,2EAA2E;QAC3E,0EAA0E;QAC1E,+BAA+B;QAC/B,eAAe,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,UAAU,EAAE,EAAE;YAC9C,MAAM,YAAY,GAAG,QAAQ,CAAC,QAAQ,EAAE,UAAU,CAAC,QAAQ,CAAC,CAAC;YAC7D,MAAM,MAAM,GAAG,IAAI,CAAC,WAAW,CAAC,YAAY,CAAC,CAAC;YAE9C,mEAAmE;YACnE,mEAAmE;YACnE,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE;gBAClB,MAAM,SAAS,GAAG,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC;gBAC9C,MAAM,EAAC,MAAM,EAAE,OAAO,EAAC,GAAG,QAAQ,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC;gBACnD,MAAM,MAAM,GAAG,2BAA2B,CAAC,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC,OAAO,CAAC,CAAC;gBAEjE,IAAI,CAAC,MAAM,EAAE;oBACX,OAAO;iBACR;gBAED,MAAM,OAAO,GAAG,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC,QAAQ,CAAC,WAAW,EAAE,MAAM,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;gBAEpF,wEAAwE;gBACxE,wBAAwB;gBACxB,MAAM,CAAC,MAAM,CAAC,SAAS,CAAC,QAAQ,EAAE,EAAE,SAAS,CAAC,QAAQ,EAAE,CAAC,CAAC;gBAC1D,MAAM,CAAC,WAAW,CAAC,SAAS,CAAC,QAAQ,EAAE,EAAE,OAAO,CAAC,CAAC;gBAElD,IAAI,MAAM,CAAC,cAAc,IAAI,OAAO,EAAE;oBACpC,MAAM,EAAC,IAAI,EAAE,SAAS,EAAC,GACnB,EAAE,CAAC,6BAA6B,CAAC,UAAU,EAAE,CAAC,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC;oBAC9E,eAAe,CAAC,IAAI,CAChB,GAAG,YAAY,IAAI,IAAI,GAAG,CAAC,IAAI,SAAS,GAAG,CAAC,KAAK,MAAM,CAAC,cAAc,IAAI,OAAO,EAAE,CAAC,CAAC;iBAC1F;YACH,CAAC,CAAC,CAAC;YAEH,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;QAC5B,CAAC,CAAC,CAAC;QAEH,OAAO,eAAe,CAAC;IACzB,CAAC;CAAA","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 {logging} from '@angular-devkit/core';\nimport {Rule, SchematicContext, SchematicsException, Tree} from '@angular-devkit/schematics';\nimport {relative} from 'path';\nimport * as ts from 'typescript';\n\nimport {NgComponentTemplateVisitor} from '../../utils/ng_component_template';\nimport {getProjectTsConfigPaths} from '../../utils/project_tsconfig_paths';\nimport {canMigrateFile, createMigrationProgram} from '../../utils/typescript/compiler_host';\n\nimport {NgQueryResolveVisitor} from './angular/ng_query_visitor';\nimport {QueryTemplateStrategy} from './strategies/template_strategy/template_strategy';\nimport {QueryTestStrategy} from './strategies/test_strategy/test_strategy';\nimport {TimingStrategy} from './strategies/timing-strategy';\nimport {QueryUsageStrategy} from './strategies/usage_strategy/usage_strategy';\nimport {getTransformedQueryCallExpr} from './transform';\n\nenum SELECTED_STRATEGY {\n  TEMPLATE,\n  USAGE,\n  TESTS,\n}\n\ninterface AnalyzedProject {\n  program: ts.Program;\n  host: ts.CompilerHost;\n  queryVisitor: NgQueryResolveVisitor;\n  sourceFiles: ts.SourceFile[];\n  basePath: string;\n  typeChecker: ts.TypeChecker;\n  tsconfigPath: string;\n}\n\n/** Entry point for the V8 static-query migration. */\nexport default function(): Rule {\n  return runMigration;\n}\n\n/** Runs the V8 migration static-query migration for all determined TypeScript projects. */\nasync function runMigration(tree: Tree, context: SchematicContext) {\n  const {buildPaths, testPaths} = getProjectTsConfigPaths(tree);\n  const basePath = process.cwd();\n  const logger = context.logger;\n\n  if (!buildPaths.length && !testPaths.length) {\n    throw new SchematicsException(\n        'Could not find any tsconfig file. Cannot migrate queries ' +\n        'to add static flag.');\n  }\n\n  const analyzedFiles = new Set<string>();\n  const buildProjects = new Set<AnalyzedProject>();\n  const failures = [];\n  const strategy = process.env['NG_STATIC_QUERY_USAGE_STRATEGY'] === 'true' ?\n      SELECTED_STRATEGY.USAGE :\n      SELECTED_STRATEGY.TEMPLATE;\n\n  for (const tsconfigPath of buildPaths) {\n    const project = analyzeProject(tree, tsconfigPath, basePath, analyzedFiles, logger);\n    if (project) {\n      buildProjects.add(project);\n    }\n  }\n\n  if (buildProjects.size) {\n    for (let project of Array.from(buildProjects.values())) {\n      failures.push(...await runStaticQueryMigration(tree, project, strategy, logger));\n    }\n  }\n\n  // For the \"test\" tsconfig projects we always want to use the test strategy as\n  // we can't detect the proper timing within spec files.\n  for (const tsconfigPath of testPaths) {\n    const project = await analyzeProject(tree, tsconfigPath, basePath, analyzedFiles, logger);\n    if (project) {\n      failures.push(\n          ...await runStaticQueryMigration(tree, project, SELECTED_STRATEGY.TESTS, logger));\n    }\n  }\n\n  if (failures.length) {\n    logger.info('');\n    logger.info('Some queries could not be migrated automatically. Please go');\n    logger.info('through these manually and apply the appropriate timing.');\n    logger.info('For more info on how to choose a flag, please see: ');\n    logger.info('https://v8.angular.io/guide/static-query-migration');\n    failures.forEach(failure => logger.warn(`⮑   ${failure}`));\n  }\n}\n\n/**\n * Analyzes the given TypeScript project by looking for queries that need to be\n * migrated. In case there are no queries that can be migrated, null is returned.\n */\nfunction analyzeProject(\n    tree: Tree, tsconfigPath: string, basePath: string, analyzedFiles: Set<string>,\n    logger: logging.LoggerApi): AnalyzedProject|null {\n  const {program, host} = createMigrationProgram(tree, tsconfigPath, basePath);\n  const syntacticDiagnostics = program.getSyntacticDiagnostics();\n\n  // Syntactic TypeScript errors can throw off the query analysis and therefore we want\n  // to notify the developer that we couldn't analyze parts of the project. Developers\n  // can just re-run the migration after fixing these failures.\n  if (syntacticDiagnostics.length) {\n    logger.warn(\n        `\\nTypeScript project \"${tsconfigPath}\" has syntactical errors which could cause ` +\n        `an incomplete migration. Please fix the following failures and rerun the migration:`);\n    logger.error(ts.formatDiagnostics(syntacticDiagnostics, host));\n    logger.info(\n        'Migration can be rerun with: \"ng update @angular/core --from 7 --to 8 --migrate-only\"\\n');\n  }\n\n  const typeChecker = program.getTypeChecker();\n  const sourceFiles =\n      program.getSourceFiles().filter(sourceFile => canMigrateFile(basePath, sourceFile, program));\n  const queryVisitor = new NgQueryResolveVisitor(typeChecker);\n\n  // Analyze all project source-files and collect all queries that\n  // need to be migrated.\n  sourceFiles.forEach(sourceFile => {\n    const relativePath = relative(basePath, sourceFile.fileName);\n\n    // Only look for queries within the current source files if the\n    // file has not been analyzed before.\n    if (!analyzedFiles.has(relativePath)) {\n      analyzedFiles.add(relativePath);\n      queryVisitor.visitNode(sourceFile);\n    }\n  });\n\n  if (queryVisitor.resolvedQueries.size === 0) {\n    return null;\n  }\n\n  return {program, host, tsconfigPath, typeChecker, basePath, queryVisitor, sourceFiles};\n}\n\n/**\n * Runs the static query migration for the given project. The schematic analyzes all\n * queries within the project and sets up the query timing based on the current usage\n * of the query property. e.g. a view query that is not used in any lifecycle hook does\n * not need to be static and can be set up with \"static: false\".\n */\nasync function runStaticQueryMigration(\n    tree: Tree, project: AnalyzedProject, selectedStrategy: SELECTED_STRATEGY,\n    logger: logging.LoggerApi): Promise<string[]> {\n  const {sourceFiles, typeChecker, host, queryVisitor, tsconfigPath, basePath} = project;\n  const printer = ts.createPrinter();\n  const failureMessages: string[] = [];\n  const templateVisitor = new NgComponentTemplateVisitor(typeChecker);\n\n  // If the \"usage\" strategy is selected, we also need to add the query visitor\n  // to the analysis visitors so that query usage in templates can be also checked.\n  if (selectedStrategy === SELECTED_STRATEGY.USAGE) {\n    sourceFiles.forEach(s => templateVisitor.visitNode(s));\n  }\n\n  const {resolvedQueries, classMetadata} = queryVisitor;\n  const {resolvedTemplates} = templateVisitor;\n\n  if (selectedStrategy === SELECTED_STRATEGY.USAGE) {\n    // Add all resolved templates to the class metadata if the usage strategy is used. This\n    // is necessary in order to be able to check component templates for static query usage.\n    resolvedTemplates.forEach(template => {\n      if (classMetadata.has(template.container)) {\n        classMetadata.get(template.container)!.template = template;\n      }\n    });\n  }\n\n  let strategy: TimingStrategy;\n  if (selectedStrategy === SELECTED_STRATEGY.USAGE) {\n    strategy = new QueryUsageStrategy(classMetadata, typeChecker);\n  } else if (selectedStrategy === SELECTED_STRATEGY.TESTS) {\n    strategy = new QueryTestStrategy();\n  } else {\n    strategy = new QueryTemplateStrategy(tsconfigPath, classMetadata, host);\n  }\n\n  try {\n    strategy.setup();\n  } catch (e) {\n    if (selectedStrategy === SELECTED_STRATEGY.TEMPLATE) {\n      logger.warn(\n          `\\nThe template migration strategy uses the Angular compiler ` +\n          `internally and therefore projects that no longer build successfully after ` +\n          `the update cannot use the template migration strategy. Please ensure ` +\n          `there are no AOT compilation errors.\\n`);\n    }\n    // In case the strategy could not be set up properly, we just exit the\n    // migration. We don't want to throw an exception as this could mean\n    // that other migrations are interrupted.\n    logger.warn(\n        `Could not setup migration strategy for \"${project.tsconfigPath}\". The ` +\n        `following error has been reported:\\n`);\n    logger.error(`${e.toString()}\\n`);\n    logger.info(\n        'Migration can be rerun with: \"ng update @angular/core --from 7 --to 8 --migrate-only\"\\n');\n    return [];\n  }\n\n  // Walk through all source files that contain resolved queries and update\n  // the source files if needed. Note that we need to update multiple queries\n  // within a source file within the same recorder in order to not throw off\n  // the TypeScript node offsets.\n  resolvedQueries.forEach((queries, sourceFile) => {\n    const relativePath = relative(basePath, sourceFile.fileName);\n    const update = tree.beginUpdate(relativePath);\n\n    // Compute the query timing for all resolved queries and update the\n    // query definitions to explicitly set the determined query timing.\n    queries.forEach(q => {\n      const queryExpr = q.decorator.node.expression;\n      const {timing, message} = strategy.detectTiming(q);\n      const result = getTransformedQueryCallExpr(q, timing, !!message);\n\n      if (!result) {\n        return;\n      }\n\n      const newText = printer.printNode(ts.EmitHint.Unspecified, result.node, sourceFile);\n\n      // Replace the existing query decorator call expression with the updated\n      // call expression node.\n      update.remove(queryExpr.getStart(), queryExpr.getWidth());\n      update.insertRight(queryExpr.getStart(), newText);\n\n      if (result.failureMessage || message) {\n        const {line, character} =\n            ts.getLineAndCharacterOfPosition(sourceFile, q.decorator.node.getStart());\n        failureMessages.push(\n            `${relativePath}@${line + 1}:${character + 1}: ${result.failureMessage || message}`);\n      }\n    });\n\n    tree.commitUpdate(update);\n  });\n\n  return failureMessages;\n}\n"]}