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/diagnostics", ["require", "exports", "tslib", "typescript", "@angular/language-service/src/diagnostic_messages", "@angular/language-service/src/expression_diagnostics", "@angular/language-service/src/ts_utils", "@angular/language-service/src/utils"], factory);
|
15 | }
|
16 | })(function (require, exports) {
|
17 | ;
|
18 | Object.defineProperty(exports, "__esModule", { value: true });
|
19 | exports.ngDiagnosticToTsDiagnostic = exports.getDeclarationDiagnostics = exports.getTemplateDiagnostics = void 0;
|
20 | var tslib_1 = require("tslib");
|
21 | var ts = require("typescript");
|
22 | var diagnostic_messages_1 = require("@angular/language-service/src/diagnostic_messages");
|
23 | var expression_diagnostics_1 = require("@angular/language-service/src/expression_diagnostics");
|
24 | var ts_utils_1 = require("@angular/language-service/src/ts_utils");
|
25 | var utils_1 = require("@angular/language-service/src/utils");
|
26 | /**
|
27 | * Return diagnostic information for the parsed AST of the template.
|
28 | * @param ast contains HTML and template AST
|
29 | */
|
30 | function getTemplateDiagnostics(ast) {
|
31 | var parseErrors = ast.parseErrors, templateAst = ast.templateAst, htmlAst = ast.htmlAst, template = ast.template;
|
32 | if (parseErrors && parseErrors.length) {
|
33 | return parseErrors.map(function (e) {
|
34 | return {
|
35 | kind: ts.DiagnosticCategory.Error,
|
36 | span: utils_1.offsetSpan(utils_1.spanOf(e.span), template.span.start),
|
37 | message: e.msg,
|
38 | };
|
39 | });
|
40 | }
|
41 | return expression_diagnostics_1.getTemplateExpressionDiagnostics({
|
42 | templateAst: templateAst,
|
43 | htmlAst: htmlAst,
|
44 | offset: template.span.start,
|
45 | query: template.query,
|
46 | members: template.members,
|
47 | source: ast.template.source,
|
48 | });
|
49 | }
|
50 | exports.getTemplateDiagnostics = getTemplateDiagnostics;
|
51 | /**
|
52 | * Performs a variety diagnostics on directive declarations.
|
53 | *
|
54 | * @param declarations Angular directive declarations
|
55 | * @param modules NgModules in the project
|
56 | * @param host TypeScript service host used to perform TypeScript queries
|
57 | * @return diagnosed errors, if any
|
58 | */
|
59 | function getDeclarationDiagnostics(declarations, modules, host) {
|
60 | var e_1, _a, e_2, _b, e_3, _c, e_4, _d;
|
61 | var directives = new Set();
|
62 | try {
|
63 | for (var _e = tslib_1.__values(modules.ngModules), _f = _e.next(); !_f.done; _f = _e.next()) {
|
64 | var ngModule = _f.value;
|
65 | try {
|
66 | for (var _g = (e_2 = void 0, tslib_1.__values(ngModule.declaredDirectives)), _h = _g.next(); !_h.done; _h = _g.next()) {
|
67 | var directive = _h.value;
|
68 | directives.add(directive.reference);
|
69 | }
|
70 | }
|
71 | catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
72 | finally {
|
73 | try {
|
74 | if (_h && !_h.done && (_b = _g.return)) _b.call(_g);
|
75 | }
|
76 | finally { if (e_2) throw e_2.error; }
|
77 | }
|
78 | }
|
79 | }
|
80 | catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
81 | finally {
|
82 | try {
|
83 | if (_f && !_f.done && (_a = _e.return)) _a.call(_e);
|
84 | }
|
85 | finally { if (e_1) throw e_1.error; }
|
86 | }
|
87 | var results = [];
|
88 | try {
|
89 | for (var declarations_1 = tslib_1.__values(declarations), declarations_1_1 = declarations_1.next(); !declarations_1_1.done; declarations_1_1 = declarations_1.next()) {
|
90 | var declaration = declarations_1_1.value;
|
91 | var errors = declaration.errors, metadata = declaration.metadata, type = declaration.type, declarationSpan = declaration.declarationSpan;
|
92 | var sf = host.getSourceFile(type.filePath);
|
93 | if (!sf) {
|
94 | host.error("directive " + type.name + " exists but has no source file");
|
95 | return [];
|
96 | }
|
97 | // TypeScript identifier of the directive declaration annotation (e.g. "Component" or
|
98 | // "Directive") on a directive class.
|
99 | var directiveIdentifier = ts_utils_1.findTightestNode(sf, declarationSpan.start);
|
100 | if (!directiveIdentifier) {
|
101 | host.error("directive " + type.name + " exists but has no identifier");
|
102 | return [];
|
103 | }
|
104 | try {
|
105 | for (var errors_1 = (e_4 = void 0, tslib_1.__values(errors)), errors_1_1 = errors_1.next(); !errors_1_1.done; errors_1_1 = errors_1.next()) {
|
106 | var error = errors_1_1.value;
|
107 | results.push({
|
108 | kind: ts.DiagnosticCategory.Error,
|
109 | message: error.message,
|
110 | span: error.span,
|
111 | });
|
112 | }
|
113 | }
|
114 | catch (e_4_1) { e_4 = { error: e_4_1 }; }
|
115 | finally {
|
116 | try {
|
117 | if (errors_1_1 && !errors_1_1.done && (_d = errors_1.return)) _d.call(errors_1);
|
118 | }
|
119 | finally { if (e_4) throw e_4.error; }
|
120 | }
|
121 | if (!modules.ngModuleByPipeOrDirective.has(declaration.type)) {
|
122 | results.push(diagnostic_messages_1.createDiagnostic(declarationSpan, diagnostic_messages_1.Diagnostic.directive_not_in_module, metadata.isComponent ? 'Component' : 'Directive', type.name));
|
123 | }
|
124 | if (metadata.isComponent) {
|
125 | var _j = metadata.template, template = _j.template, templateUrl = _j.templateUrl, styleUrls = _j.styleUrls;
|
126 | if (template === null && !templateUrl) {
|
127 | results.push(diagnostic_messages_1.createDiagnostic(declarationSpan, diagnostic_messages_1.Diagnostic.missing_template_and_templateurl, type.name));
|
128 | }
|
129 | else if (templateUrl) {
|
130 | if (template) {
|
131 | results.push(diagnostic_messages_1.createDiagnostic(declarationSpan, diagnostic_messages_1.Diagnostic.both_template_and_templateurl, type.name));
|
132 | }
|
133 | // Find templateUrl value from the directive call expression, which is the parent of the
|
134 | // directive identifier.
|
135 | //
|
136 | // TODO: We should create an enum of the various properties a directive can have to use
|
137 | // instead of string literals. We can then perform a mass migration of all literal usages.
|
138 | var templateUrlNode = ts_utils_1.findPropertyValueOfType(directiveIdentifier.parent, 'templateUrl', ts.isLiteralExpression);
|
139 | if (!templateUrlNode) {
|
140 | host.error("templateUrl " + templateUrl + " exists but its TypeScript node doesn't");
|
141 | return [];
|
142 | }
|
143 | results.push.apply(results, tslib_1.__spread(validateUrls([templateUrlNode], host.tsLsHost)));
|
144 | }
|
145 | if (styleUrls.length > 0) {
|
146 | // Find styleUrls value from the directive call expression, which is the parent of the
|
147 | // directive identifier.
|
148 | var styleUrlsNode = ts_utils_1.findPropertyValueOfType(directiveIdentifier.parent, 'styleUrls', ts.isArrayLiteralExpression);
|
149 | if (!styleUrlsNode) {
|
150 | host.error("styleUrls property exists but its TypeScript node doesn't'");
|
151 | return [];
|
152 | }
|
153 | results.push.apply(results, tslib_1.__spread(validateUrls(styleUrlsNode.elements, host.tsLsHost)));
|
154 | }
|
155 | }
|
156 | }
|
157 | }
|
158 | catch (e_3_1) { e_3 = { error: e_3_1 }; }
|
159 | finally {
|
160 | try {
|
161 | if (declarations_1_1 && !declarations_1_1.done && (_c = declarations_1.return)) _c.call(declarations_1);
|
162 | }
|
163 | finally { if (e_3) throw e_3.error; }
|
164 | }
|
165 | return results;
|
166 | }
|
167 | exports.getDeclarationDiagnostics = getDeclarationDiagnostics;
|
168 | /**
|
169 | * Checks that URLs on a directive point to a valid file.
|
170 | * Note that this diagnostic check may require a filesystem hit, and thus may be slower than other
|
171 | * checks.
|
172 | *
|
173 | * @param urls urls to check for validity
|
174 | * @param tsLsHost TS LS host used for querying filesystem information
|
175 | * @return diagnosed url errors, if any
|
176 | */
|
177 | function validateUrls(urls, tsLsHost) {
|
178 | if (!tsLsHost.fileExists) {
|
179 | return [];
|
180 | }
|
181 | var allErrors = [];
|
182 | // TODO(ayazhafiz): most of this logic can be unified with the logic in
|
183 | // definitions.ts#getUrlFromProperty. Create a utility function to be used by both.
|
184 | for (var i = 0; i < urls.length; ++i) {
|
185 | var urlNode = urls[i];
|
186 | if (!ts.isStringLiteralLike(urlNode)) {
|
187 | // If a non-string value is assigned to a URL node (like `templateUrl`), a type error will be
|
188 | // picked up by the TS Language Server.
|
189 | continue;
|
190 | }
|
191 | var url = utils_1.extractAbsoluteFilePath(urlNode);
|
192 | if (tsLsHost.fileExists(url))
|
193 | continue;
|
194 | // Exclude opening and closing quotes in the url span.
|
195 | var urlSpan = { start: urlNode.getStart() + 1, end: urlNode.end - 1 };
|
196 | allErrors.push(diagnostic_messages_1.createDiagnostic(urlSpan, diagnostic_messages_1.Diagnostic.invalid_templateurl));
|
197 | }
|
198 | return allErrors;
|
199 | }
|
200 | /**
|
201 | * Return a recursive data structure that chains diagnostic messages.
|
202 | * @param chain
|
203 | */
|
204 | function chainDiagnostics(chain) {
|
205 | return {
|
206 | messageText: chain.message,
|
207 | category: ts.DiagnosticCategory.Error,
|
208 | code: 0,
|
209 | next: chain.next ? chain.next.map(chainDiagnostics) : undefined
|
210 | };
|
211 | }
|
212 | /**
|
213 | * Convert ng.Diagnostic to ts.Diagnostic.
|
214 | * @param d diagnostic
|
215 | * @param file
|
216 | */
|
217 | function ngDiagnosticToTsDiagnostic(d, file) {
|
218 | return {
|
219 | file: file,
|
220 | start: d.span.start,
|
221 | length: d.span.end - d.span.start,
|
222 | messageText: typeof d.message === 'string' ? d.message : chainDiagnostics(d.message),
|
223 | category: d.kind,
|
224 | code: 0,
|
225 | source: 'ng',
|
226 | };
|
227 | }
|
228 | exports.ngDiagnosticToTsDiagnostic = ngDiagnosticToTsDiagnostic;
|
229 | });
|
230 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diagnostics.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/diagnostics.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 {NgAnalyzedModules} from '@angular/compiler';\nimport * as ts from 'typescript';\n\nimport {createDiagnostic, Diagnostic} from './diagnostic_messages';\nimport {getTemplateExpressionDiagnostics} from './expression_diagnostics';\nimport {findPropertyValueOfType, findTightestNode} from './ts_utils';\nimport * as ng from './types';\nimport {TypeScriptServiceHost} from './typescript_host';\nimport {extractAbsoluteFilePath, offsetSpan, spanOf} from './utils';\n\n/**\n * Return diagnostic information for the parsed AST of the template.\n * @param ast contains HTML and template AST\n */\nexport function getTemplateDiagnostics(ast: ng.AstResult): ng.Diagnostic[] {\n  const {parseErrors, templateAst, htmlAst, template} = ast;\n  if (parseErrors && parseErrors.length) {\n    return parseErrors.map(e => {\n      return {\n        kind: ts.DiagnosticCategory.Error,\n        span: offsetSpan(spanOf(e.span), template.span.start),\n        message: e.msg,\n      };\n    });\n  }\n  return getTemplateExpressionDiagnostics({\n    templateAst: templateAst,\n    htmlAst: htmlAst,\n    offset: template.span.start,\n    query: template.query,\n    members: template.members,\n    source: ast.template.source,\n  });\n}\n\n/**\n * Performs a variety diagnostics on directive declarations.\n *\n * @param declarations Angular directive declarations\n * @param modules NgModules in the project\n * @param host TypeScript service host used to perform TypeScript queries\n * @return diagnosed errors, if any\n */\nexport function getDeclarationDiagnostics(\n    declarations: ng.Declaration[], modules: NgAnalyzedModules,\n    host: Readonly<TypeScriptServiceHost>): ng.Diagnostic[] {\n  const directives = new Set<ng.StaticSymbol>();\n  for (const ngModule of modules.ngModules) {\n    for (const directive of ngModule.declaredDirectives) {\n      directives.add(directive.reference);\n    }\n  }\n\n  const results: ng.Diagnostic[] = [];\n\n  for (const declaration of declarations) {\n    const {errors, metadata, type, declarationSpan} = declaration;\n\n    const sf = host.getSourceFile(type.filePath);\n    if (!sf) {\n      host.error(`directive ${type.name} exists but has no source file`);\n      return [];\n    }\n    // TypeScript identifier of the directive declaration annotation (e.g. \"Component\" or\n    // \"Directive\") on a directive class.\n    const directiveIdentifier = findTightestNode(sf, declarationSpan.start);\n    if (!directiveIdentifier) {\n      host.error(`directive ${type.name} exists but has no identifier`);\n      return [];\n    }\n\n    for (const error of errors) {\n      results.push({\n        kind: ts.DiagnosticCategory.Error,\n        message: error.message,\n        span: error.span,\n      });\n    }\n\n    if (!modules.ngModuleByPipeOrDirective.has(declaration.type)) {\n      results.push(createDiagnostic(\n          declarationSpan, Diagnostic.directive_not_in_module,\n          metadata.isComponent ? 'Component' : 'Directive', type.name));\n    }\n\n    if (metadata.isComponent) {\n      const {template, templateUrl, styleUrls} = metadata.template !;\n      if (template === null && !templateUrl) {\n        results.push(createDiagnostic(\n            declarationSpan, Diagnostic.missing_template_and_templateurl, type.name));\n      } else if (templateUrl) {\n        if (template) {\n          results.push(createDiagnostic(\n              declarationSpan, Diagnostic.both_template_and_templateurl, type.name));\n        }\n\n        // Find templateUrl value from the directive call expression, which is the parent of the\n        // directive identifier.\n        //\n        // TODO: We should create an enum of the various properties a directive can have to use\n        // instead of string literals. We can then perform a mass migration of all literal usages.\n        const templateUrlNode = findPropertyValueOfType(\n            directiveIdentifier.parent, 'templateUrl', ts.isLiteralExpression);\n        if (!templateUrlNode) {\n          host.error(`templateUrl ${templateUrl} exists but its TypeScript node doesn't`);\n          return [];\n        }\n\n        results.push(...validateUrls([templateUrlNode], host.tsLsHost));\n      }\n\n      if (styleUrls.length > 0) {\n        // Find styleUrls value from the directive call expression, which is the parent of the\n        // directive identifier.\n        const styleUrlsNode = findPropertyValueOfType(\n            directiveIdentifier.parent, 'styleUrls', ts.isArrayLiteralExpression);\n        if (!styleUrlsNode) {\n          host.error(`styleUrls property exists but its TypeScript node doesn't'`);\n          return [];\n        }\n\n        results.push(...validateUrls(styleUrlsNode.elements, host.tsLsHost));\n      }\n    }\n  }\n\n  return results;\n}\n\n/**\n * Checks that URLs on a directive point to a valid file.\n * Note that this diagnostic check may require a filesystem hit, and thus may be slower than other\n * checks.\n *\n * @param urls urls to check for validity\n * @param tsLsHost TS LS host used for querying filesystem information\n * @return diagnosed url errors, if any\n */\nfunction validateUrls(\n    urls: ArrayLike<ts.Expression>, tsLsHost: Readonly<ts.LanguageServiceHost>): ng.Diagnostic[] {\n  if (!tsLsHost.fileExists) {\n    return [];\n  }\n\n  const allErrors: ng.Diagnostic[] = [];\n  // TODO(ayazhafiz): most of this logic can be unified with the logic in\n  // definitions.ts#getUrlFromProperty. Create a utility function to be used by both.\n  for (let i = 0; i < urls.length; ++i) {\n    const urlNode = urls[i];\n    if (!ts.isStringLiteralLike(urlNode)) {\n      // If a non-string value is assigned to a URL node (like `templateUrl`), a type error will be\n      // picked up by the TS Language Server.\n      continue;\n    }\n\n    const url = extractAbsoluteFilePath(urlNode);\n    if (tsLsHost.fileExists(url)) continue;\n\n    // Exclude opening and closing quotes in the url span.\n    const urlSpan = {start: urlNode.getStart() + 1, end: urlNode.end - 1};\n    allErrors.push(createDiagnostic(urlSpan, Diagnostic.invalid_templateurl));\n  }\n  return allErrors;\n}\n\n/**\n * Return a recursive data structure that chains diagnostic messages.\n * @param chain\n */\nfunction chainDiagnostics(chain: ng.DiagnosticMessageChain): ts.DiagnosticMessageChain {\n  return {\n    messageText: chain.message,\n    category: ts.DiagnosticCategory.Error,\n    code: 0,\n    next: chain.next ? chain.next.map(chainDiagnostics) : undefined\n  };\n}\n\n/**\n * Convert ng.Diagnostic to ts.Diagnostic.\n * @param d diagnostic\n * @param file\n */\nexport function ngDiagnosticToTsDiagnostic(\n    d: ng.Diagnostic, file: ts.SourceFile|undefined): ts.Diagnostic {\n  return {\n    file,\n    start: d.span.start,\n    length: d.span.end - d.span.start,\n    messageText: typeof d.message === 'string' ? d.message : chainDiagnostics(d.message),\n    category: d.kind,\n    code: 0,\n    source: 'ng',\n  };\n}\n"]} |
\ | No newline at end of file |