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 | ;
|
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":"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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 {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 |