UNPKG

87.4 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/language_service", "@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 = exports.createLanguageServiceFromTypescript = 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 language_service_1 = require("@angular/language-service/src/language_service");
26 var reflector_host_1 = require("@angular/language-service/src/reflector_host");
27 var template_1 = require("@angular/language-service/src/template");
28 var ts_utils_1 = require("@angular/language-service/src/ts_utils");
29 /**
30 * Create a `LanguageServiceHost`
31 */
32 function createLanguageServiceFromTypescript(host, service) {
33 var ngHost = new TypeScriptServiceHost(host, service);
34 var ngServer = language_service_1.createLanguageService(ngHost);
35 return ngServer;
36 }
37 exports.createLanguageServiceFromTypescript = createLanguageServiceFromTypescript;
38 /**
39 * The language service never needs the normalized versions of the metadata. To avoid parsing
40 * the content and resolving references, return an empty file. This also allows normalizing
41 * template that are syntatically incorrect which is required to provide completions in
42 * syntactically incorrect templates.
43 */
44 var DummyHtmlParser = /** @class */ (function (_super) {
45 tslib_1.__extends(DummyHtmlParser, _super);
46 function DummyHtmlParser() {
47 return _super !== null && _super.apply(this, arguments) || this;
48 }
49 DummyHtmlParser.prototype.parse = function () {
50 return new compiler_1.ParseTreeResult([], []);
51 };
52 return DummyHtmlParser;
53 }(compiler_1.HtmlParser));
54 exports.DummyHtmlParser = DummyHtmlParser;
55 /**
56 * Avoid loading resources in the language servcie by using a dummy loader.
57 */
58 var DummyResourceLoader = /** @class */ (function (_super) {
59 tslib_1.__extends(DummyResourceLoader, _super);
60 function DummyResourceLoader() {
61 return _super !== null && _super.apply(this, arguments) || this;
62 }
63 DummyResourceLoader.prototype.get = function (_url) {
64 return Promise.resolve('');
65 };
66 return DummyResourceLoader;
67 }(compiler_1.ResourceLoader));
68 exports.DummyResourceLoader = DummyResourceLoader;
69 /**
70 * An implementation of a `LanguageServiceHost` for a TypeScript project.
71 *
72 * The `TypeScriptServiceHost` implements the Angular `LanguageServiceHost` using
73 * the TypeScript language services.
74 *
75 * @publicApi
76 */
77 var TypeScriptServiceHost = /** @class */ (function () {
78 function TypeScriptServiceHost(tsLsHost, tsLS) {
79 var _this = this;
80 this.tsLsHost = tsLsHost;
81 this.tsLS = tsLS;
82 this.staticSymbolCache = new compiler_1.StaticSymbolCache();
83 this.fileToComponent = new Map();
84 this.collectedErrors = new Map();
85 this.fileVersions = new Map();
86 this.lastProgram = undefined;
87 this.analyzedModules = {
88 files: [],
89 ngModuleByPipeOrDirective: new Map(),
90 ngModules: [],
91 };
92 this.summaryResolver = new compiler_1.AotSummaryResolver({
93 loadSummary: function (_filePath) {
94 return null;
95 },
96 isSourceFile: function (_sourceFilePath) {
97 return true;
98 },
99 toSummaryFileName: function (sourceFilePath) {
100 return sourceFilePath;
101 },
102 fromSummaryFileName: function (filePath) {
103 return filePath;
104 },
105 }, this.staticSymbolCache);
106 this.reflectorHost = new reflector_host_1.ReflectorHost(function () { return _this.program; }, tsLsHost);
107 this.staticSymbolResolver = new compiler_1.StaticSymbolResolver(this.reflectorHost, this.staticSymbolCache, this.summaryResolver, function (e, filePath) { return _this.collectError(e, filePath); });
108 this.urlResolver = {
109 resolve: function (baseUrl, url) {
110 // In practice, `directoryExists` is always defined.
111 // https://github.com/microsoft/TypeScript/blob/0b6c9254a850dd07056259d4eefca7721745af75/src/server/project.ts#L1608-L1614
112 if (tsLsHost.directoryExists(baseUrl)) {
113 return path.resolve(baseUrl, url);
114 }
115 return path.resolve(path.dirname(baseUrl), url);
116 }
117 };
118 }
119 Object.defineProperty(TypeScriptServiceHost.prototype, "resolver", {
120 /**
121 * Return the singleton instance of the MetadataResolver.
122 */
123 get: function () {
124 var _this = this;
125 if (this._resolver) {
126 return this._resolver;
127 }
128 // StaticReflector keeps its own private caches that are not clearable.
129 // We have no choice but to create a new instance to invalidate the caches.
130 // TODO: Revisit this when language service gets rewritten for Ivy.
131 var staticReflector = new compiler_1.StaticReflector(this.summaryResolver, this.staticSymbolResolver, [], // knownMetadataClasses
132 [], // knownMetadataFunctions
133 function (e, filePath) { return _this.collectError(e, filePath); });
134 // Because static reflector above is changed, we need to create a new
135 // resolver.
136 var moduleResolver = new compiler_1.NgModuleResolver(staticReflector);
137 var directiveResolver = new compiler_1.DirectiveResolver(staticReflector);
138 var pipeResolver = new compiler_1.PipeResolver(staticReflector);
139 var elementSchemaRegistry = new compiler_1.DomElementSchemaRegistry();
140 var resourceLoader = new DummyResourceLoader();
141 var htmlParser = new DummyHtmlParser();
142 // This tracks the CompileConfig in codegen.ts. Currently these options
143 // are hard-coded.
144 var config = new compiler_1.CompilerConfig({
145 defaultEncapsulation: core_1.ViewEncapsulation.Emulated,
146 useJit: false,
147 });
148 var directiveNormalizer = new compiler_1.DirectiveNormalizer(resourceLoader, this.urlResolver, htmlParser, config);
149 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); });
150 return this._resolver;
151 },
152 enumerable: false,
153 configurable: true
154 });
155 Object.defineProperty(TypeScriptServiceHost.prototype, "reflector", {
156 /**
157 * Return the singleton instance of the StaticReflector hosted in the
158 * MetadataResolver.
159 */
160 get: function () {
161 return this.resolver.getReflector();
162 },
163 enumerable: false,
164 configurable: true
165 });
166 /**
167 * Return all known external templates.
168 */
169 TypeScriptServiceHost.prototype.getExternalTemplates = function () {
170 return tslib_1.__spread(this.fileToComponent.keys());
171 };
172 /**
173 * Checks whether the program has changed and returns all analyzed modules.
174 * If program has changed, invalidate all caches and update fileToComponent
175 * and templateReferences.
176 * In addition to returning information about NgModules, this method plays the
177 * same role as 'synchronizeHostData' in tsserver.
178 */
179 TypeScriptServiceHost.prototype.getAnalyzedModules = function () {
180 var e_1, _a, e_2, _b;
181 if (this.upToDate()) {
182 return this.analyzedModules;
183 }
184 // Invalidate caches
185 this.fileToComponent.clear();
186 this.collectedErrors.clear();
187 this.resolver.clearCache();
188 var analyzeHost = {
189 isSourceFile: function (_filePath) {
190 return true;
191 }
192 };
193 var programFiles = this.program.getSourceFiles().map(function (sf) { return sf.fileName; });
194 try {
195 this.analyzedModules =
196 compiler_1.analyzeNgModules(programFiles, analyzeHost, this.staticSymbolResolver, this.resolver);
197 }
198 catch (e) {
199 // Analyzing modules may throw; in that case, reuse the old modules.
200 this.error("Analyzing NgModules failed. " + e);
201 return this.analyzedModules;
202 }
203 try {
204 // update template references and fileToComponent
205 for (var _c = tslib_1.__values(this.analyzedModules.ngModules), _d = _c.next(); !_d.done; _d = _c.next()) {
206 var ngModule = _d.value;
207 try {
208 for (var _e = (e_2 = void 0, tslib_1.__values(ngModule.declaredDirectives)), _f = _e.next(); !_f.done; _f = _e.next()) {
209 var directive = _f.value;
210 var metadata = this.resolver.getNonNormalizedDirectiveMetadata(directive.reference).metadata;
211 if (metadata.isComponent && metadata.template && metadata.template.templateUrl) {
212 var templateName = this.urlResolver.resolve(this.reflector.componentModuleUrl(directive.reference), metadata.template.templateUrl);
213 this.fileToComponent.set(templateName, directive.reference);
214 }
215 }
216 }
217 catch (e_2_1) { e_2 = { error: e_2_1 }; }
218 finally {
219 try {
220 if (_f && !_f.done && (_b = _e.return)) _b.call(_e);
221 }
222 finally { if (e_2) throw e_2.error; }
223 }
224 }
225 }
226 catch (e_1_1) { e_1 = { error: e_1_1 }; }
227 finally {
228 try {
229 if (_d && !_d.done && (_a = _c.return)) _a.call(_c);
230 }
231 finally { if (e_1) throw e_1.error; }
232 }
233 return this.analyzedModules;
234 };
235 /**
236 * Checks whether the program has changed, and invalidate static symbols in
237 * the source files that have changed.
238 * Returns true if modules are up-to-date, false otherwise.
239 * This should only be called by getAnalyzedModules().
240 */
241 TypeScriptServiceHost.prototype.upToDate = function () {
242 var e_3, _a, e_4, _b, e_5, _c;
243 var _d = this, lastProgram = _d.lastProgram, program = _d.program;
244 if (lastProgram === program) {
245 return true;
246 }
247 this.lastProgram = program;
248 // Even though the program has changed, it could be the case that none of
249 // the source files have changed. If all source files remain the same, then
250 // program is still up-to-date, and we should not invalidate caches.
251 var filesAdded = 0;
252 var filesChangedOrRemoved = [];
253 // Check if any source files have been added / changed since last computation.
254 var seen = new Set();
255 var ANGULAR_CORE = '@angular/core';
256 var corePath = this.reflectorHost.moduleNameToFileName(ANGULAR_CORE);
257 try {
258 for (var _e = tslib_1.__values(program.getSourceFiles()), _f = _e.next(); !_f.done; _f = _e.next()) {
259 var fileName = _f.value.fileName;
260 // If `@angular/core` is edited, the language service would have to be
261 // restarted, so ignore changes to `@angular/core`.
262 // When the StaticReflector is initialized at startup, it loads core
263 // symbols from @angular/core by calling initializeConversionMap(). This
264 // is only done once. If the file is invalidated, some of the core symbols
265 // will be lost permanently.
266 if (fileName === corePath) {
267 continue;
268 }
269 seen.add(fileName);
270 var version = this.tsLsHost.getScriptVersion(fileName);
271 var lastVersion = this.fileVersions.get(fileName);
272 if (lastVersion === undefined) {
273 filesAdded++;
274 this.fileVersions.set(fileName, version);
275 }
276 else if (version !== lastVersion) {
277 filesChangedOrRemoved.push(fileName); // changed
278 this.fileVersions.set(fileName, version);
279 }
280 }
281 }
282 catch (e_3_1) { e_3 = { error: e_3_1 }; }
283 finally {
284 try {
285 if (_f && !_f.done && (_a = _e.return)) _a.call(_e);
286 }
287 finally { if (e_3) throw e_3.error; }
288 }
289 try {
290 // Check if any source files have been removed since last computation.
291 for (var _g = tslib_1.__values(this.fileVersions), _h = _g.next(); !_h.done; _h = _g.next()) {
292 var _j = tslib_1.__read(_h.value, 1), fileName = _j[0];
293 if (!seen.has(fileName)) {
294 filesChangedOrRemoved.push(fileName); // removed
295 // Because Maps are iterated in insertion order, it is safe to delete
296 // entries from the same map while iterating.
297 // See https://stackoverflow.com/questions/35940216 and
298 // https://www.ecma-international.org/ecma-262/10.0/index.html#sec-map.prototype.foreach
299 this.fileVersions.delete(fileName);
300 }
301 }
302 }
303 catch (e_4_1) { e_4 = { error: e_4_1 }; }
304 finally {
305 try {
306 if (_h && !_h.done && (_b = _g.return)) _b.call(_g);
307 }
308 finally { if (e_4) throw e_4.error; }
309 }
310 try {
311 for (var filesChangedOrRemoved_1 = tslib_1.__values(filesChangedOrRemoved), filesChangedOrRemoved_1_1 = filesChangedOrRemoved_1.next(); !filesChangedOrRemoved_1_1.done; filesChangedOrRemoved_1_1 = filesChangedOrRemoved_1.next()) {
312 var fileName = filesChangedOrRemoved_1_1.value;
313 var symbols = this.staticSymbolResolver.invalidateFile(fileName);
314 this.reflector.invalidateSymbols(symbols);
315 }
316 }
317 catch (e_5_1) { e_5 = { error: e_5_1 }; }
318 finally {
319 try {
320 if (filesChangedOrRemoved_1_1 && !filesChangedOrRemoved_1_1.done && (_c = filesChangedOrRemoved_1.return)) _c.call(filesChangedOrRemoved_1);
321 }
322 finally { if (e_5) throw e_5.error; }
323 }
324 // Program is up-to-date iff no files are added, changed, or removed.
325 return filesAdded === 0 && filesChangedOrRemoved.length === 0;
326 };
327 /**
328 * Find all templates in the specified `file`.
329 * @param fileName TS or HTML file
330 */
331 TypeScriptServiceHost.prototype.getTemplates = function (fileName) {
332 var _this = this;
333 var results = [];
334 if (fileName.endsWith('.ts')) {
335 // Find every template string in the file
336 var visit_1 = function (child) {
337 var template = _this.getInternalTemplate(child);
338 if (template) {
339 results.push(template);
340 }
341 else {
342 tss.forEachChild(child, visit_1);
343 }
344 };
345 var sourceFile = this.getSourceFile(fileName);
346 if (sourceFile) {
347 tss.forEachChild(sourceFile, visit_1);
348 }
349 }
350 else {
351 var template = this.getExternalTemplate(fileName);
352 if (template) {
353 results.push(template);
354 }
355 }
356 return results;
357 };
358 /**
359 * Return metadata about all class declarations in the file that are Angular
360 * directives. Potential matches are `@NgModule`, `@Component`, `@Directive`,
361 * `@Pipes`, etc. class declarations.
362 *
363 * @param fileName TS file
364 */
365 TypeScriptServiceHost.prototype.getDeclarations = function (fileName) {
366 var _this = this;
367 if (!fileName.endsWith('.ts')) {
368 return [];
369 }
370 var sourceFile = this.getSourceFile(fileName);
371 if (!sourceFile) {
372 return [];
373 }
374 var results = [];
375 var visit = function (child) {
376 var candidate = ts_utils_1.getDirectiveClassLike(child);
377 if (candidate) {
378 var classId = candidate.classId;
379 var declarationSpan = spanOf(classId);
380 var className = classId.getText();
381 var classSymbol = _this.reflector.getStaticSymbol(sourceFile.fileName, className);
382 // Ask the resolver to check if candidate is actually Angular directive
383 if (!_this.resolver.isDirective(classSymbol)) {
384 return;
385 }
386 var data = _this.resolver.getNonNormalizedDirectiveMetadata(classSymbol);
387 if (!data) {
388 return;
389 }
390 results.push({
391 type: classSymbol,
392 declarationSpan: declarationSpan,
393 metadata: data.metadata,
394 errors: _this.getCollectedErrors(declarationSpan, sourceFile),
395 });
396 }
397 else {
398 child.forEachChild(visit);
399 }
400 };
401 tss.forEachChild(sourceFile, visit);
402 return results;
403 };
404 TypeScriptServiceHost.prototype.getSourceFile = function (fileName) {
405 if (!fileName.endsWith('.ts')) {
406 throw new Error("Non-TS source file requested: " + fileName);
407 }
408 return this.program.getSourceFile(fileName);
409 };
410 Object.defineProperty(TypeScriptServiceHost.prototype, "program", {
411 get: function () {
412 var program = this.tsLS.getProgram();
413 if (!program) {
414 // Program is very very unlikely to be undefined.
415 throw new Error('No program in language service!');
416 }
417 return program;
418 },
419 enumerable: false,
420 configurable: true
421 });
422 /**
423 * Return the TemplateSource if `node` is a template node.
424 *
425 * For example,
426 *
427 * @Component({
428 * template: '<div></div>' <-- template node
429 * })
430 * class AppComponent {}
431 * ^---- class declaration node
432 *
433 * @param node Potential template node
434 */
435 TypeScriptServiceHost.prototype.getInternalTemplate = function (node) {
436 if (!tss.isStringLiteralLike(node)) {
437 return;
438 }
439 var tmplAsgn = ts_utils_1.getPropertyAssignmentFromValue(node, 'template');
440 if (!tmplAsgn) {
441 return;
442 }
443 var classDecl = ts_utils_1.getClassDeclFromDecoratorProp(tmplAsgn);
444 if (!classDecl || !classDecl.name) { // Does not handle anonymous class
445 return;
446 }
447 var fileName = node.getSourceFile().fileName;
448 var classSymbol = this.reflector.getStaticSymbol(fileName, classDecl.name.text);
449 return new template_1.InlineTemplate(node, classDecl, classSymbol, this);
450 };
451 /**
452 * Return the external template for `fileName`.
453 * @param fileName HTML file
454 */
455 TypeScriptServiceHost.prototype.getExternalTemplate = function (fileName) {
456 // First get the text for the template
457 var snapshot = this.tsLsHost.getScriptSnapshot(fileName);
458 if (!snapshot) {
459 return;
460 }
461 var source = snapshot.getText(0, snapshot.getLength());
462 // Next find the component class symbol
463 var classSymbol = this.fileToComponent.get(fileName);
464 if (!classSymbol) {
465 return;
466 }
467 // Then use the class symbol to find the actual ts.ClassDeclaration node
468 var sourceFile = this.getSourceFile(classSymbol.filePath);
469 if (!sourceFile) {
470 return;
471 }
472 // TODO: This only considers top-level class declarations in a source file.
473 // This would not find a class declaration in a namespace, for example.
474 var classDecl = sourceFile.forEachChild(function (child) {
475 if (tss.isClassDeclaration(child) && child.name && child.name.text === classSymbol.name) {
476 return child;
477 }
478 });
479 if (!classDecl) {
480 return;
481 }
482 return new template_1.ExternalTemplate(source, fileName, classDecl, classSymbol, this);
483 };
484 TypeScriptServiceHost.prototype.collectError = function (error, filePath) {
485 if (filePath) {
486 var errors = this.collectedErrors.get(filePath);
487 if (!errors) {
488 errors = [];
489 this.collectedErrors.set(filePath, errors);
490 }
491 errors.push(error);
492 }
493 };
494 TypeScriptServiceHost.prototype.getCollectedErrors = function (defaultSpan, sourceFile) {
495 var errors = this.collectedErrors.get(sourceFile.fileName);
496 if (!errors) {
497 return [];
498 }
499 // TODO: Add better typings for the errors
500 return errors.map(function (e) {
501 var line = e.line || (e.position && e.position.line);
502 var column = e.column || (e.position && e.position.column);
503 var span = spanAt(sourceFile, line, column) || defaultSpan;
504 if (compiler_1.isFormattedError(e)) {
505 return errorToDiagnosticWithChain(e, span);
506 }
507 return { message: e.message, span: span };
508 });
509 };
510 /**
511 * Return the parsed template for the template at the specified `position`.
512 * @param fileName TS or HTML file
513 * @param position Position of the template in the TS file, otherwise ignored.
514 */
515 TypeScriptServiceHost.prototype.getTemplateAstAtPosition = function (fileName, position) {
516 var template;
517 if (fileName.endsWith('.ts')) {
518 var sourceFile = this.getSourceFile(fileName);
519 if (!sourceFile) {
520 return;
521 }
522 // Find the node that most closely matches the position
523 var node = ts_utils_1.findTightestNode(sourceFile, position);
524 if (!node) {
525 return;
526 }
527 template = this.getInternalTemplate(node);
528 }
529 else {
530 template = this.getExternalTemplate(fileName);
531 }
532 if (!template) {
533 return;
534 }
535 return this.getTemplateAst(template);
536 };
537 /**
538 * Find the NgModule which the directive associated with the `classSymbol`
539 * belongs to, then return its schema and transitive directives and pipes.
540 * @param classSymbol Angular Symbol that defines a directive
541 */
542 TypeScriptServiceHost.prototype.getModuleMetadataForDirective = function (classSymbol) {
543 var e_6, _a, e_7, _b, _c;
544 var result = {
545 directives: [],
546 pipes: [],
547 schemas: [],
548 };
549 // First find which NgModule the directive belongs to.
550 var ngModule = this.analyzedModules.ngModuleByPipeOrDirective.get(classSymbol) ||
551 findSuitableDefaultModule(this.analyzedModules);
552 if (!ngModule) {
553 return result;
554 }
555 // Then gather all transitive directives and pipes.
556 var _d = ngModule.transitiveModule, directives = _d.directives, pipes = _d.pipes;
557 try {
558 for (var directives_1 = tslib_1.__values(directives), directives_1_1 = directives_1.next(); !directives_1_1.done; directives_1_1 = directives_1.next()) {
559 var directive = directives_1_1.value;
560 var data = this.resolver.getNonNormalizedDirectiveMetadata(directive.reference);
561 if (data) {
562 result.directives.push(data.metadata.toSummary());
563 }
564 }
565 }
566 catch (e_6_1) { e_6 = { error: e_6_1 }; }
567 finally {
568 try {
569 if (directives_1_1 && !directives_1_1.done && (_a = directives_1.return)) _a.call(directives_1);
570 }
571 finally { if (e_6) throw e_6.error; }
572 }
573 try {
574 for (var pipes_1 = tslib_1.__values(pipes), pipes_1_1 = pipes_1.next(); !pipes_1_1.done; pipes_1_1 = pipes_1.next()) {
575 var pipe = pipes_1_1.value;
576 var metadata = this.resolver.getOrLoadPipeMetadata(pipe.reference);
577 result.pipes.push(metadata.toSummary());
578 }
579 }
580 catch (e_7_1) { e_7 = { error: e_7_1 }; }
581 finally {
582 try {
583 if (pipes_1_1 && !pipes_1_1.done && (_b = pipes_1.return)) _b.call(pipes_1);
584 }
585 finally { if (e_7) throw e_7.error; }
586 }
587 (_c = result.schemas).push.apply(_c, tslib_1.__spread(ngModule.schemas));
588 return result;
589 };
590 /**
591 * Parse the `template` and return its AST, if any.
592 * @param template template to be parsed
593 */
594 TypeScriptServiceHost.prototype.getTemplateAst = function (template) {
595 var classSymbol = template.type, fileName = template.fileName;
596 var data = this.resolver.getNonNormalizedDirectiveMetadata(classSymbol);
597 if (!data) {
598 return;
599 }
600 var htmlParser = new compiler_1.HtmlParser();
601 var expressionParser = new compiler_1.Parser(new compiler_1.Lexer());
602 var parser = new compiler_1.TemplateParser(new compiler_1.CompilerConfig(), this.reflector, expressionParser, new compiler_1.DomElementSchemaRegistry(), htmlParser, null, // console
603 [] // tranforms
604 );
605 var htmlResult = htmlParser.parse(template.source, fileName, {
606 tokenizeExpansionForms: true,
607 preserveLineEndings: true,
608 });
609 var _a = this.getModuleMetadataForDirective(classSymbol), directives = _a.directives, pipes = _a.pipes, schemas = _a.schemas;
610 var parseResult = parser.tryParseHtml(htmlResult, data.metadata, directives, pipes, schemas);
611 if (!parseResult.templateAst) {
612 return;
613 }
614 return {
615 htmlAst: htmlResult.rootNodes,
616 templateAst: parseResult.templateAst,
617 directive: data.metadata,
618 directives: directives,
619 pipes: pipes,
620 parseErrors: parseResult.errors,
621 expressionParser: expressionParser,
622 template: template,
623 };
624 };
625 /**
626 * Log the specified `msg` to file at INFO level. If logging is not enabled
627 * this method is a no-op.
628 * @param msg Log message
629 */
630 TypeScriptServiceHost.prototype.log = function (msg) {
631 if (this.tsLsHost.log) {
632 this.tsLsHost.log(msg);
633 }
634 };
635 /**
636 * Log the specified `msg` to file at ERROR level. If logging is not enabled
637 * this method is a no-op.
638 * @param msg error message
639 */
640 TypeScriptServiceHost.prototype.error = function (msg) {
641 if (this.tsLsHost.error) {
642 this.tsLsHost.error(msg);
643 }
644 };
645 /**
646 * Log debugging info to file at INFO level, only if verbose setting is turned
647 * on. Otherwise, this method is a no-op.
648 * @param msg debugging message
649 */
650 TypeScriptServiceHost.prototype.debug = function (msg) {
651 var project = this.tsLsHost;
652 if (!project.projectService) {
653 // tsLsHost is not a Project
654 return;
655 }
656 var logger = project.projectService.logger;
657 if (logger.hasLevel(tss.server.LogLevel.verbose)) {
658 logger.info(msg);
659 }
660 };
661 return TypeScriptServiceHost;
662 }());
663 exports.TypeScriptServiceHost = TypeScriptServiceHost;
664 function findSuitableDefaultModule(modules) {
665 var e_8, _a;
666 var result = undefined;
667 var resultSize = 0;
668 try {
669 for (var _b = tslib_1.__values(modules.ngModules), _c = _b.next(); !_c.done; _c = _b.next()) {
670 var module_1 = _c.value;
671 var moduleSize = module_1.transitiveModule.directives.length;
672 if (moduleSize > resultSize) {
673 result = module_1;
674 resultSize = moduleSize;
675 }
676 }
677 }
678 catch (e_8_1) { e_8 = { error: e_8_1 }; }
679 finally {
680 try {
681 if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
682 }
683 finally { if (e_8) throw e_8.error; }
684 }
685 return result;
686 }
687 function spanOf(node) {
688 return { start: node.getStart(), end: node.getEnd() };
689 }
690 function spanAt(sourceFile, line, column) {
691 if (line != null && column != null) {
692 var position_1 = tss.getPositionOfLineAndCharacter(sourceFile, line, column);
693 var findChild = function findChild(node) {
694 if (node.kind > tss.SyntaxKind.LastToken && node.pos <= position_1 && node.end > position_1) {
695 var betterNode = tss.forEachChild(node, findChild);
696 return betterNode || node;
697 }
698 };
699 var node = tss.forEachChild(sourceFile, findChild);
700 if (node) {
701 return { start: node.getStart(), end: node.getEnd() };
702 }
703 }
704 }
705 function convertChain(chain) {
706 return { message: chain.message, next: chain.next ? chain.next.map(convertChain) : undefined };
707 }
708 function errorToDiagnosticWithChain(error, span) {
709 return { message: error.chain ? convertChain(error.chain) : error.message, span: span };
710 }
711});
712//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"typescript_host.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/typescript_host.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\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 {createLanguageService} from './language_service';\nimport {ReflectorHost} from './reflector_host';\nimport {ExternalTemplate, InlineTemplate} from './template';\nimport {findTightestNode, getClassDeclFromDecoratorProp, getDirectiveClassLike, getPropertyAssignmentFromValue} from './ts_utils';\nimport {AstResult, Declaration, DeclarationError, DiagnosticMessageChain, LanguageService, LanguageServiceHost, Span, TemplateSource} from './types';\n\n/**\n * Create a `LanguageServiceHost`\n */\nexport function createLanguageServiceFromTypescript(\n    host: tss.LanguageServiceHost, service: tss.LanguageService): LanguageService {\n  const ngHost = new TypeScriptServiceHost(host, service);\n  const ngServer = createLanguageService(ngHost);\n  return ngServer;\n}\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  private readonly fileToComponent = new Map<string, 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(): string[] {\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(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(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