UNPKG

86.8 kBJavaScriptView Raw
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/language-service/src/typescript_host", ["require", "exports", "tslib", "@angular/compiler", "@angular/core", "path", "typescript/lib/tsserverlibrary", "@angular/language-service/src/reflector_host", "@angular/language-service/src/template", "@angular/language-service/src/ts_utils"], factory);
15 }
16})(function (require, exports) {
17 "use strict";
18 Object.defineProperty(exports, "__esModule", { value: true });
19 exports.TypeScriptServiceHost = exports.DummyResourceLoader = exports.DummyHtmlParser = void 0;
20 var tslib_1 = require("tslib");
21 var compiler_1 = require("@angular/compiler");
22 var core_1 = require("@angular/core");
23 var path = require("path");
24 var tss = require("typescript/lib/tsserverlibrary");
25 var reflector_host_1 = require("@angular/language-service/src/reflector_host");
26 var template_1 = require("@angular/language-service/src/template");
27 var ts_utils_1 = require("@angular/language-service/src/ts_utils");
28 /**
29 * The language service never needs the normalized versions of the metadata. To avoid parsing
30 * the content and resolving references, return an empty file. This also allows normalizing
31 * template that are syntatically incorrect which is required to provide completions in
32 * syntactically incorrect templates.
33 */
34 var DummyHtmlParser = /** @class */ (function (_super) {
35 tslib_1.__extends(DummyHtmlParser, _super);
36 function DummyHtmlParser() {
37 return _super !== null && _super.apply(this, arguments) || this;
38 }
39 DummyHtmlParser.prototype.parse = function () {
40 return new compiler_1.ParseTreeResult([], []);
41 };
42 return DummyHtmlParser;
43 }(compiler_1.HtmlParser));
44 exports.DummyHtmlParser = DummyHtmlParser;
45 /**
46 * Avoid loading resources in the language servcie by using a dummy loader.
47 */
48 var DummyResourceLoader = /** @class */ (function (_super) {
49 tslib_1.__extends(DummyResourceLoader, _super);
50 function DummyResourceLoader() {
51 return _super !== null && _super.apply(this, arguments) || this;
52 }
53 DummyResourceLoader.prototype.get = function (_url) {
54 return Promise.resolve('');
55 };
56 return DummyResourceLoader;
57 }(compiler_1.ResourceLoader));
58 exports.DummyResourceLoader = DummyResourceLoader;
59 /**
60 * An implementation of a `LanguageServiceHost` for a TypeScript project.
61 *
62 * The `TypeScriptServiceHost` implements the Angular `LanguageServiceHost` using
63 * the TypeScript language services.
64 *
65 * @publicApi
66 */
67 var TypeScriptServiceHost = /** @class */ (function () {
68 function TypeScriptServiceHost(tsLsHost, tsLS) {
69 var _this = this;
70 this.tsLsHost = tsLsHost;
71 this.tsLS = tsLS;
72 this.staticSymbolCache = new compiler_1.StaticSymbolCache();
73 /**
74 * Key of the `fileToComponent` map must be TS internal normalized path (path
75 * separator must be `/`), value of the map is the StaticSymbol for the
76 * Component class declaration.
77 */
78 this.fileToComponent = new Map();
79 this.collectedErrors = new Map();
80 this.fileVersions = new Map();
81 this.lastProgram = undefined;
82 this.analyzedModules = {
83 files: [],
84 ngModuleByPipeOrDirective: new Map(),
85 ngModules: [],
86 };
87 this.summaryResolver = new compiler_1.AotSummaryResolver({
88 loadSummary: function (_filePath) {
89 return null;
90 },
91 isSourceFile: function (_sourceFilePath) {
92 return true;
93 },
94 toSummaryFileName: function (sourceFilePath) {
95 return sourceFilePath;
96 },
97 fromSummaryFileName: function (filePath) {
98 return filePath;
99 },
100 }, this.staticSymbolCache);
101 this.reflectorHost = new reflector_host_1.ReflectorHost(function () { return _this.program; }, tsLsHost);
102 this.staticSymbolResolver = new compiler_1.StaticSymbolResolver(this.reflectorHost, this.staticSymbolCache, this.summaryResolver, function (e, filePath) { return _this.collectError(e, filePath); });
103 this.urlResolver = {
104 resolve: function (baseUrl, url) {
105 // In practice, `directoryExists` is always defined.
106 // https://github.com/microsoft/TypeScript/blob/0b6c9254a850dd07056259d4eefca7721745af75/src/server/project.ts#L1608-L1614
107 if (tsLsHost.directoryExists(baseUrl)) {
108 return path.resolve(baseUrl, url);
109 }
110 return path.resolve(path.dirname(baseUrl), url);
111 }
112 };
113 }
114 Object.defineProperty(TypeScriptServiceHost.prototype, "resolver", {
115 /**
116 * Return the singleton instance of the MetadataResolver.
117 */
118 get: function () {
119 var _this = this;
120 if (this._resolver) {
121 return this._resolver;
122 }
123 // StaticReflector keeps its own private caches that are not clearable.
124 // We have no choice but to create a new instance to invalidate the caches.
125 // TODO: Revisit this when language service gets rewritten for Ivy.
126 var staticReflector = new compiler_1.StaticReflector(this.summaryResolver, this.staticSymbolResolver, [], // knownMetadataClasses
127 [], // knownMetadataFunctions
128 function (e, filePath) { return _this.collectError(e, filePath); });
129 // Because static reflector above is changed, we need to create a new
130 // resolver.
131 var moduleResolver = new compiler_1.NgModuleResolver(staticReflector);
132 var directiveResolver = new compiler_1.DirectiveResolver(staticReflector);
133 var pipeResolver = new compiler_1.PipeResolver(staticReflector);
134 var elementSchemaRegistry = new compiler_1.DomElementSchemaRegistry();
135 var resourceLoader = new DummyResourceLoader();
136 var htmlParser = new DummyHtmlParser();
137 // This tracks the CompileConfig in codegen.ts. Currently these options
138 // are hard-coded.
139 var config = new compiler_1.CompilerConfig({
140 defaultEncapsulation: core_1.ViewEncapsulation.Emulated,
141 useJit: false,
142 });
143 var directiveNormalizer = new compiler_1.DirectiveNormalizer(resourceLoader, this.urlResolver, htmlParser, config);
144 this._resolver = new compiler_1.CompileMetadataResolver(config, htmlParser, moduleResolver, directiveResolver, pipeResolver, new compiler_1.JitSummaryResolver(), elementSchemaRegistry, directiveNormalizer, new core_1.ɵConsole(), this.staticSymbolCache, staticReflector, function (error, type) { return _this.collectError(error, type && type.filePath); });
145 return this._resolver;
146 },
147 enumerable: false,
148 configurable: true
149 });
150 Object.defineProperty(TypeScriptServiceHost.prototype, "reflector", {
151 /**
152 * Return the singleton instance of the StaticReflector hosted in the
153 * MetadataResolver.
154 */
155 get: function () {
156 return this.resolver.getReflector();
157 },
158 enumerable: false,
159 configurable: true
160 });
161 /**
162 * Return all known external templates.
163 */
164 TypeScriptServiceHost.prototype.getExternalTemplates = function () {
165 return tslib_1.__spread(this.fileToComponent.keys());
166 };
167 /**
168 * Checks whether the program has changed and returns all analyzed modules.
169 * If program has changed, invalidate all caches and update fileToComponent
170 * and templateReferences.
171 * In addition to returning information about NgModules, this method plays the
172 * same role as 'synchronizeHostData' in tsserver.
173 */
174 TypeScriptServiceHost.prototype.getAnalyzedModules = function () {
175 var e_1, _a, e_2, _b;
176 if (this.upToDate()) {
177 return this.analyzedModules;
178 }
179 // Invalidate caches
180 this.fileToComponent.clear();
181 this.collectedErrors.clear();
182 this.resolver.clearCache();
183 var analyzeHost = {
184 isSourceFile: function (_filePath) {
185 return true;
186 }
187 };
188 var programFiles = this.program.getSourceFiles().map(function (sf) { return sf.fileName; });
189 try {
190 this.analyzedModules =
191 compiler_1.analyzeNgModules(programFiles, analyzeHost, this.staticSymbolResolver, this.resolver);
192 }
193 catch (e) {
194 // Analyzing modules may throw; in that case, reuse the old modules.
195 this.error("Analyzing NgModules failed. " + e);
196 return this.analyzedModules;
197 }
198 try {
199 // update template references and fileToComponent
200 for (var _c = tslib_1.__values(this.analyzedModules.ngModules), _d = _c.next(); !_d.done; _d = _c.next()) {
201 var ngModule = _d.value;
202 try {
203 for (var _e = (e_2 = void 0, tslib_1.__values(ngModule.declaredDirectives)), _f = _e.next(); !_f.done; _f = _e.next()) {
204 var directive = _f.value;
205 var metadata = this.resolver.getNonNormalizedDirectiveMetadata(directive.reference).metadata;
206 if (metadata.isComponent && metadata.template && metadata.template.templateUrl) {
207 var templateName = this.urlResolver.resolve(this.reflector.componentModuleUrl(directive.reference), metadata.template.templateUrl);
208 this.fileToComponent.set(tss.server.toNormalizedPath(templateName), directive.reference);
209 }
210 }
211 }
212 catch (e_2_1) { e_2 = { error: e_2_1 }; }
213 finally {
214 try {
215 if (_f && !_f.done && (_b = _e.return)) _b.call(_e);
216 }
217 finally { if (e_2) throw e_2.error; }
218 }
219 }
220 }
221 catch (e_1_1) { e_1 = { error: e_1_1 }; }
222 finally {
223 try {
224 if (_d && !_d.done && (_a = _c.return)) _a.call(_c);
225 }
226 finally { if (e_1) throw e_1.error; }
227 }
228 return this.analyzedModules;
229 };
230 /**
231 * Checks whether the program has changed, and invalidate static symbols in
232 * the source files that have changed.
233 * Returns true if modules are up-to-date, false otherwise.
234 * This should only be called by getAnalyzedModules().
235 */
236 TypeScriptServiceHost.prototype.upToDate = function () {
237 var e_3, _a, e_4, _b, e_5, _c;
238 var _d = this, lastProgram = _d.lastProgram, program = _d.program;
239 if (lastProgram === program) {
240 return true;
241 }
242 this.lastProgram = program;
243 // Even though the program has changed, it could be the case that none of
244 // the source files have changed. If all source files remain the same, then
245 // program is still up-to-date, and we should not invalidate caches.
246 var filesAdded = 0;
247 var filesChangedOrRemoved = [];
248 // Check if any source files have been added / changed since last computation.
249 var seen = new Set();
250 var ANGULAR_CORE = '@angular/core';
251 var corePath = this.reflectorHost.moduleNameToFileName(ANGULAR_CORE);
252 try {
253 for (var _e = tslib_1.__values(program.getSourceFiles()), _f = _e.next(); !_f.done; _f = _e.next()) {
254 var fileName = _f.value.fileName;
255 // If `@angular/core` is edited, the language service would have to be
256 // restarted, so ignore changes to `@angular/core`.
257 // When the StaticReflector is initialized at startup, it loads core
258 // symbols from @angular/core by calling initializeConversionMap(). This
259 // is only done once. If the file is invalidated, some of the core symbols
260 // will be lost permanently.
261 if (fileName === corePath) {
262 continue;
263 }
264 seen.add(fileName);
265 var version = this.tsLsHost.getScriptVersion(fileName);
266 var lastVersion = this.fileVersions.get(fileName);
267 if (lastVersion === undefined) {
268 filesAdded++;
269 this.fileVersions.set(fileName, version);
270 }
271 else if (version !== lastVersion) {
272 filesChangedOrRemoved.push(fileName); // changed
273 this.fileVersions.set(fileName, version);
274 }
275 }
276 }
277 catch (e_3_1) { e_3 = { error: e_3_1 }; }
278 finally {
279 try {
280 if (_f && !_f.done && (_a = _e.return)) _a.call(_e);
281 }
282 finally { if (e_3) throw e_3.error; }
283 }
284 try {
285 // Check if any source files have been removed since last computation.
286 for (var _g = tslib_1.__values(this.fileVersions), _h = _g.next(); !_h.done; _h = _g.next()) {
287 var _j = tslib_1.__read(_h.value, 1), fileName = _j[0];
288 if (!seen.has(fileName)) {
289 filesChangedOrRemoved.push(fileName); // removed
290 // Because Maps are iterated in insertion order, it is safe to delete
291 // entries from the same map while iterating.
292 // See https://stackoverflow.com/questions/35940216 and
293 // https://www.ecma-international.org/ecma-262/10.0/index.html#sec-map.prototype.foreach
294 this.fileVersions.delete(fileName);
295 }
296 }
297 }
298 catch (e_4_1) { e_4 = { error: e_4_1 }; }
299 finally {
300 try {
301 if (_h && !_h.done && (_b = _g.return)) _b.call(_g);
302 }
303 finally { if (e_4) throw e_4.error; }
304 }
305 try {
306 for (var filesChangedOrRemoved_1 = tslib_1.__values(filesChangedOrRemoved), filesChangedOrRemoved_1_1 = filesChangedOrRemoved_1.next(); !filesChangedOrRemoved_1_1.done; filesChangedOrRemoved_1_1 = filesChangedOrRemoved_1.next()) {
307 var fileName = filesChangedOrRemoved_1_1.value;
308 var symbols = this.staticSymbolResolver.invalidateFile(fileName);
309 this.reflector.invalidateSymbols(symbols);
310 }
311 }
312 catch (e_5_1) { e_5 = { error: e_5_1 }; }
313 finally {
314 try {
315 if (filesChangedOrRemoved_1_1 && !filesChangedOrRemoved_1_1.done && (_c = filesChangedOrRemoved_1.return)) _c.call(filesChangedOrRemoved_1);
316 }
317 finally { if (e_5) throw e_5.error; }
318 }
319 // Program is up-to-date iff no files are added, changed, or removed.
320 return filesAdded === 0 && filesChangedOrRemoved.length === 0;
321 };
322 /**
323 * Find all templates in the specified `file`.
324 * @param fileName TS or HTML file
325 */
326 TypeScriptServiceHost.prototype.getTemplates = function (fileName) {
327 var _this = this;
328 var results = [];
329 if (fileName.endsWith('.ts')) {
330 // Find every template string in the file
331 var visit_1 = function (child) {
332 var template = _this.getInternalTemplate(child);
333 if (template) {
334 results.push(template);
335 }
336 else {
337 tss.forEachChild(child, visit_1);
338 }
339 };
340 var sourceFile = this.getSourceFile(fileName);
341 if (sourceFile) {
342 tss.forEachChild(sourceFile, visit_1);
343 }
344 }
345 else {
346 var template = this.getExternalTemplate(fileName);
347 if (template) {
348 results.push(template);
349 }
350 }
351 return results;
352 };
353 /**
354 * Return metadata about all class declarations in the file that are Angular
355 * directives. Potential matches are `@NgModule`, `@Component`, `@Directive`,
356 * `@Pipes`, etc. class declarations.
357 *
358 * @param fileName TS file
359 */
360 TypeScriptServiceHost.prototype.getDeclarations = function (fileName) {
361 var _this = this;
362 if (!fileName.endsWith('.ts')) {
363 return [];
364 }
365 var sourceFile = this.getSourceFile(fileName);
366 if (!sourceFile) {
367 return [];
368 }
369 var results = [];
370 var visit = function (child) {
371 var candidate = ts_utils_1.getDirectiveClassLike(child);
372 if (candidate) {
373 var classId = candidate.classId;
374 var declarationSpan = spanOf(classId);
375 var className = classId.getText();
376 var classSymbol = _this.reflector.getStaticSymbol(sourceFile.fileName, className);
377 // Ask the resolver to check if candidate is actually Angular directive
378 if (!_this.resolver.isDirective(classSymbol)) {
379 return;
380 }
381 var data = _this.resolver.getNonNormalizedDirectiveMetadata(classSymbol);
382 if (!data) {
383 return;
384 }
385 results.push({
386 type: classSymbol,
387 declarationSpan: declarationSpan,
388 metadata: data.metadata,
389 errors: _this.getCollectedErrors(declarationSpan, sourceFile),
390 });
391 }
392 else {
393 child.forEachChild(visit);
394 }
395 };
396 tss.forEachChild(sourceFile, visit);
397 return results;
398 };
399 TypeScriptServiceHost.prototype.getSourceFile = function (fileName) {
400 if (!fileName.endsWith('.ts')) {
401 throw new Error("Non-TS source file requested: " + fileName);
402 }
403 return this.program.getSourceFile(fileName);
404 };
405 Object.defineProperty(TypeScriptServiceHost.prototype, "program", {
406 get: function () {
407 var program = this.tsLS.getProgram();
408 if (!program) {
409 // Program is very very unlikely to be undefined.
410 throw new Error('No program in language service!');
411 }
412 return program;
413 },
414 enumerable: false,
415 configurable: true
416 });
417 /**
418 * Return the TemplateSource if `node` is a template node.
419 *
420 * For example,
421 *
422 * @Component({
423 * template: '<div></div>' <-- template node
424 * })
425 * class AppComponent {}
426 * ^---- class declaration node
427 *
428 * @param node Potential template node
429 */
430 TypeScriptServiceHost.prototype.getInternalTemplate = function (node) {
431 if (!tss.isStringLiteralLike(node)) {
432 return;
433 }
434 var tmplAsgn = ts_utils_1.getPropertyAssignmentFromValue(node, 'template');
435 if (!tmplAsgn) {
436 return;
437 }
438 var classDecl = ts_utils_1.getClassDeclFromDecoratorProp(tmplAsgn);
439 if (!classDecl || !classDecl.name) { // Does not handle anonymous class
440 return;
441 }
442 var fileName = node.getSourceFile().fileName;
443 var classSymbol = this.reflector.getStaticSymbol(fileName, classDecl.name.text);
444 return new template_1.InlineTemplate(node, classDecl, classSymbol, this);
445 };
446 /**
447 * Return the external template for `fileName`.
448 * @param fileName HTML file
449 */
450 TypeScriptServiceHost.prototype.getExternalTemplate = function (fileName) {
451 // First get the text for the template
452 var snapshot = this.tsLsHost.getScriptSnapshot(fileName);
453 if (!snapshot) {
454 return;
455 }
456 var source = snapshot.getText(0, snapshot.getLength());
457 // Next find the component class symbol
458 var classSymbol = this.fileToComponent.get(tss.server.toNormalizedPath(fileName));
459 if (!classSymbol) {
460 return;
461 }
462 // Then use the class symbol to find the actual ts.ClassDeclaration node
463 var sourceFile = this.getSourceFile(classSymbol.filePath);
464 if (!sourceFile) {
465 return;
466 }
467 // TODO: This only considers top-level class declarations in a source file.
468 // This would not find a class declaration in a namespace, for example.
469 var classDecl = sourceFile.forEachChild(function (child) {
470 if (tss.isClassDeclaration(child) && child.name && child.name.text === classSymbol.name) {
471 return child;
472 }
473 });
474 if (!classDecl) {
475 return;
476 }
477 return new template_1.ExternalTemplate(source, fileName, classDecl, classSymbol, this);
478 };
479 TypeScriptServiceHost.prototype.collectError = function (error, filePath) {
480 if (filePath) {
481 var errors = this.collectedErrors.get(filePath);
482 if (!errors) {
483 errors = [];
484 this.collectedErrors.set(filePath, errors);
485 }
486 errors.push(error);
487 }
488 };
489 TypeScriptServiceHost.prototype.getCollectedErrors = function (defaultSpan, sourceFile) {
490 var errors = this.collectedErrors.get(sourceFile.fileName);
491 if (!errors) {
492 return [];
493 }
494 // TODO: Add better typings for the errors
495 return errors.map(function (e) {
496 var line = e.line || (e.position && e.position.line);
497 var column = e.column || (e.position && e.position.column);
498 var span = spanAt(sourceFile, line, column) || defaultSpan;
499 if (compiler_1.isFormattedError(e)) {
500 return errorToDiagnosticWithChain(e, span);
501 }
502 return { message: e.message, span: span };
503 });
504 };
505 /**
506 * Return the parsed template for the template at the specified `position`.
507 * @param fileName TS or HTML file
508 * @param position Position of the template in the TS file, otherwise ignored.
509 */
510 TypeScriptServiceHost.prototype.getTemplateAstAtPosition = function (fileName, position) {
511 var template;
512 if (fileName.endsWith('.ts')) {
513 var sourceFile = this.getSourceFile(fileName);
514 if (!sourceFile) {
515 return;
516 }
517 // Find the node that most closely matches the position
518 var node = ts_utils_1.findTightestNode(sourceFile, position);
519 if (!node) {
520 return;
521 }
522 template = this.getInternalTemplate(node);
523 }
524 else {
525 template = this.getExternalTemplate(fileName);
526 }
527 if (!template) {
528 return;
529 }
530 return this.getTemplateAst(template);
531 };
532 /**
533 * Find the NgModule which the directive associated with the `classSymbol`
534 * belongs to, then return its schema and transitive directives and pipes.
535 * @param classSymbol Angular Symbol that defines a directive
536 */
537 TypeScriptServiceHost.prototype.getModuleMetadataForDirective = function (classSymbol) {
538 var e_6, _a, e_7, _b, _c;
539 var result = {
540 directives: [],
541 pipes: [],
542 schemas: [],
543 };
544 // First find which NgModule the directive belongs to.
545 var ngModule = this.analyzedModules.ngModuleByPipeOrDirective.get(classSymbol) ||
546 findSuitableDefaultModule(this.analyzedModules);
547 if (!ngModule) {
548 return result;
549 }
550 // Then gather all transitive directives and pipes.
551 var _d = ngModule.transitiveModule, directives = _d.directives, pipes = _d.pipes;
552 try {
553 for (var directives_1 = tslib_1.__values(directives), directives_1_1 = directives_1.next(); !directives_1_1.done; directives_1_1 = directives_1.next()) {
554 var directive = directives_1_1.value;
555 var data = this.resolver.getNonNormalizedDirectiveMetadata(directive.reference);
556 if (data) {
557 result.directives.push(data.metadata.toSummary());
558 }
559 }
560 }
561 catch (e_6_1) { e_6 = { error: e_6_1 }; }
562 finally {
563 try {
564 if (directives_1_1 && !directives_1_1.done && (_a = directives_1.return)) _a.call(directives_1);
565 }
566 finally { if (e_6) throw e_6.error; }
567 }
568 try {
569 for (var pipes_1 = tslib_1.__values(pipes), pipes_1_1 = pipes_1.next(); !pipes_1_1.done; pipes_1_1 = pipes_1.next()) {
570 var pipe = pipes_1_1.value;
571 var metadata = this.resolver.getOrLoadPipeMetadata(pipe.reference);
572 result.pipes.push(metadata.toSummary());
573 }
574 }
575 catch (e_7_1) { e_7 = { error: e_7_1 }; }
576 finally {
577 try {
578 if (pipes_1_1 && !pipes_1_1.done && (_b = pipes_1.return)) _b.call(pipes_1);
579 }
580 finally { if (e_7) throw e_7.error; }
581 }
582 (_c = result.schemas).push.apply(_c, tslib_1.__spread(ngModule.schemas));
583 return result;
584 };
585 /**
586 * Parse the `template` and return its AST, if any.
587 * @param template template to be parsed
588 */
589 TypeScriptServiceHost.prototype.getTemplateAst = function (template) {
590 var classSymbol = template.type, fileName = template.fileName;
591 var data = this.resolver.getNonNormalizedDirectiveMetadata(classSymbol);
592 if (!data) {
593 return;
594 }
595 var htmlParser = new compiler_1.HtmlParser();
596 var expressionParser = new compiler_1.Parser(new compiler_1.Lexer());
597 var parser = new compiler_1.TemplateParser(new compiler_1.CompilerConfig(), this.reflector, expressionParser, new compiler_1.DomElementSchemaRegistry(), htmlParser, null, // console
598 [] // tranforms
599 );
600 var htmlResult = htmlParser.parse(template.source, fileName, {
601 tokenizeExpansionForms: true,
602 preserveLineEndings: true,
603 });
604 var _a = this.getModuleMetadataForDirective(classSymbol), directives = _a.directives, pipes = _a.pipes, schemas = _a.schemas;
605 var parseResult = parser.tryParseHtml(htmlResult, data.metadata, directives, pipes, schemas);
606 if (!parseResult.templateAst) {
607 return;
608 }
609 return {
610 htmlAst: htmlResult.rootNodes,
611 templateAst: parseResult.templateAst,
612 directive: data.metadata,
613 directives: directives,
614 pipes: pipes,
615 parseErrors: parseResult.errors,
616 expressionParser: expressionParser,
617 template: template,
618 };
619 };
620 /**
621 * Log the specified `msg` to file at INFO level. If logging is not enabled
622 * this method is a no-op.
623 * @param msg Log message
624 */
625 TypeScriptServiceHost.prototype.log = function (msg) {
626 if (this.tsLsHost.log) {
627 this.tsLsHost.log(msg);
628 }
629 };
630 /**
631 * Log the specified `msg` to file at ERROR level. If logging is not enabled
632 * this method is a no-op.
633 * @param msg error message
634 */
635 TypeScriptServiceHost.prototype.error = function (msg) {
636 if (this.tsLsHost.error) {
637 this.tsLsHost.error(msg);
638 }
639 };
640 /**
641 * Log debugging info to file at INFO level, only if verbose setting is turned
642 * on. Otherwise, this method is a no-op.
643 * @param msg debugging message
644 */
645 TypeScriptServiceHost.prototype.debug = function (msg) {
646 var project = this.tsLsHost;
647 if (!project.projectService) {
648 // tsLsHost is not a Project
649 return;
650 }
651 var logger = project.projectService.logger;
652 if (logger.hasLevel(tss.server.LogLevel.verbose)) {
653 logger.info(msg);
654 }
655 };
656 return TypeScriptServiceHost;
657 }());
658 exports.TypeScriptServiceHost = TypeScriptServiceHost;
659 function findSuitableDefaultModule(modules) {
660 var e_8, _a;
661 var result = undefined;
662 var resultSize = 0;
663 try {
664 for (var _b = tslib_1.__values(modules.ngModules), _c = _b.next(); !_c.done; _c = _b.next()) {
665 var module_1 = _c.value;
666 var moduleSize = module_1.transitiveModule.directives.length;
667 if (moduleSize > resultSize) {
668 result = module_1;
669 resultSize = moduleSize;
670 }
671 }
672 }
673 catch (e_8_1) { e_8 = { error: e_8_1 }; }
674 finally {
675 try {
676 if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
677 }
678 finally { if (e_8) throw e_8.error; }
679 }
680 return result;
681 }
682 function spanOf(node) {
683 return { start: node.getStart(), end: node.getEnd() };
684 }
685 function spanAt(sourceFile, line, column) {
686 if (line != null && column != null) {
687 var position_1 = tss.getPositionOfLineAndCharacter(sourceFile, line, column);
688 var findChild = function findChild(node) {
689 if (node.kind > tss.SyntaxKind.LastToken && node.pos <= position_1 && node.end > position_1) {
690 var betterNode = tss.forEachChild(node, findChild);
691 return betterNode || node;
692 }
693 };
694 var node = tss.forEachChild(sourceFile, findChild);
695 if (node) {
696 return { start: node.getStart(), end: node.getEnd() };
697 }
698 }
699 }
700 function convertChain(chain) {
701 return { message: chain.message, next: chain.next ? chain.next.map(convertChain) : undefined };
702 }
703 function errorToDiagnosticWithChain(error, span) {
704 return { message: error.chain ? convertChain(error.chain) : error.message, span: span };
705 }
706});
707//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"typescript_host.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/typescript_host.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;;;;;;;;;;;;;;IAEH,8CAAyhB;IACzhB,sCAAqF;IACrF,2BAA6B;IAC7B,oDAAsD;IAEtD,+EAA+C;IAC/C,mEAA4D;IAC5D,mEAAkI;IAGlI;;;;;OAKG;IACH;QAAqC,2CAAU;QAA/C;;QAIA,CAAC;QAHC,+BAAK,GAAL;YACE,OAAO,IAAI,0BAAe,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;QACrC,CAAC;QACH,sBAAC;IAAD,CAAC,AAJD,CAAqC,qBAAU,GAI9C;IAJY,0CAAe;IAM5B;;OAEG;IACH;QAAyC,+CAAc;QAAvD;;QAIA,CAAC;QAHC,iCAAG,GAAH,UAAI,IAAY;YACd,OAAO,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;QAC7B,CAAC;QACH,0BAAC;IAAD,CAAC,AAJD,CAAyC,yBAAc,GAItD;IAJY,kDAAmB;IAMhC;;;;;;;OAOG;IACH;QAuBE,+BAAqB,QAAiC,EAAW,IAAyB;YAA1F,iBA+BC;YA/BoB,aAAQ,GAAR,QAAQ,CAAyB;YAAW,SAAI,GAAJ,IAAI,CAAqB;YAlBzE,sBAAiB,GAAG,IAAI,4BAAiB,EAAE,CAAC;YAC7D;;;;eAIG;YACc,oBAAe,GAAG,IAAI,GAAG,EAA0C,CAAC;YACpE,oBAAe,GAAG,IAAI,GAAG,EAAiB,CAAC;YAC3C,iBAAY,GAAG,IAAI,GAAG,EAAkB,CAAC;YAGlD,gBAAW,GAA0B,SAAS,CAAC;YAC/C,oBAAe,GAAsB;gBAC3C,KAAK,EAAE,EAAE;gBACT,yBAAyB,EAAE,IAAI,GAAG,EAAE;gBACpC,SAAS,EAAE,EAAE;aACd,CAAC;YAGA,IAAI,CAAC,eAAe,GAAG,IAAI,6BAAkB,CACzC;gBACE,WAAW,EAAX,UAAY,SAAiB;oBAC3B,OAAO,IAAI,CAAC;gBACd,CAAC;gBACD,YAAY,EAAZ,UAAa,eAAuB;oBAClC,OAAO,IAAI,CAAC;gBACd,CAAC;gBACD,iBAAiB,EAAjB,UAAkB,cAAsB;oBACtC,OAAO,cAAc,CAAC;gBACxB,CAAC;gBACD,mBAAmB,EAAnB,UAAoB,QAAgB;oBAClC,OAAO,QAAQ,CAAC;gBAClB,CAAC;aACF,EACD,IAAI,CAAC,iBAAiB,CAAC,CAAC;YAC5B,IAAI,CAAC,aAAa,GAAG,IAAI,8BAAa,CAAC,cAAM,OAAA,KAAI,CAAC,OAAO,EAAZ,CAAY,EAAE,QAAQ,CAAC,CAAC;YACrE,IAAI,CAAC,oBAAoB,GAAG,IAAI,+BAAoB,CAChD,IAAI,CAAC,aAAa,EAAE,IAAI,CAAC,iBAAiB,EAAE,IAAI,CAAC,eAAe,EAChE,UAAC,CAAC,EAAE,QAAQ,IAAK,OAAA,KAAI,CAAC,YAAY,CAAC,CAAC,EAAE,QAAQ,CAAC,EAA9B,CAA8B,CAAC,CAAC;YACrD,IAAI,CAAC,WAAW,GAAG;gBACjB,OAAO,EAAE,UAAC,OAAe,EAAE,GAAW;oBACpC,oDAAoD;oBACpD,0HAA0H;oBAC1H,IAAI,QAAQ,CAAC,eAAgB,CAAC,OAAO,CAAC,EAAE;wBACtC,OAAO,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;qBACnC;oBACD,OAAO,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,GAAG,CAAC,CAAC;gBAClD,CAAC;aACF,CAAC;QACJ,CAAC;QAaD,sBAAY,2CAAQ;YAHpB;;eAEG;iBACH;gBAAA,iBAkCC;gBAjCC,IAAI,IAAI,CAAC,SAAS,EAAE;oBAClB,OAAO,IAAI,CAAC,SAAS,CAAC;iBACvB;gBACD,uEAAuE;gBACvE,2EAA2E;gBAC3E,mEAAmE;gBACnE,IAAM,eAAe,GAAG,IAAI,0BAAe,CACvC,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,oBAAoB,EAC/C,EAAE,EAAG,uBAAuB;gBAC5B,EAAE,EAAG,yBAAyB;gBAC9B,UAAC,CAAC,EAAE,QAAQ,IAAK,OAAA,KAAI,CAAC,YAAY,CAAC,CAAC,EAAE,QAAQ,CAAC,EAA9B,CAA8B,CAAC,CAAC;gBACrD,qEAAqE;gBACrE,YAAY;gBACZ,IAAM,cAAc,GAAG,IAAI,2BAAgB,CAAC,eAAe,CAAC,CAAC;gBAC7D,IAAM,iBAAiB,GAAG,IAAI,4BAAiB,CAAC,eAAe,CAAC,CAAC;gBACjE,IAAM,YAAY,GAAG,IAAI,uBAAY,CAAC,eAAe,CAAC,CAAC;gBACvD,IAAM,qBAAqB,GAAG,IAAI,mCAAwB,EAAE,CAAC;gBAC7D,IAAM,cAAc,GAAG,IAAI,mBAAmB,EAAE,CAAC;gBACjD,IAAM,UAAU,GAAG,IAAI,eAAe,EAAE,CAAC;gBACzC,uEAAuE;gBACvE,kBAAkB;gBAClB,IAAM,MAAM,GAAG,IAAI,yBAAc,CAAC;oBAChC,oBAAoB,EAAE,wBAAiB,CAAC,QAAQ;oBAChD,MAAM,EAAE,KAAK;iBACd,CAAC,CAAC;gBACH,IAAM,mBAAmB,GACrB,IAAI,8BAAmB,CAAC,cAAc,EAAE,IAAI,CAAC,WAAW,EAAE,UAAU,EAAE,MAAM,CAAC,CAAC;gBAClF,IAAI,CAAC,SAAS,GAAG,IAAI,kCAAuB,CACxC,MAAM,EAAE,UAAU,EAAE,cAAc,EAAE,iBAAiB,EAAE,YAAY,EACnE,IAAI,6BAAkB,EAAE,EAAE,qBAAqB,EAAE,mBAAmB,EAAE,IAAI,eAAO,EAAE,EACnF,IAAI,CAAC,iBAAiB,EAAE,eAAe,EACvC,UAAC,KAAK,EAAE,IAAI,IAAK,OAAA,KAAI,CAAC,YAAY,CAAC,KAAK,EAAE,IAAI,IAAI,IAAI,CAAC,QAAQ,CAAC,EAA/C,CAA+C,CAAC,CAAC;gBACtE,OAAO,IAAI,CAAC,SAAS,CAAC;YACxB,CAAC;;;WAAA;QAMD,sBAAY,4CAAS;YAJrB;;;eAGG;iBACH;gBACE,OAAO,IAAI,CAAC,QAAQ,CAAC,YAAY,EAAqB,CAAC;YACzD,CAAC;;;WAAA;QAED;;WAEG;QACH,oDAAoB,GAApB;YACE,wBAAW,IAAI,CAAC,eAAe,CAAC,IAAI,EAAE,EAAE;QAC1C,CAAC;QAED;;;;;;WAMG;QACH,kDAAkB,GAAlB;;YACE,IAAI,IAAI,CAAC,QAAQ,EAAE,EAAE;gBACnB,OAAO,IAAI,CAAC,eAAe,CAAC;aAC7B;YAED,oBAAoB;YACpB,IAAI,CAAC,eAAe,CAAC,KAAK,EAAE,CAAC;YAC7B,IAAI,CAAC,eAAe,CAAC,KAAK,EAAE,CAAC;YAC7B,IAAI,CAAC,QAAQ,CAAC,UAAU,EAAE,CAAC;YAE3B,IAAM,WAAW,GAAG;gBAClB,YAAY,EAAZ,UAAa,SAAiB;oBAC5B,OAAO,IAAI,CAAC;gBACd,CAAC;aACF,CAAC;YACF,IAAM,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,cAAc,EAAE,CAAC,GAAG,CAAC,UAAA,EAAE,IAAI,OAAA,EAAE,CAAC,QAAQ,EAAX,CAAW,CAAC,CAAC;YAE1E,IAAI;gBACF,IAAI,CAAC,eAAe;oBAChB,2BAAgB,CAAC,YAAY,EAAE,WAAW,EAAE,IAAI,CAAC,oBAAoB,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC3F;YAAC,OAAO,CAAC,EAAE;gBACV,oEAAoE;gBACpE,IAAI,CAAC,KAAK,CAAC,iCAA+B,CAAG,CAAC,CAAC;gBAC/C,OAAO,IAAI,CAAC,eAAe,CAAC;aAC7B;;gBAED,iDAAiD;gBACjD,KAAuB,IAAA,KAAA,iBAAA,IAAI,CAAC,eAAe,CAAC,SAAS,CAAA,gBAAA,4BAAE;oBAAlD,IAAM,QAAQ,WAAA;;wBACjB,KAAwB,IAAA,oBAAA,iBAAA,QAAQ,CAAC,kBAAkB,CAAA,CAAA,gBAAA,4BAAE;4BAAhD,IAAM,SAAS,WAAA;4BACX,IAAA,QAAQ,GAAI,IAAI,CAAC,QAAQ,CAAC,iCAAiC,CAAC,SAAS,CAAC,SAAS,CAAE,SAAzE,CAA0E;4BACzF,IAAI,QAAQ,CAAC,WAAW,IAAI,QAAQ,CAAC,QAAQ,IAAI,QAAQ,CAAC,QAAQ,CAAC,WAAW,EAAE;gCAC9E,IAAM,YAAY,GAAG,IAAI,CAAC,WAAW,CAAC,OAAO,CACzC,IAAI,CAAC,SAAS,CAAC,kBAAkB,CAAC,SAAS,CAAC,SAAS,CAAC,EACtD,QAAQ,CAAC,QAAQ,CAAC,WAAW,CAAC,CAAC;gCACnC,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,gBAAgB,CAAC,YAAY,CAAC,EAAE,SAAS,CAAC,SAAS,CAAC,CAAC;6BAC1F;yBACF;;;;;;;;;iBACF;;;;;;;;;YAED,OAAO,IAAI,CAAC,eAAe,CAAC;QAC9B,CAAC;QAED;;;;;WAKG;QACK,wCAAQ,GAAhB;;YACQ,IAAA,KAAyB,IAAI,EAA5B,WAAW,iBAAA,EAAE,OAAO,aAAQ,CAAC;YACpC,IAAI,WAAW,KAAK,OAAO,EAAE;gBAC3B,OAAO,IAAI,CAAC;aACb;YACD,IAAI,CAAC,WAAW,GAAG,OAAO,CAAC;YAE3B,yEAAyE;YACzE,2EAA2E;YAC3E,oEAAoE;YACpE,IAAI,UAAU,GAAG,CAAC,CAAC;YACnB,IAAM,qBAAqB,GAAa,EAAE,CAAC;YAE3C,8EAA8E;YAC9E,IAAM,IAAI,GAAG,IAAI,GAAG,EAAU,CAAC;YAC/B,IAAM,YAAY,GAAG,eAAe,CAAC;YACrC,IAAM,QAAQ,GAAG,IAAI,CAAC,aAAa,CAAC,oBAAoB,CAAC,YAAY,CAAC,CAAC;;gBACvE,KAAyB,IAAA,KAAA,iBAAA,OAAO,CAAC,cAAc,EAAE,CAAA,gBAAA,4BAAE;oBAAvC,IAAA,QAAQ,oBAAA;oBAClB,sEAAsE;oBACtE,mDAAmD;oBACnD,oEAAoE;oBACpE,wEAAwE;oBACxE,0EAA0E;oBAC1E,4BAA4B;oBAC5B,IAAI,QAAQ,KAAK,QAAQ,EAAE;wBACzB,SAAS;qBACV;oBACD,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;oBACnB,IAAM,OAAO,GAAG,IAAI,CAAC,QAAQ,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC;oBACzD,IAAM,WAAW,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;oBACpD,IAAI,WAAW,KAAK,SAAS,EAAE;wBAC7B,UAAU,EAAE,CAAC;wBACb,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;qBAC1C;yBAAM,IAAI,OAAO,KAAK,WAAW,EAAE;wBAClC,qBAAqB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAE,UAAU;wBACjD,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;qBAC1C;iBACF;;;;;;;;;;gBAED,sEAAsE;gBACtE,KAAyB,IAAA,KAAA,iBAAA,IAAI,CAAC,YAAY,CAAA,gBAAA,4BAAE;oBAAjC,IAAA,KAAA,2BAAU,EAAT,QAAQ,QAAA;oBAClB,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE;wBACvB,qBAAqB,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAE,UAAU;wBACjD,qEAAqE;wBACrE,6CAA6C;wBAC7C,uDAAuD;wBACvD,wFAAwF;wBACxF,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;qBACpC;iBACF;;;;;;;;;;gBAED,KAAuB,IAAA,0BAAA,iBAAA,qBAAqB,CAAA,4DAAA,+FAAE;oBAAzC,IAAM,QAAQ,kCAAA;oBACjB,IAAM,OAAO,GAAG,IAAI,CAAC,oBAAoB,CAAC,cAAc,CAAC,QAAQ,CAAC,CAAC;oBACnE,IAAI,CAAC,SAAS,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC;iBAC3C;;;;;;;;;YAED,qEAAqE;YACrE,OAAO,UAAU,KAAK,CAAC,IAAI,qBAAqB,CAAC,MAAM,KAAK,CAAC,CAAC;QAChE,CAAC;QAED;;;WAGG;QACH,4CAAY,GAAZ,UAAa,QAAgB;YAA7B,iBAuBC;YAtBC,IAAM,OAAO,GAAqB,EAAE,CAAC;YACrC,IAAI,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE;gBAC5B,yCAAyC;gBACzC,IAAM,OAAK,GAAG,UAAC,KAAe;oBAC5B,IAAM,QAAQ,GAAG,KAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,CAAC;oBACjD,IAAI,QAAQ,EAAE;wBACZ,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;qBACxB;yBAAM;wBACL,GAAG,CAAC,YAAY,CAAC,KAAK,EAAE,OAAK,CAAC,CAAC;qBAChC;gBACH,CAAC,CAAC;gBACF,IAAM,UAAU,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;gBAChD,IAAI,UAAU,EAAE;oBACd,GAAG,CAAC,YAAY,CAAC,UAAU,EAAE,OAAK,CAAC,CAAC;iBACrC;aACF;iBAAM;gBACL,IAAM,QAAQ,GAAG,IAAI,CAAC,mBAAmB,CAAC,QAAQ,CAAC,CAAC;gBACpD,IAAI,QAAQ,EAAE;oBACZ,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;iBACxB;aACF;YACD,OAAO,OAAO,CAAC;QACjB,CAAC;QAED;;;;;;WAMG;QACH,+CAAe,GAAf,UAAgB,QAAgB;YAAhC,iBAqCC;YApCC,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE;gBAC7B,OAAO,EAAE,CAAC;aACX;YACD,IAAM,UAAU,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;YAChD,IAAI,CAAC,UAAU,EAAE;gBACf,OAAO,EAAE,CAAC;aACX;YACD,IAAM,OAAO,GAAkB,EAAE,CAAC;YAClC,IAAM,KAAK,GAAG,UAAC,KAAe;gBAC5B,IAAM,SAAS,GAAG,gCAAqB,CAAC,KAAK,CAAC,CAAC;gBAC/C,IAAI,SAAS,EAAE;oBACN,IAAA,OAAO,GAAI,SAAS,QAAb,CAAc;oBAC5B,IAAM,eAAe,GAAG,MAAM,CAAC,OAAO,CAAC,CAAC;oBACxC,IAAM,SAAS,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC;oBACpC,IAAM,WAAW,GAAG,KAAI,CAAC,SAAS,CAAC,eAAe,CAAC,UAAU,CAAC,QAAQ,EAAE,SAAS,CAAC,CAAC;oBACnF,uEAAuE;oBACvE,IAAI,CAAC,KAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,WAAW,CAAC,EAAE;wBAC3C,OAAO;qBACR;oBACD,IAAM,IAAI,GAAG,KAAI,CAAC,QAAQ,CAAC,iCAAiC,CAAC,WAAW,CAAC,CAAC;oBAC1E,IAAI,CAAC,IAAI,EAAE;wBACT,OAAO;qBACR;oBACD,OAAO,CAAC,IAAI,CAAC;wBACX,IAAI,EAAE,WAAW;wBACjB,eAAe,iBAAA;wBACf,QAAQ,EAAE,IAAI,CAAC,QAAQ;wBACvB,MAAM,EAAE,KAAI,CAAC,kBAAkB,CAAC,eAAe,EAAE,UAAU,CAAC;qBAC7D,CAAC,CAAC;iBACJ;qBAAM;oBACL,KAAK,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;iBAC3B;YACH,CAAC,CAAC;YACF,GAAG,CAAC,YAAY,CAAC,UAAU,EAAE,KAAK,CAAC,CAAC;YAEpC,OAAO,OAAO,CAAC;QACjB,CAAC;QAED,6CAAa,GAAb,UAAc,QAAgB;YAC5B,IAAI,CAAC,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE;gBAC7B,MAAM,IAAI,KAAK,CAAC,mCAAiC,QAAU,CAAC,CAAC;aAC9D;YACD,OAAO,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;QAC9C,CAAC;QAED,sBAAI,0CAAO;iBAAX;gBACE,IAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;gBACvC,IAAI,CAAC,OAAO,EAAE;oBACZ,iDAAiD;oBACjD,MAAM,IAAI,KAAK,CAAC,iCAAiC,CAAC,CAAC;iBACpD;gBACD,OAAO,OAAO,CAAC;YACjB,CAAC;;;WAAA;QAED;;;;;;;;;;;;WAYG;QACK,mDAAmB,GAA3B,UAA4B,IAAc;YACxC,IAAI,CAAC,GAAG,CAAC,mBAAmB,CAAC,IAAI,CAAC,EAAE;gBAClC,OAAO;aACR;YACD,IAAM,QAAQ,GAAG,yCAA8B,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;YAClE,IAAI,CAAC,QAAQ,EAAE;gBACb,OAAO;aACR;YACD,IAAM,SAAS,GAAG,wCAA6B,CAAC,QAAQ,CAAC,CAAC;YAC1D,IAAI,CAAC,SAAS,IAAI,CAAC,SAAS,CAAC,IAAI,EAAE,EAAG,kCAAkC;gBACtE,OAAO;aACR;YACD,IAAM,QAAQ,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC,QAAQ,CAAC;YAC/C,IAAM,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC,eAAe,CAAC,QAAQ,EAAE,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAClF,OAAO,IAAI,yBAAc,CAAC,IAAI,EAAE,SAAS,EAAE,WAAW,EAAE,IAAI,CAAC,CAAC;QAChE,CAAC;QAED;;;WAGG;QACK,mDAAmB,GAA3B,UAA4B,QAAgB;YAC1C,sCAAsC;YACtC,IAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC;YAC3D,IAAI,CAAC,QAAQ,EAAE;gBACb,OAAO;aACR;YACD,IAAM,MAAM,GAAG,QAAQ,CAAC,OAAO,CAAC,CAAC,EAAE,QAAQ,CAAC,SAAS,EAAE,CAAC,CAAC;YACzD,uCAAuC;YACvC,IAAM,WAAW,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,GAAG,CAAC,MAAM,CAAC,gBAAgB,CAAC,QAAQ,CAAC,CAAC,CAAC;YACpF,IAAI,CAAC,WAAW,EAAE;gBAChB,OAAO;aACR;YACD,wEAAwE;YACxE,IAAM,UAAU,GAAG,IAAI,CAAC,aAAa,CAAC,WAAW,CAAC,QAAQ,CAAC,CAAC;YAC5D,IAAI,CAAC,UAAU,EAAE;gBACf,OAAO;aACR;YACD,2EAA2E;YAC3E,uEAAuE;YACvE,IAAM,SAAS,GAAG,UAAU,CAAC,YAAY,CAAC,UAAC,KAAK;gBAC9C,IAAI,GAAG,CAAC,kBAAkB,CAAC,KAAK,CAAC,IAAI,KAAK,CAAC,IAAI,IAAI,KAAK,CAAC,IAAI,CAAC,IAAI,KAAK,WAAW,CAAC,IAAI,EAAE;oBACvF,OAAO,KAAK,CAAC;iBACd;YACH,CAAC,CAAC,CAAC;YACH,IAAI,CAAC,SAAS,EAAE;gBACd,OAAO;aACR;YACD,OAAO,IAAI,2BAAgB,CAAC,MAAM,EAAE,QAAQ,EAAE,SAAS,EAAE,WAAW,EAAE,IAAI,CAAC,CAAC;QAC9E,CAAC;QAEO,4CAAY,GAApB,UAAqB,KAAU,EAAE,QAAiB;YAChD,IAAI,QAAQ,EAAE;gBACZ,IAAI,MAAM,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;gBAChD,IAAI,CAAC,MAAM,EAAE;oBACX,MAAM,GAAG,EAAE,CAAC;oBACZ,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;iBAC5C;gBACD,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;aACpB;QACH,CAAC;QAEO,kDAAkB,GAA1B,UAA2B,WAAiB,EAAE,UAA0B;YACtE,IAAM,MAAM,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,UAAU,CAAC,QAAQ,CAAC,CAAC;YAC7D,IAAI,CAAC,MAAM,EAAE;gBACX,OAAO,EAAE,CAAC;aACX;YACD,0CAA0C;YAC1C,OAAO,MAAM,CAAC,GAAG,CAAC,UAAC,CAAM;gBACvB,IAAM,IAAI,GAAG,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,CAAC,QAAQ,IAAI,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;gBACvD,IAAM,MAAM,GAAG,CAAC,CAAC,MAAM,IAAI,CAAC,CAAC,CAAC,QAAQ,IAAI,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;gBAC7D,IAAM,IAAI,GAAG,MAAM,CAAC,UAAU,EAAE,IAAI,EAAE,MAAM,CAAC,IAAI,WAAW,CAAC;gBAC7D,IAAI,2BAAgB,CAAC,CAAC,CAAC,EAAE;oBACvB,OAAO,0BAA0B,CAAC,CAAC,EAAE,IAAI,CAAC,CAAC;iBAC5C;gBACD,OAAO,EAAC,OAAO,EAAE,CAAC,CAAC,OAAO,EAAE,IAAI,MAAA,EAAC,CAAC;YACpC,CAAC,CAAC,CAAC;QACL,CAAC;QAED;;;;WAIG;QACH,wDAAwB,GAAxB,UAAyB,QAAgB,EAAE,QAAgB;YACzD,IAAI,QAAkC,CAAC;YACvC,IAAI,QAAQ,CAAC,QAAQ,CAAC,KAAK,CAAC,EAAE;gBAC5B,IAAM,UAAU,GAAG,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC;gBAChD,IAAI,CAAC,UAAU,EAAE;oBACf,OAAO;iBACR;gBACD,uDAAuD;gBACvD,IAAM,IAAI,GAAG,2BAAgB,CAAC,UAAU,EAAE,QAAQ,CAAC,CAAC;gBACpD,IAAI,CAAC,IAAI,EAAE;oBACT,OAAO;iBACR;gBACD,QAAQ,GAAG,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,CAAC;aAC3C;iBAAM;gBACL,QAAQ,GAAG,IAAI,CAAC,mBAAmB,CAAC,QAAQ,CAAC,CAAC;aAC/C;YACD,IAAI,CAAC,QAAQ,EAAE;gBACb,OAAO;aACR;YACD,OAAO,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,CAAC;QACvC,CAAC;QAED;;;;WAIG;QACK,6DAA6B,GAArC,UAAsC,WAAyB;;YAC7D,IAAM,MAAM,GAAG;gBACb,UAAU,EAAE,EAA+B;gBAC3C,KAAK,EAAE,EAA0B;gBACjC,OAAO,EAAE,EAAsB;aAChC,CAAC;YACF,sDAAsD;YACtD,IAAM,QAAQ,GAAG,IAAI,CAAC,eAAe,CAAC,yBAAyB,CAAC,GAAG,CAAC,WAAW,CAAC;gBAC5E,yBAAyB,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;YACpD,IAAI,CAAC,QAAQ,EAAE;gBACb,OAAO,MAAM,CAAC;aACf;YACD,mDAAmD;YAC7C,IAAA,KAAsB,QAAQ,CAAC,gBAAgB,EAA9C,UAAU,gBAAA,EAAE,KAAK,WAA6B,CAAC;;gBACtD,KAAwB,IAAA,eAAA,iBAAA,UAAU,CAAA,sCAAA,8DAAE;oBAA/B,IAAM,SAAS,uBAAA;oBAClB,IAAM,IAAI,GAAG,IAAI,CAAC,QAAQ,CAAC,iCAAiC,CAAC,SAAS,CAAC,SAAS,CAAC,CAAC;oBAClF,IAAI,IAAI,EAAE;wBACR,MAAM,CAAC,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC,CAAC;qBACnD;iBACF;;;;;;;;;;gBACD,KAAmB,IAAA,UAAA,iBAAA,KAAK,CAAA,4BAAA,+CAAE;oBAArB,IAAM,IAAI,kBAAA;oBACb,IAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,qBAAqB,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;oBACrE,MAAM,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,EAAE,CAAC,CAAC;iBACzC;;;;;;;;;YACD,CAAA,KAAA,MAAM,CAAC,OAAO,CAAA,CAAC,IAAI,4BAAI,QAAQ,CAAC,OAAO,GAAE;YACzC,OAAO,MAAM,CAAC;QAChB,CAAC;QAED;;;WAGG;QACH,8CAAc,GAAd,UAAe,QAAwB;YAC9B,IAAM,WAAW,GAAc,QAAQ,KAAtB,EAAE,QAAQ,GAAI,QAAQ,SAAZ,CAAa;YAC/C,IAAM,IAAI,GAAG,IAAI,CAAC,QAAQ,CAAC,iCAAiC,CAAC,WAAW,CAAC,CAAC;YAC1E,IAAI,CAAC,IAAI,EAAE;gBACT,OAAO;aACR;YACD,IAAM,UAAU,GAAG,IAAI,qBAAU,EAAE,CAAC;YACpC,IAAM,gBAAgB,GAAG,IAAI,iBAAM,CAAC,IAAI,gBAAK,EAAE,CAAC,CAAC;YACjD,IAAM,MAAM,GAAG,IAAI,yBAAc,CAC7B,IAAI,yBAAc,EAAE,EAAE,IAAI,CAAC,SAAS,EAAE,gBAAgB,EAAE,IAAI,mCAAwB,EAAE,EACtF,UAAU,EACV,IAAI,EAAG,UAAU;YACjB,EAAE,CAAK,YAAY;aACtB,CAAC;YACF,IAAM,UAAU,GAAG,UAAU,CAAC,KAAK,CAAC,QAAQ,CAAC,MAAM,EAAE,QAAQ,EAAE;gBAC7D,sBAAsB,EAAE,IAAI;gBAC5B,mBAAmB,EAAE,IAAI;aAC1B,CAAC,CAAC;YACG,IAAA,KAA+B,IAAI,CAAC,6BAA6B,CAAC,WAAW,CAAC,EAA7E,UAAU,gBAAA,EAAE,KAAK,WAAA,EAAE,OAAO,aAAmD,CAAC;YACrF,IAAM,WAAW,GAAG,MAAM,CAAC,YAAY,CAAC,UAAU,EAAE,IAAI,CAAC,QAAQ,EAAE,UAAU,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC;YAC/F,IAAI,CAAC,WAAW,CAAC,WAAW,EAAE;gBAC5B,OAAO;aACR;YACD,OAAO;gBACL,OAAO,EAAE,UAAU,CAAC,SAAS;gBAC7B,WAAW,EAAE,WAAW,CAAC,WAAW;gBACpC,SAAS,EAAE,IAAI,CAAC,QAAQ;gBACxB,UAAU,YAAA;gBACV,KAAK,OAAA;gBACL,WAAW,EAAE,WAAW,CAAC,MAAM;gBAC/B,gBAAgB,kBAAA;gBAChB,QAAQ,UAAA;aACT,CAAC;QACJ,CAAC;QAED;;;;WAIG;QACH,mCAAG,GAAH,UAAI,GAAW;YACb,IAAI,IAAI,CAAC,QAAQ,CAAC,GAAG,EAAE;gBACrB,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;aACxB;QACH,CAAC;QAED;;;;WAIG;QACH,qCAAK,GAAL,UAAM,GAAW;YACf,IAAI,IAAI,CAAC,QAAQ,CAAC,KAAK,EAAE;gBACvB,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;aAC1B;QACH,CAAC;QAED;;;;WAIG;QACH,qCAAK,GAAL,UAAM,GAAW;YACf,IAAM,OAAO,GAAG,IAAI,CAAC,QAA8B,CAAC;YACpD,IAAI,CAAC,OAAO,CAAC,cAAc,EAAE;gBAC3B,4BAA4B;gBAC5B,OAAO;aACR;YACM,IAAA,MAAM,GAAI,OAAO,CAAC,cAAc,OAA1B,CAA2B;YACxC,IAAI,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,MAAM,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE;gBAChD,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;aAClB;QACH,CAAC;QACH,4BAAC;IAAD,CAAC,AAziBD,IAyiBC;IAziBY,sDAAqB;IA2iBlC,SAAS,yBAAyB,CAAC,OAA0B;;QAC3D,IAAI,MAAM,GAAsC,SAAS,CAAC;QAC1D,IAAI,UAAU,GAAG,CAAC,CAAC;;YACnB,KAAqB,IAAA,KAAA,iBAAA,OAAO,CAAC,SAAS,CAAA,gBAAA,4BAAE;gBAAnC,IAAM,QAAM,WAAA;gBACf,IAAM,UAAU,GAAG,QAAM,CAAC,gBAAgB,CAAC,UAAU,CAAC,MAAM,CAAC;gBAC7D,IAAI,UAAU,GAAG,UAAU,EAAE;oBAC3B,MAAM,GAAG,QAAM,CAAC;oBAChB,UAAU,GAAG,UAAU,CAAC;iBACzB;aACF;;;;;;;;;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,SAAS,MAAM,CAAC,IAAc;QAC5B,OAAO,EAAC,KAAK,EAAE,IAAI,CAAC,QAAQ,EAAE,EAAE,GAAG,EAAE,IAAI,CAAC,MAAM,EAAE,EAAC,CAAC;IACtD,CAAC;IAED,SAAS,MAAM,CAAC,UAA0B,EAAE,IAAY,EAAE,MAAc;QACtE,IAAI,IAAI,IAAI,IAAI,IAAI,MAAM,IAAI,IAAI,EAAE;YAClC,IAAM,UAAQ,GAAG,GAAG,CAAC,6BAA6B,CAAC,UAAU,EAAE,IAAI,EAAE,MAAM,CAAC,CAAC;YAC7E,IAAM,SAAS,GAAG,SAAS,SAAS,CAAC,IAAc;gBACjD,IAAI,IAAI,CAAC,IAAI,GAAG,GAAG,CAAC,UAAU,CAAC,SAAS,IAAI,IAAI,CAAC,GAAG,IAAI,UAAQ,IAAI,IAAI,CAAC,GAAG,GAAG,UAAQ,EAAE;oBACvF,IAAM,UAAU,GAAG,GAAG,CAAC,YAAY,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;oBACrD,OAAO,UAAU,IAAI,IAAI,CAAC;iBAC3B;YACH,CAAC,CAAC;YAEF,IAAM,IAAI,GAAG,GAAG,CAAC,YAAY,CAAC,UAAU,EAAE,SAAS,CAAC,CAAC;YACrD,IAAI,IAAI,EAAE;gBACR,OAAO,EAAC,KAAK,EAAE,IAAI,CAAC,QAAQ,EAAE,EAAE,GAAG,EAAE,IAAI,CAAC,MAAM,EAAE,EAAC,CAAC;aACrD;SACF;IACH,CAAC;IAED,SAAS,YAAY,CAAC,KAA4B;QAChD,OAAO,EAAC,OAAO,EAAE,KAAK,CAAC,OAAO,EAAE,IAAI,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC,CAAC,CAAC,SAAS,EAAC,CAAC;IAC/F,CAAC;IAED,SAAS,0BAA0B,CAAC,KAAqB,EAAE,IAAU;QACnE,OAAO,EAAC,OAAO,EAAE,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,YAAY,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,OAAO,EAAE,IAAI,MAAA,EAAC,CAAC;IAClF,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 {analyzeNgModules, AotSummaryResolver, CompileDirectiveSummary, CompileMetadataResolver, CompileNgModuleMetadata, CompilePipeSummary, CompilerConfig, DirectiveNormalizer, DirectiveResolver, DomElementSchemaRegistry, FormattedError, FormattedMessageChain, HtmlParser, isFormattedError, JitSummaryResolver, Lexer, NgAnalyzedModules, NgModuleResolver, Parser, ParseTreeResult, PipeResolver, ResourceLoader, StaticReflector, StaticSymbol, StaticSymbolCache, StaticSymbolResolver, TemplateParser, UrlResolver} from '@angular/compiler';\nimport {SchemaMetadata, ViewEncapsulation, ɵConsole as Console} from '@angular/core';\nimport * as path from 'path';\nimport * as tss from 'typescript/lib/tsserverlibrary';\n\nimport {ReflectorHost} from './reflector_host';\nimport {ExternalTemplate, InlineTemplate} from './template';\nimport {findTightestNode, getClassDeclFromDecoratorProp, getDirectiveClassLike, getPropertyAssignmentFromValue} from './ts_utils';\nimport {AstResult, Declaration, DeclarationError, DiagnosticMessageChain, LanguageServiceHost, Span, TemplateSource} from './types';\n\n/**\n * The language service never needs the normalized versions of the metadata. To avoid parsing\n * the content and resolving references, return an empty file. This also allows normalizing\n * template that are syntatically incorrect which is required to provide completions in\n * syntactically incorrect templates.\n */\nexport class DummyHtmlParser extends HtmlParser {\n  parse(): ParseTreeResult {\n    return new ParseTreeResult([], []);\n  }\n}\n\n/**\n * Avoid loading resources in the language servcie by using a dummy loader.\n */\nexport class DummyResourceLoader extends ResourceLoader {\n  get(_url: string): Promise<string> {\n    return Promise.resolve('');\n  }\n}\n\n/**\n * An implementation of a `LanguageServiceHost` for a TypeScript project.\n *\n * The `TypeScriptServiceHost` implements the Angular `LanguageServiceHost` using\n * the TypeScript language services.\n *\n * @publicApi\n */\nexport class TypeScriptServiceHost implements LanguageServiceHost {\n  private readonly summaryResolver: AotSummaryResolver;\n  private readonly reflectorHost: ReflectorHost;\n  private readonly staticSymbolResolver: StaticSymbolResolver;\n\n  private readonly staticSymbolCache = new StaticSymbolCache();\n  /**\n   * Key of the `fileToComponent` map must be TS internal normalized path (path\n   * separator must be `/`), value of the map is the StaticSymbol for the\n   * Component class declaration.\n   */\n  private readonly fileToComponent = new Map<ts.server.NormalizedPath, StaticSymbol>();\n  private readonly collectedErrors = new Map<string, any[]>();\n  private readonly fileVersions = new Map<string, string>();\n  private readonly urlResolver: UrlResolver;\n\n  private lastProgram: tss.Program|undefined = undefined;\n  private analyzedModules: NgAnalyzedModules = {\n    files: [],\n    ngModuleByPipeOrDirective: new Map(),\n    ngModules: [],\n  };\n\n  constructor(readonly tsLsHost: tss.LanguageServiceHost, readonly tsLS: tss.LanguageService) {\n    this.summaryResolver = new AotSummaryResolver(\n        {\n          loadSummary(_filePath: string) {\n            return null;\n          },\n          isSourceFile(_sourceFilePath: string) {\n            return true;\n          },\n          toSummaryFileName(sourceFilePath: string) {\n            return sourceFilePath;\n          },\n          fromSummaryFileName(filePath: string): string {\n            return filePath;\n          },\n        },\n        this.staticSymbolCache);\n    this.reflectorHost = new ReflectorHost(() => this.program, tsLsHost);\n    this.staticSymbolResolver = new StaticSymbolResolver(\n        this.reflectorHost, this.staticSymbolCache, this.summaryResolver,\n        (e, filePath) => this.collectError(e, filePath));\n    this.urlResolver = {\n      resolve: (baseUrl: string, url: string) => {\n        // In practice, `directoryExists` is always defined.\n        // https://github.com/microsoft/TypeScript/blob/0b6c9254a850dd07056259d4eefca7721745af75/src/server/project.ts#L1608-L1614\n        if (tsLsHost.directoryExists!(baseUrl)) {\n          return path.resolve(baseUrl, url);\n        }\n        return path.resolve(path.dirname(baseUrl), url);\n      }\n    };\n  }\n\n  // The resolver is instantiated lazily and should not be accessed directly.\n  // Instead, call the resolver getter. The instantiation of the resolver also\n  // requires instantiation of the StaticReflector, and the latter requires\n  // resolution of core Angular symbols. Module resolution should not be done\n  // during instantiation to avoid cyclic dependency between the plugin and the\n  // containing Project, so the Singleton pattern is used here.\n  private _resolver: CompileMetadataResolver|undefined;\n\n  /**\n   * Return the singleton instance of the MetadataResolver.\n   */\n  private get resolver(): CompileMetadataResolver {\n    if (this._resolver) {\n      return this._resolver;\n    }\n    // StaticReflector keeps its own private caches that are not clearable.\n    // We have no choice but to create a new instance to invalidate the caches.\n    // TODO: Revisit this when language service gets rewritten for Ivy.\n    const staticReflector = new StaticReflector(\n        this.summaryResolver, this.staticSymbolResolver,\n        [],  // knownMetadataClasses\n        [],  // knownMetadataFunctions\n        (e, filePath) => this.collectError(e, filePath));\n    // Because static reflector above is changed, we need to create a new\n    // resolver.\n    const moduleResolver = new NgModuleResolver(staticReflector);\n    const directiveResolver = new DirectiveResolver(staticReflector);\n    const pipeResolver = new PipeResolver(staticReflector);\n    const elementSchemaRegistry = new DomElementSchemaRegistry();\n    const resourceLoader = new DummyResourceLoader();\n    const htmlParser = new DummyHtmlParser();\n    // This tracks the CompileConfig in codegen.ts. Currently these options\n    // are hard-coded.\n    const config = new CompilerConfig({\n      defaultEncapsulation: ViewEncapsulation.Emulated,\n      useJit: false,\n    });\n    const directiveNormalizer =\n        new DirectiveNormalizer(resourceLoader, this.urlResolver, htmlParser, config);\n    this._resolver = new CompileMetadataResolver(\n        config, htmlParser, moduleResolver, directiveResolver, pipeResolver,\n        new JitSummaryResolver(), elementSchemaRegistry, directiveNormalizer, new Console(),\n        this.staticSymbolCache, staticReflector,\n        (error, type) => this.collectError(error, type && type.filePath));\n    return this._resolver;\n  }\n\n  /**\n   * Return the singleton instance of the StaticReflector hosted in the\n   * MetadataResolver.\n   */\n  private get reflector(): StaticReflector {\n    return this.resolver.getReflector() as StaticReflector;\n  }\n\n  /**\n   * Return all known external templates.\n   */\n  getExternalTemplates(): ts.server.NormalizedPath[] {\n    return [...this.fileToComponent.keys()];\n  }\n\n  /**\n   * Checks whether the program has changed and returns all analyzed modules.\n   * If program has changed, invalidate all caches and update fileToComponent\n   * and templateReferences.\n   * In addition to returning information about NgModules, this method plays the\n   * same role as 'synchronizeHostData' in tsserver.\n   */\n  getAnalyzedModules(): NgAnalyzedModules {\n    if (this.upToDate()) {\n      return this.analyzedModules;\n    }\n\n    // Invalidate caches\n    this.fileToComponent.clear();\n    this.collectedErrors.clear();\n    this.resolver.clearCache();\n\n    const analyzeHost = {\n      isSourceFile(_filePath: string) {\n        return true;\n      }\n    };\n    const programFiles = this.program.getSourceFiles().map(sf => sf.fileName);\n\n    try {\n      this.analyzedModules =\n          analyzeNgModules(programFiles, analyzeHost, this.staticSymbolResolver, this.resolver);\n    } catch (e) {\n      // Analyzing modules may throw; in that case, reuse the old modules.\n      this.error(`Analyzing NgModules failed. ${e}`);\n      return this.analyzedModules;\n    }\n\n    // update template references and fileToComponent\n    for (const ngModule of this.analyzedModules.ngModules) {\n      for (const directive of ngModule.declaredDirectives) {\n        const {metadata} = this.resolver.getNonNormalizedDirectiveMetadata(directive.reference)!;\n        if (metadata.isComponent && metadata.template && metadata.template.templateUrl) {\n          const templateName = this.urlResolver.resolve(\n              this.reflector.componentModuleUrl(directive.reference),\n              metadata.template.templateUrl);\n          this.fileToComponent.set(tss.server.toNormalizedPath(templateName), directive.reference);\n        }\n      }\n    }\n\n    return this.analyzedModules;\n  }\n\n  /**\n   * Checks whether the program has changed, and invalidate static symbols in\n   * the source files that have changed.\n   * Returns true if modules are up-to-date, false otherwise.\n   * This should only be called by getAnalyzedModules().\n   */\n  private upToDate(): boolean {\n    const {lastProgram, program} = this;\n    if (lastProgram === program) {\n      return true;\n    }\n    this.lastProgram = program;\n\n    // Even though the program has changed, it could be the case that none of\n    // the source files have changed. If all source files remain the same, then\n    // program is still up-to-date, and we should not invalidate caches.\n    let filesAdded = 0;\n    const filesChangedOrRemoved: string[] = [];\n\n    // Check if any source files have been added / changed since last computation.\n    const seen = new Set<string>();\n    const ANGULAR_CORE = '@angular/core';\n    const corePath = this.reflectorHost.moduleNameToFileName(ANGULAR_CORE);\n    for (const {fileName} of program.getSourceFiles()) {\n      // If `@angular/core` is edited, the language service would have to be\n      // restarted, so ignore changes to `@angular/core`.\n      // When the StaticReflector is initialized at startup, it loads core\n      // symbols from @angular/core by calling initializeConversionMap(). This\n      // is only done once. If the file is invalidated, some of the core symbols\n      // will be lost permanently.\n      if (fileName === corePath) {\n        continue;\n      }\n      seen.add(fileName);\n      const version = this.tsLsHost.getScriptVersion(fileName);\n      const lastVersion = this.fileVersions.get(fileName);\n      if (lastVersion === undefined) {\n        filesAdded++;\n        this.fileVersions.set(fileName, version);\n      } else if (version !== lastVersion) {\n        filesChangedOrRemoved.push(fileName);  // changed\n        this.fileVersions.set(fileName, version);\n      }\n    }\n\n    // Check if any source files have been removed since last computation.\n    for (const [fileName] of this.fileVersions) {\n      if (!seen.has(fileName)) {\n        filesChangedOrRemoved.push(fileName);  // removed\n        // Because Maps are iterated in insertion order, it is safe to delete\n        // entries from the same map while iterating.\n        // See https://stackoverflow.com/questions/35940216 and\n        // https://www.ecma-international.org/ecma-262/10.0/index.html#sec-map.prototype.foreach\n        this.fileVersions.delete(fileName);\n      }\n    }\n\n    for (const fileName of filesChangedOrRemoved) {\n      const symbols = this.staticSymbolResolver.invalidateFile(fileName);\n      this.reflector.invalidateSymbols(symbols);\n    }\n\n    // Program is up-to-date iff no files are added, changed, or removed.\n    return filesAdded === 0 && filesChangedOrRemoved.length === 0;\n  }\n\n  /**\n   * Find all templates in the specified `file`.\n   * @param fileName TS or HTML file\n   */\n  getTemplates(fileName: string): TemplateSource[] {\n    const results: TemplateSource[] = [];\n    if (fileName.endsWith('.ts')) {\n      // Find every template string in the file\n      const visit = (child: tss.Node) => {\n        const template = this.getInternalTemplate(child);\n        if (template) {\n          results.push(template);\n        } else {\n          tss.forEachChild(child, visit);\n        }\n      };\n      const sourceFile = this.getSourceFile(fileName);\n      if (sourceFile) {\n        tss.forEachChild(sourceFile, visit);\n      }\n    } else {\n      const template = this.getExternalTemplate(fileName);\n      if (template) {\n        results.push(template);\n      }\n    }\n    return results;\n  }\n\n  /**\n   * Return metadata about all class declarations in the file that are Angular\n   * directives. Potential matches are `@NgModule`, `@Component`, `@Directive`,\n   * `@Pipes`, etc. class declarations.\n   *\n   * @param fileName TS file\n   */\n  getDeclarations(fileName: string): Declaration[] {\n    if (!fileName.endsWith('.ts')) {\n      return [];\n    }\n    const sourceFile = this.getSourceFile(fileName);\n    if (!sourceFile) {\n      return [];\n    }\n    const results: Declaration[] = [];\n    const visit = (child: tss.Node) => {\n      const candidate = getDirectiveClassLike(child);\n      if (candidate) {\n        const {classId} = candidate;\n        const declarationSpan = spanOf(classId);\n        const className = classId.getText();\n        const classSymbol = this.reflector.getStaticSymbol(sourceFile.fileName, className);\n        // Ask the resolver to check if candidate is actually Angular directive\n        if (!this.resolver.isDirective(classSymbol)) {\n          return;\n        }\n        const data = this.resolver.getNonNormalizedDirectiveMetadata(classSymbol);\n        if (!data) {\n          return;\n        }\n        results.push({\n          type: classSymbol,\n          declarationSpan,\n          metadata: data.metadata,\n          errors: this.getCollectedErrors(declarationSpan, sourceFile),\n        });\n      } else {\n        child.forEachChild(visit);\n      }\n    };\n    tss.forEachChild(sourceFile, visit);\n\n    return results;\n  }\n\n  getSourceFile(fileName: string): tss.SourceFile|undefined {\n    if (!fileName.endsWith('.ts')) {\n      throw new Error(`Non-TS source file requested: ${fileName}`);\n    }\n    return this.program.getSourceFile(fileName);\n  }\n\n  get program(): tss.Program {\n    const program = this.tsLS.getProgram();\n    if (!program) {\n      // Program is very very unlikely to be undefined.\n      throw new Error('No program in language service!');\n    }\n    return program;\n  }\n\n  /**\n   * Return the TemplateSource if `node` is a template node.\n   *\n   * For example,\n   *\n   * @Component({\n   *   template: '<div></div>' <-- template node\n   * })\n   * class AppComponent {}\n   *           ^---- class declaration node\n   *\n   * @param node Potential template node\n   */\n  private getInternalTemplate(node: tss.Node): TemplateSource|undefined {\n    if (!tss.isStringLiteralLike(node)) {\n      return;\n    }\n    const tmplAsgn = getPropertyAssignmentFromValue(node, 'template');\n    if (!tmplAsgn) {\n      return;\n    }\n    const classDecl = getClassDeclFromDecoratorProp(tmplAsgn);\n    if (!classDecl || !classDecl.name) {  // Does not handle anonymous class\n      return;\n    }\n    const fileName = node.getSourceFile().fileName;\n    const classSymbol = this.reflector.getStaticSymbol(fileName, classDecl.name.text);\n    return new InlineTemplate(node, classDecl, classSymbol, this);\n  }\n\n  /**\n   * Return the external template for `fileName`.\n   * @param fileName HTML file\n   */\n  private getExternalTemplate(fileName: string): TemplateSource|undefined {\n    // First get the text for the template\n    const snapshot = this.tsLsHost.getScriptSnapshot(fileName);\n    if (!snapshot) {\n      return;\n    }\n    const source = snapshot.getText(0, snapshot.getLength());\n    // Next find the component class symbol\n    const classSymbol = this.fileToComponent.get(tss.server.toNormalizedPath(fileName));\n    if (!classSymbol) {\n      return;\n    }\n    // Then use the class symbol to find the actual ts.ClassDeclaration node\n    const sourceFile = this.getSourceFile(classSymbol.filePath);\n    if (!sourceFile) {\n      return;\n    }\n    // TODO: This only considers top-level class declarations in a source file.\n    // This would not find a class declaration in a namespace, for example.\n    const classDecl = sourceFile.forEachChild((child) => {\n      if (tss.isClassDeclaration(child) && child.name && child.name.text === classSymbol.name) {\n        return child;\n      }\n    });\n    if (!classDecl) {\n      return;\n    }\n    return new ExternalTemplate(source, fileName, classDecl, classSymbol, this);\n  }\n\n  private collectError(error: any, filePath?: string) {\n    if (filePath) {\n      let errors = this.collectedErrors.get(filePath);\n      if (!errors) {\n        errors = [];\n        this.collectedErrors.set(filePath, errors);\n      }\n      errors.push(error);\n    }\n  }\n\n  private getCollectedErrors(defaultSpan: Span, sourceFile: tss.SourceFile): DeclarationError[] {\n    const errors = this.collectedErrors.get(sourceFile.fileName);\n    if (!errors) {\n      return [];\n    }\n    // TODO: Add better typings for the errors\n    return errors.map((e: any) => {\n      const line = e.line || (e.position && e.position.line);\n      const column = e.column || (e.position && e.position.column);\n      const span = spanAt(sourceFile, line, column) || defaultSpan;\n      if (isFormattedError(e)) {\n        return errorToDiagnosticWithChain(e, span);\n      }\n      return {message: e.message, span};\n    });\n  }\n\n  /**\n   * Return the parsed template for the template at the specified `position`.\n   * @param fileName TS or HTML file\n   * @param position Position of the template in the TS file, otherwise ignored.\n   */\n  getTemplateAstAtPosition(fileName: string, position: number): AstResult|undefined {\n    let template: TemplateSource|undefined;\n    if (fileName.endsWith('.ts')) {\n      const sourceFile = this.getSourceFile(fileName);\n      if (!sourceFile) {\n        return;\n      }\n      // Find the node that most closely matches the position\n      const node = findTightestNode(sourceFile, position);\n      if (!node) {\n        return;\n      }\n      template = this.getInternalTemplate(node);\n    } else {\n      template = this.getExternalTemplate(fileName);\n    }\n    if (!template) {\n      return;\n    }\n    return this.getTemplateAst(template);\n  }\n\n  /**\n   * Find the NgModule which the directive associated with the `classSymbol`\n   * belongs to, then return its schema and transitive directives and pipes.\n   * @param classSymbol Angular Symbol that defines a directive\n   */\n  private getModuleMetadataForDirective(classSymbol: StaticSymbol) {\n    const result = {\n      directives: [] as CompileDirectiveSummary[],\n      pipes: [] as CompilePipeSummary[],\n      schemas: [] as SchemaMetadata[],\n    };\n    // First find which NgModule the directive belongs to.\n    const ngModule = this.analyzedModules.ngModuleByPipeOrDirective.get(classSymbol) ||\n        findSuitableDefaultModule(this.analyzedModules);\n    if (!ngModule) {\n      return result;\n    }\n    // Then gather all transitive directives and pipes.\n    const {directives, pipes} = ngModule.transitiveModule;\n    for (const directive of directives) {\n      const data = this.resolver.getNonNormalizedDirectiveMetadata(directive.reference);\n      if (data) {\n        result.directives.push(data.metadata.toSummary());\n      }\n    }\n    for (const pipe of pipes) {\n      const metadata = this.resolver.getOrLoadPipeMetadata(pipe.reference);\n      result.pipes.push(metadata.toSummary());\n    }\n    result.schemas.push(...ngModule.schemas);\n    return result;\n  }\n\n  /**\n   * Parse the `template` and return its AST, if any.\n   * @param template template to be parsed\n   */\n  getTemplateAst(template: TemplateSource): AstResult|undefined {\n    const {type: classSymbol, fileName} = template;\n    const data = this.resolver.getNonNormalizedDirectiveMetadata(classSymbol);\n    if (!data) {\n      return;\n    }\n    const htmlParser = new HtmlParser();\n    const expressionParser = new Parser(new Lexer());\n    const parser = new TemplateParser(\n        new CompilerConfig(), this.reflector, expressionParser, new DomElementSchemaRegistry(),\n        htmlParser,\n        null,  // console\n        []     // tranforms\n    );\n    const htmlResult = htmlParser.parse(template.source, fileName, {\n      tokenizeExpansionForms: true,\n      preserveLineEndings: true,  // do not convert CRLF to LF\n    });\n    const {directives, pipes, schemas} = this.getModuleMetadataForDirective(classSymbol);\n    const parseResult = parser.tryParseHtml(htmlResult, data.metadata, directives, pipes, schemas);\n    if (!parseResult.templateAst) {\n      return;\n    }\n    return {\n      htmlAst: htmlResult.rootNodes,\n      templateAst: parseResult.templateAst,\n      directive: data.metadata,\n      directives,\n      pipes,\n      parseErrors: parseResult.errors,\n      expressionParser,\n      template,\n    };\n  }\n\n  /**\n   * Log the specified `msg` to file at INFO level. If logging is not enabled\n   * this method is a no-op.\n   * @param msg Log message\n   */\n  log(msg: string) {\n    if (this.tsLsHost.log) {\n      this.tsLsHost.log(msg);\n    }\n  }\n\n  /**\n   * Log the specified `msg` to file at ERROR level. If logging is not enabled\n   * this method is a no-op.\n   * @param msg error message\n   */\n  error(msg: string) {\n    if (this.tsLsHost.error) {\n      this.tsLsHost.error(msg);\n    }\n  }\n\n  /**\n   * Log debugging info to file at INFO level, only if verbose setting is turned\n   * on. Otherwise, this method is a no-op.\n   * @param msg debugging message\n   */\n  debug(msg: string) {\n    const project = this.tsLsHost as tss.server.Project;\n    if (!project.projectService) {\n      // tsLsHost is not a Project\n      return;\n    }\n    const {logger} = project.projectService;\n    if (logger.hasLevel(tss.server.LogLevel.verbose)) {\n      logger.info(msg);\n    }\n  }\n}\n\nfunction findSuitableDefaultModule(modules: NgAnalyzedModules): CompileNgModuleMetadata|undefined {\n  let result: CompileNgModuleMetadata|undefined = undefined;\n  let resultSize = 0;\n  for (const module of modules.ngModules) {\n    const moduleSize = module.transitiveModule.directives.length;\n    if (moduleSize > resultSize) {\n      result = module;\n      resultSize = moduleSize;\n    }\n  }\n  return result;\n}\n\nfunction spanOf(node: tss.Node): Span {\n  return {start: node.getStart(), end: node.getEnd()};\n}\n\nfunction spanAt(sourceFile: tss.SourceFile, line: number, column: number): Span|undefined {\n  if (line != null && column != null) {\n    const position = tss.getPositionOfLineAndCharacter(sourceFile, line, column);\n    const findChild = function findChild(node: tss.Node): tss.Node|undefined {\n      if (node.kind > tss.SyntaxKind.LastToken && node.pos <= position && node.end > position) {\n        const betterNode = tss.forEachChild(node, findChild);\n        return betterNode || node;\n      }\n    };\n\n    const node = tss.forEachChild(sourceFile, findChild);\n    if (node) {\n      return {start: node.getStart(), end: node.getEnd()};\n    }\n  }\n}\n\nfunction convertChain(chain: FormattedMessageChain): DiagnosticMessageChain {\n  return {message: chain.message, next: chain.next ? chain.next.map(convertChain) : undefined};\n}\n\nfunction errorToDiagnosticWithChain(error: FormattedError, span: Span): DeclarationError {\n  return {message: error.chain ? convertChain(error.chain) : error.message, span};\n}\n"]}
\No newline at end of file