1 | /**
|
---|
2 | * @license
|
---|
3 | * Copyright Google LLC All Rights Reserved.
|
---|
4 | *
|
---|
5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
6 | * found in the LICENSE file at https://angular.io/license
|
---|
7 | */
|
---|
8 | import { ParseError, ParseSourceSpan } from '../parse_util';
|
---|
9 | import * as html from './ast';
|
---|
10 | import { NAMED_ENTITIES } from './entities';
|
---|
11 | import { tokenize } from './lexer';
|
---|
12 | import { getNsPrefix, mergeNsAndName, splitNsName } from './tags';
|
---|
13 | export class TreeError extends ParseError {
|
---|
14 | constructor(elementName, span, msg) {
|
---|
15 | super(span, msg);
|
---|
16 | this.elementName = elementName;
|
---|
17 | }
|
---|
18 | static create(elementName, span, msg) {
|
---|
19 | return new TreeError(elementName, span, msg);
|
---|
20 | }
|
---|
21 | }
|
---|
22 | export class ParseTreeResult {
|
---|
23 | constructor(rootNodes, errors) {
|
---|
24 | this.rootNodes = rootNodes;
|
---|
25 | this.errors = errors;
|
---|
26 | }
|
---|
27 | }
|
---|
28 | export class Parser {
|
---|
29 | constructor(getTagDefinition) {
|
---|
30 | this.getTagDefinition = getTagDefinition;
|
---|
31 | }
|
---|
32 | parse(source, url, options) {
|
---|
33 | const tokenizeResult = tokenize(source, url, this.getTagDefinition, options);
|
---|
34 | const parser = new _TreeBuilder(tokenizeResult.tokens, this.getTagDefinition);
|
---|
35 | parser.build();
|
---|
36 | return new ParseTreeResult(parser.rootNodes, tokenizeResult.errors.concat(parser.errors));
|
---|
37 | }
|
---|
38 | }
|
---|
39 | class _TreeBuilder {
|
---|
40 | constructor(tokens, getTagDefinition) {
|
---|
41 | this.tokens = tokens;
|
---|
42 | this.getTagDefinition = getTagDefinition;
|
---|
43 | this._index = -1;
|
---|
44 | this._elementStack = [];
|
---|
45 | this.rootNodes = [];
|
---|
46 | this.errors = [];
|
---|
47 | this._advance();
|
---|
48 | }
|
---|
49 | build() {
|
---|
50 | while (this._peek.type !== 24 /* EOF */) {
|
---|
51 | if (this._peek.type === 0 /* TAG_OPEN_START */ ||
|
---|
52 | this._peek.type === 4 /* INCOMPLETE_TAG_OPEN */) {
|
---|
53 | this._consumeStartTag(this._advance());
|
---|
54 | }
|
---|
55 | else if (this._peek.type === 3 /* TAG_CLOSE */) {
|
---|
56 | this._consumeEndTag(this._advance());
|
---|
57 | }
|
---|
58 | else if (this._peek.type === 12 /* CDATA_START */) {
|
---|
59 | this._closeVoidElement();
|
---|
60 | this._consumeCdata(this._advance());
|
---|
61 | }
|
---|
62 | else if (this._peek.type === 10 /* COMMENT_START */) {
|
---|
63 | this._closeVoidElement();
|
---|
64 | this._consumeComment(this._advance());
|
---|
65 | }
|
---|
66 | else if (this._peek.type === 5 /* TEXT */ || this._peek.type === 7 /* RAW_TEXT */ ||
|
---|
67 | this._peek.type === 6 /* ESCAPABLE_RAW_TEXT */) {
|
---|
68 | this._closeVoidElement();
|
---|
69 | this._consumeText(this._advance());
|
---|
70 | }
|
---|
71 | else if (this._peek.type === 19 /* EXPANSION_FORM_START */) {
|
---|
72 | this._consumeExpansion(this._advance());
|
---|
73 | }
|
---|
74 | else {
|
---|
75 | // Skip all other tokens...
|
---|
76 | this._advance();
|
---|
77 | }
|
---|
78 | }
|
---|
79 | }
|
---|
80 | _advance() {
|
---|
81 | const prev = this._peek;
|
---|
82 | if (this._index < this.tokens.length - 1) {
|
---|
83 | // Note: there is always an EOF token at the end
|
---|
84 | this._index++;
|
---|
85 | }
|
---|
86 | this._peek = this.tokens[this._index];
|
---|
87 | return prev;
|
---|
88 | }
|
---|
89 | _advanceIf(type) {
|
---|
90 | if (this._peek.type === type) {
|
---|
91 | return this._advance();
|
---|
92 | }
|
---|
93 | return null;
|
---|
94 | }
|
---|
95 | _consumeCdata(_startToken) {
|
---|
96 | this._consumeText(this._advance());
|
---|
97 | this._advanceIf(13 /* CDATA_END */);
|
---|
98 | }
|
---|
99 | _consumeComment(token) {
|
---|
100 | const text = this._advanceIf(7 /* RAW_TEXT */);
|
---|
101 | this._advanceIf(11 /* COMMENT_END */);
|
---|
102 | const value = text != null ? text.parts[0].trim() : null;
|
---|
103 | this._addToParent(new html.Comment(value, token.sourceSpan));
|
---|
104 | }
|
---|
105 | _consumeExpansion(token) {
|
---|
106 | const switchValue = this._advance();
|
---|
107 | const type = this._advance();
|
---|
108 | const cases = [];
|
---|
109 | // read =
|
---|
110 | while (this._peek.type === 20 /* EXPANSION_CASE_VALUE */) {
|
---|
111 | const expCase = this._parseExpansionCase();
|
---|
112 | if (!expCase)
|
---|
113 | return; // error
|
---|
114 | cases.push(expCase);
|
---|
115 | }
|
---|
116 | // read the final }
|
---|
117 | if (this._peek.type !== 23 /* EXPANSION_FORM_END */) {
|
---|
118 | this.errors.push(TreeError.create(null, this._peek.sourceSpan, `Invalid ICU message. Missing '}'.`));
|
---|
119 | return;
|
---|
120 | }
|
---|
121 | const sourceSpan = new ParseSourceSpan(token.sourceSpan.start, this._peek.sourceSpan.end, token.sourceSpan.fullStart);
|
---|
122 | this._addToParent(new html.Expansion(switchValue.parts[0], type.parts[0], cases, sourceSpan, switchValue.sourceSpan));
|
---|
123 | this._advance();
|
---|
124 | }
|
---|
125 | _parseExpansionCase() {
|
---|
126 | const value = this._advance();
|
---|
127 | // read {
|
---|
128 | if (this._peek.type !== 21 /* EXPANSION_CASE_EXP_START */) {
|
---|
129 | this.errors.push(TreeError.create(null, this._peek.sourceSpan, `Invalid ICU message. Missing '{'.`));
|
---|
130 | return null;
|
---|
131 | }
|
---|
132 | // read until }
|
---|
133 | const start = this._advance();
|
---|
134 | const exp = this._collectExpansionExpTokens(start);
|
---|
135 | if (!exp)
|
---|
136 | return null;
|
---|
137 | const end = this._advance();
|
---|
138 | exp.push({ type: 24 /* EOF */, parts: [], sourceSpan: end.sourceSpan });
|
---|
139 | // parse everything in between { and }
|
---|
140 | const expansionCaseParser = new _TreeBuilder(exp, this.getTagDefinition);
|
---|
141 | expansionCaseParser.build();
|
---|
142 | if (expansionCaseParser.errors.length > 0) {
|
---|
143 | this.errors = this.errors.concat(expansionCaseParser.errors);
|
---|
144 | return null;
|
---|
145 | }
|
---|
146 | const sourceSpan = new ParseSourceSpan(value.sourceSpan.start, end.sourceSpan.end, value.sourceSpan.fullStart);
|
---|
147 | const expSourceSpan = new ParseSourceSpan(start.sourceSpan.start, end.sourceSpan.end, start.sourceSpan.fullStart);
|
---|
148 | return new html.ExpansionCase(value.parts[0], expansionCaseParser.rootNodes, sourceSpan, value.sourceSpan, expSourceSpan);
|
---|
149 | }
|
---|
150 | _collectExpansionExpTokens(start) {
|
---|
151 | const exp = [];
|
---|
152 | const expansionFormStack = [21 /* EXPANSION_CASE_EXP_START */];
|
---|
153 | while (true) {
|
---|
154 | if (this._peek.type === 19 /* EXPANSION_FORM_START */ ||
|
---|
155 | this._peek.type === 21 /* EXPANSION_CASE_EXP_START */) {
|
---|
156 | expansionFormStack.push(this._peek.type);
|
---|
157 | }
|
---|
158 | if (this._peek.type === 22 /* EXPANSION_CASE_EXP_END */) {
|
---|
159 | if (lastOnStack(expansionFormStack, 21 /* EXPANSION_CASE_EXP_START */)) {
|
---|
160 | expansionFormStack.pop();
|
---|
161 | if (expansionFormStack.length === 0)
|
---|
162 | return exp;
|
---|
163 | }
|
---|
164 | else {
|
---|
165 | this.errors.push(TreeError.create(null, start.sourceSpan, `Invalid ICU message. Missing '}'.`));
|
---|
166 | return null;
|
---|
167 | }
|
---|
168 | }
|
---|
169 | if (this._peek.type === 23 /* EXPANSION_FORM_END */) {
|
---|
170 | if (lastOnStack(expansionFormStack, 19 /* EXPANSION_FORM_START */)) {
|
---|
171 | expansionFormStack.pop();
|
---|
172 | }
|
---|
173 | else {
|
---|
174 | this.errors.push(TreeError.create(null, start.sourceSpan, `Invalid ICU message. Missing '}'.`));
|
---|
175 | return null;
|
---|
176 | }
|
---|
177 | }
|
---|
178 | if (this._peek.type === 24 /* EOF */) {
|
---|
179 | this.errors.push(TreeError.create(null, start.sourceSpan, `Invalid ICU message. Missing '}'.`));
|
---|
180 | return null;
|
---|
181 | }
|
---|
182 | exp.push(this._advance());
|
---|
183 | }
|
---|
184 | }
|
---|
185 | _consumeText(token) {
|
---|
186 | const tokens = [token];
|
---|
187 | const startSpan = token.sourceSpan;
|
---|
188 | let text = token.parts[0];
|
---|
189 | if (text.length > 0 && text[0] === '\n') {
|
---|
190 | const parent = this._getParentElement();
|
---|
191 | if (parent != null && parent.children.length === 0 &&
|
---|
192 | this.getTagDefinition(parent.name).ignoreFirstLf) {
|
---|
193 | text = text.substring(1);
|
---|
194 | tokens[0] = { type: token.type, sourceSpan: token.sourceSpan, parts: [text] };
|
---|
195 | }
|
---|
196 | }
|
---|
197 | while (this._peek.type === 8 /* INTERPOLATION */ || this._peek.type === 5 /* TEXT */ ||
|
---|
198 | this._peek.type === 9 /* ENCODED_ENTITY */) {
|
---|
199 | token = this._advance();
|
---|
200 | tokens.push(token);
|
---|
201 | if (token.type === 8 /* INTERPOLATION */) {
|
---|
202 | // For backward compatibility we decode HTML entities that appear in interpolation
|
---|
203 | // expressions. This is arguably a bug, but it could be a considerable breaking change to
|
---|
204 | // fix it. It should be addressed in a larger project to refactor the entire parser/lexer
|
---|
205 | // chain after View Engine has been removed.
|
---|
206 | text += token.parts.join('').replace(/&([^;]+);/g, decodeEntity);
|
---|
207 | }
|
---|
208 | else if (token.type === 9 /* ENCODED_ENTITY */) {
|
---|
209 | text += token.parts[0];
|
---|
210 | }
|
---|
211 | else {
|
---|
212 | text += token.parts.join('');
|
---|
213 | }
|
---|
214 | }
|
---|
215 | if (text.length > 0) {
|
---|
216 | const endSpan = token.sourceSpan;
|
---|
217 | this._addToParent(new html.Text(text, new ParseSourceSpan(startSpan.start, endSpan.end, startSpan.fullStart, startSpan.details), tokens));
|
---|
218 | }
|
---|
219 | }
|
---|
220 | _closeVoidElement() {
|
---|
221 | const el = this._getParentElement();
|
---|
222 | if (el && this.getTagDefinition(el.name).isVoid) {
|
---|
223 | this._elementStack.pop();
|
---|
224 | }
|
---|
225 | }
|
---|
226 | _consumeStartTag(startTagToken) {
|
---|
227 | const [prefix, name] = startTagToken.parts;
|
---|
228 | const attrs = [];
|
---|
229 | while (this._peek.type === 14 /* ATTR_NAME */) {
|
---|
230 | attrs.push(this._consumeAttr(this._advance()));
|
---|
231 | }
|
---|
232 | const fullName = this._getElementFullName(prefix, name, this._getParentElement());
|
---|
233 | let selfClosing = false;
|
---|
234 | // Note: There could have been a tokenizer error
|
---|
235 | // so that we don't get a token for the end tag...
|
---|
236 | if (this._peek.type === 2 /* TAG_OPEN_END_VOID */) {
|
---|
237 | this._advance();
|
---|
238 | selfClosing = true;
|
---|
239 | const tagDef = this.getTagDefinition(fullName);
|
---|
240 | if (!(tagDef.canSelfClose || getNsPrefix(fullName) !== null || tagDef.isVoid)) {
|
---|
241 | this.errors.push(TreeError.create(fullName, startTagToken.sourceSpan, `Only void and foreign elements can be self closed "${startTagToken.parts[1]}"`));
|
---|
242 | }
|
---|
243 | }
|
---|
244 | else if (this._peek.type === 1 /* TAG_OPEN_END */) {
|
---|
245 | this._advance();
|
---|
246 | selfClosing = false;
|
---|
247 | }
|
---|
248 | const end = this._peek.sourceSpan.fullStart;
|
---|
249 | const span = new ParseSourceSpan(startTagToken.sourceSpan.start, end, startTagToken.sourceSpan.fullStart);
|
---|
250 | // Create a separate `startSpan` because `span` will be modified when there is an `end` span.
|
---|
251 | const startSpan = new ParseSourceSpan(startTagToken.sourceSpan.start, end, startTagToken.sourceSpan.fullStart);
|
---|
252 | const el = new html.Element(fullName, attrs, [], span, startSpan, undefined);
|
---|
253 | this._pushElement(el);
|
---|
254 | if (selfClosing) {
|
---|
255 | // Elements that are self-closed have their `endSourceSpan` set to the full span, as the
|
---|
256 | // element start tag also represents the end tag.
|
---|
257 | this._popElement(fullName, span);
|
---|
258 | }
|
---|
259 | else if (startTagToken.type === 4 /* INCOMPLETE_TAG_OPEN */) {
|
---|
260 | // We already know the opening tag is not complete, so it is unlikely it has a corresponding
|
---|
261 | // close tag. Let's optimistically parse it as a full element and emit an error.
|
---|
262 | this._popElement(fullName, null);
|
---|
263 | this.errors.push(TreeError.create(fullName, span, `Opening tag "${fullName}" not terminated.`));
|
---|
264 | }
|
---|
265 | }
|
---|
266 | _pushElement(el) {
|
---|
267 | const parentEl = this._getParentElement();
|
---|
268 | if (parentEl && this.getTagDefinition(parentEl.name).isClosedByChild(el.name)) {
|
---|
269 | this._elementStack.pop();
|
---|
270 | }
|
---|
271 | this._addToParent(el);
|
---|
272 | this._elementStack.push(el);
|
---|
273 | }
|
---|
274 | _consumeEndTag(endTagToken) {
|
---|
275 | const fullName = this._getElementFullName(endTagToken.parts[0], endTagToken.parts[1], this._getParentElement());
|
---|
276 | if (this.getTagDefinition(fullName).isVoid) {
|
---|
277 | this.errors.push(TreeError.create(fullName, endTagToken.sourceSpan, `Void elements do not have end tags "${endTagToken.parts[1]}"`));
|
---|
278 | }
|
---|
279 | else if (!this._popElement(fullName, endTagToken.sourceSpan)) {
|
---|
280 | const errMsg = `Unexpected closing tag "${fullName}". It may happen when the tag has already been closed by another tag. For more info see https://www.w3.org/TR/html5/syntax.html#closing-elements-that-have-implied-end-tags`;
|
---|
281 | this.errors.push(TreeError.create(fullName, endTagToken.sourceSpan, errMsg));
|
---|
282 | }
|
---|
283 | }
|
---|
284 | /**
|
---|
285 | * Closes the nearest element with the tag name `fullName` in the parse tree.
|
---|
286 | * `endSourceSpan` is the span of the closing tag, or null if the element does
|
---|
287 | * not have a closing tag (for example, this happens when an incomplete
|
---|
288 | * opening tag is recovered).
|
---|
289 | */
|
---|
290 | _popElement(fullName, endSourceSpan) {
|
---|
291 | let unexpectedCloseTagDetected = false;
|
---|
292 | for (let stackIndex = this._elementStack.length - 1; stackIndex >= 0; stackIndex--) {
|
---|
293 | const el = this._elementStack[stackIndex];
|
---|
294 | if (el.name === fullName) {
|
---|
295 | // Record the parse span with the element that is being closed. Any elements that are
|
---|
296 | // removed from the element stack at this point are closed implicitly, so they won't get
|
---|
297 | // an end source span (as there is no explicit closing element).
|
---|
298 | el.endSourceSpan = endSourceSpan;
|
---|
299 | el.sourceSpan.end = endSourceSpan !== null ? endSourceSpan.end : el.sourceSpan.end;
|
---|
300 | this._elementStack.splice(stackIndex, this._elementStack.length - stackIndex);
|
---|
301 | return !unexpectedCloseTagDetected;
|
---|
302 | }
|
---|
303 | if (!this.getTagDefinition(el.name).closedByParent) {
|
---|
304 | // Note that we encountered an unexpected close tag but continue processing the element
|
---|
305 | // stack so we can assign an `endSourceSpan` if there is a corresponding start tag for this
|
---|
306 | // end tag in the stack.
|
---|
307 | unexpectedCloseTagDetected = true;
|
---|
308 | }
|
---|
309 | }
|
---|
310 | return false;
|
---|
311 | }
|
---|
312 | _consumeAttr(attrName) {
|
---|
313 | const fullName = mergeNsAndName(attrName.parts[0], attrName.parts[1]);
|
---|
314 | let attrEnd = attrName.sourceSpan.end;
|
---|
315 | // Consume any quote
|
---|
316 | if (this._peek.type === 15 /* ATTR_QUOTE */) {
|
---|
317 | this._advance();
|
---|
318 | }
|
---|
319 | // Consume the attribute value
|
---|
320 | let value = '';
|
---|
321 | const valueTokens = [];
|
---|
322 | let valueStartSpan = undefined;
|
---|
323 | let valueEnd = undefined;
|
---|
324 | // NOTE: We need to use a new variable `nextTokenType` here to hide the actual type of
|
---|
325 | // `_peek.type` from TS. Otherwise TS will narrow the type of `_peek.type` preventing it from
|
---|
326 | // being able to consider `ATTR_VALUE_INTERPOLATION` as an option. This is because TS is not
|
---|
327 | // able to see that `_advance()` will actually mutate `_peek`.
|
---|
328 | const nextTokenType = this._peek.type;
|
---|
329 | if (nextTokenType === 16 /* ATTR_VALUE_TEXT */) {
|
---|
330 | valueStartSpan = this._peek.sourceSpan;
|
---|
331 | valueEnd = this._peek.sourceSpan.end;
|
---|
332 | while (this._peek.type === 16 /* ATTR_VALUE_TEXT */ ||
|
---|
333 | this._peek.type === 17 /* ATTR_VALUE_INTERPOLATION */ ||
|
---|
334 | this._peek.type === 9 /* ENCODED_ENTITY */) {
|
---|
335 | const valueToken = this._advance();
|
---|
336 | valueTokens.push(valueToken);
|
---|
337 | if (valueToken.type === 17 /* ATTR_VALUE_INTERPOLATION */) {
|
---|
338 | // For backward compatibility we decode HTML entities that appear in interpolation
|
---|
339 | // expressions. This is arguably a bug, but it could be a considerable breaking change to
|
---|
340 | // fix it. It should be addressed in a larger project to refactor the entire parser/lexer
|
---|
341 | // chain after View Engine has been removed.
|
---|
342 | value += valueToken.parts.join('').replace(/&([^;]+);/g, decodeEntity);
|
---|
343 | }
|
---|
344 | else if (valueToken.type === 9 /* ENCODED_ENTITY */) {
|
---|
345 | value += valueToken.parts[0];
|
---|
346 | }
|
---|
347 | else {
|
---|
348 | value += valueToken.parts.join('');
|
---|
349 | }
|
---|
350 | valueEnd = attrEnd = valueToken.sourceSpan.end;
|
---|
351 | }
|
---|
352 | }
|
---|
353 | // Consume any quote
|
---|
354 | if (this._peek.type === 15 /* ATTR_QUOTE */) {
|
---|
355 | const quoteToken = this._advance();
|
---|
356 | attrEnd = quoteToken.sourceSpan.end;
|
---|
357 | }
|
---|
358 | const valueSpan = valueStartSpan && valueEnd &&
|
---|
359 | new ParseSourceSpan(valueStartSpan.start, valueEnd, valueStartSpan.fullStart);
|
---|
360 | return new html.Attribute(fullName, value, new ParseSourceSpan(attrName.sourceSpan.start, attrEnd, attrName.sourceSpan.fullStart), attrName.sourceSpan, valueSpan, valueTokens.length > 0 ? valueTokens : undefined, undefined);
|
---|
361 | }
|
---|
362 | _getParentElement() {
|
---|
363 | return this._elementStack.length > 0 ? this._elementStack[this._elementStack.length - 1] : null;
|
---|
364 | }
|
---|
365 | _addToParent(node) {
|
---|
366 | const parent = this._getParentElement();
|
---|
367 | if (parent != null) {
|
---|
368 | parent.children.push(node);
|
---|
369 | }
|
---|
370 | else {
|
---|
371 | this.rootNodes.push(node);
|
---|
372 | }
|
---|
373 | }
|
---|
374 | _getElementFullName(prefix, localName, parentElement) {
|
---|
375 | if (prefix === '') {
|
---|
376 | prefix = this.getTagDefinition(localName).implicitNamespacePrefix || '';
|
---|
377 | if (prefix === '' && parentElement != null) {
|
---|
378 | const parentTagName = splitNsName(parentElement.name)[1];
|
---|
379 | const parentTagDefinition = this.getTagDefinition(parentTagName);
|
---|
380 | if (!parentTagDefinition.preventNamespaceInheritance) {
|
---|
381 | prefix = getNsPrefix(parentElement.name);
|
---|
382 | }
|
---|
383 | }
|
---|
384 | }
|
---|
385 | return mergeNsAndName(prefix, localName);
|
---|
386 | }
|
---|
387 | }
|
---|
388 | function lastOnStack(stack, element) {
|
---|
389 | return stack.length > 0 && stack[stack.length - 1] === element;
|
---|
390 | }
|
---|
391 | /**
|
---|
392 | * Decode the `entity` string, which we believe is the contents of an HTML entity.
|
---|
393 | *
|
---|
394 | * If the string is not actually a valid/known entity then just return the original `match` string.
|
---|
395 | */
|
---|
396 | function decodeEntity(match, entity) {
|
---|
397 | if (NAMED_ENTITIES[entity] !== undefined) {
|
---|
398 | return NAMED_ENTITIES[entity] || match;
|
---|
399 | }
|
---|
400 | if (/^#x[a-f0-9]+$/i.test(entity)) {
|
---|
401 | return String.fromCodePoint(parseInt(entity.slice(2), 16));
|
---|
402 | }
|
---|
403 | if (/^#\d+$/.test(entity)) {
|
---|
404 | return String.fromCodePoint(parseInt(entity.slice(1), 10));
|
---|
405 | }
|
---|
406 | return match;
|
---|
407 | }
|
---|
408 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"parser.js","sourceRoot":"","sources":["../../../../../../../packages/compiler/src/ml_parser/parser.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 {ParseError, ParseLocation, ParseSourceSpan} from '../parse_util';\n\nimport * as html from './ast';\nimport {NAMED_ENTITIES} from './entities';\nimport {tokenize, TokenizeOptions} from './lexer';\nimport {getNsPrefix, mergeNsAndName, splitNsName, TagDefinition} from './tags';\nimport {AttributeNameToken, AttributeQuoteToken, CdataStartToken, CommentStartToken, ExpansionCaseExpressionEndToken, ExpansionCaseExpressionStartToken, ExpansionCaseValueToken, ExpansionFormStartToken, IncompleteTagOpenToken, InterpolatedAttributeToken, InterpolatedTextToken, TagCloseToken, TagOpenStartToken, TextToken, Token, TokenType} from './tokens';\n\nexport class TreeError extends ParseError {\n  static create(elementName: string|null, span: ParseSourceSpan, msg: string): TreeError {\n    return new TreeError(elementName, span, msg);\n  }\n\n  constructor(public elementName: string|null, span: ParseSourceSpan, msg: string) {\n    super(span, msg);\n  }\n}\n\nexport class ParseTreeResult {\n  constructor(public rootNodes: html.Node[], public errors: ParseError[]) {}\n}\n\nexport class Parser {\n  constructor(public getTagDefinition: (tagName: string) => TagDefinition) {}\n\n  parse(source: string, url: string, options?: TokenizeOptions): ParseTreeResult {\n    const tokenizeResult = tokenize(source, url, this.getTagDefinition, options);\n    const parser = new _TreeBuilder(tokenizeResult.tokens, this.getTagDefinition);\n    parser.build();\n    return new ParseTreeResult(\n        parser.rootNodes,\n        (tokenizeResult.errors as ParseError[]).concat(parser.errors),\n    );\n  }\n}\n\nclass _TreeBuilder {\n  private _index: number = -1;\n  // `_peek` will be initialized by the call to `_advance()` in the constructor.\n  private _peek!: Token;\n  private _elementStack: html.Element[] = [];\n\n  rootNodes: html.Node[] = [];\n  errors: TreeError[] = [];\n\n  constructor(\n      private tokens: Token[], private getTagDefinition: (tagName: string) => TagDefinition) {\n    this._advance();\n  }\n\n  build(): void {\n    while (this._peek.type !== TokenType.EOF) {\n      if (this._peek.type === TokenType.TAG_OPEN_START ||\n          this._peek.type === TokenType.INCOMPLETE_TAG_OPEN) {\n        this._consumeStartTag(this._advance<TagOpenStartToken|IncompleteTagOpenToken>());\n      } else if (this._peek.type === TokenType.TAG_CLOSE) {\n        this._consumeEndTag(this._advance<TagCloseToken>());\n      } else if (this._peek.type === TokenType.CDATA_START) {\n        this._closeVoidElement();\n        this._consumeCdata(this._advance<CdataStartToken>());\n      } else if (this._peek.type === TokenType.COMMENT_START) {\n        this._closeVoidElement();\n        this._consumeComment(this._advance<CommentStartToken>());\n      } else if (\n          this._peek.type === TokenType.TEXT || this._peek.type === TokenType.RAW_TEXT ||\n          this._peek.type === TokenType.ESCAPABLE_RAW_TEXT) {\n        this._closeVoidElement();\n        this._consumeText(this._advance<TextToken>());\n      } else if (this._peek.type === TokenType.EXPANSION_FORM_START) {\n        this._consumeExpansion(this._advance<ExpansionFormStartToken>());\n      } else {\n        // Skip all other tokens...\n        this._advance();\n      }\n    }\n  }\n\n  private _advance<T extends Token>(): T {\n    const prev = this._peek;\n    if (this._index < this.tokens.length - 1) {\n      // Note: there is always an EOF token at the end\n      this._index++;\n    }\n    this._peek = this.tokens[this._index];\n    return prev as T;\n  }\n\n  private _advanceIf<T extends TokenType>(type: T): (Token&{type: T})|null {\n    if (this._peek.type === type) {\n      return this._advance<Token&{type: T}>();\n    }\n    return null;\n  }\n\n  private _consumeCdata(_startToken: CdataStartToken) {\n    this._consumeText(this._advance<TextToken>());\n    this._advanceIf(TokenType.CDATA_END);\n  }\n\n  private _consumeComment(token: CommentStartToken) {\n    const text = this._advanceIf(TokenType.RAW_TEXT);\n    this._advanceIf(TokenType.COMMENT_END);\n    const value = text != null ? text.parts[0].trim() : null;\n    this._addToParent(new html.Comment(value, token.sourceSpan));\n  }\n\n  private _consumeExpansion(token: ExpansionFormStartToken) {\n    const switchValue = this._advance<TextToken>();\n\n    const type = this._advance<TextToken>();\n    const cases: html.ExpansionCase[] = [];\n\n    // read =\n    while (this._peek.type === TokenType.EXPANSION_CASE_VALUE) {\n      const expCase = this._parseExpansionCase();\n      if (!expCase) return;  // error\n      cases.push(expCase);\n    }\n\n    // read the final }\n    if (this._peek.type !== TokenType.EXPANSION_FORM_END) {\n      this.errors.push(\n          TreeError.create(null, this._peek.sourceSpan, `Invalid ICU message. Missing '}'.`));\n      return;\n    }\n    const sourceSpan = new ParseSourceSpan(\n        token.sourceSpan.start, this._peek.sourceSpan.end, token.sourceSpan.fullStart);\n    this._addToParent(new html.Expansion(\n        switchValue.parts[0], type.parts[0], cases, sourceSpan, switchValue.sourceSpan));\n\n    this._advance();\n  }\n\n  private _parseExpansionCase(): html.ExpansionCase|null {\n    const value = this._advance<ExpansionCaseValueToken>();\n\n    // read {\n    if (this._peek.type !== TokenType.EXPANSION_CASE_EXP_START) {\n      this.errors.push(\n          TreeError.create(null, this._peek.sourceSpan, `Invalid ICU message. Missing '{'.`));\n      return null;\n    }\n\n    // read until }\n    const start = this._advance<ExpansionCaseExpressionStartToken>();\n\n    const exp = this._collectExpansionExpTokens(start);\n    if (!exp) return null;\n\n    const end = this._advance<ExpansionCaseExpressionEndToken>();\n    exp.push({type: TokenType.EOF, parts: [], sourceSpan: end.sourceSpan});\n\n    // parse everything in between { and }\n    const expansionCaseParser = new _TreeBuilder(exp, this.getTagDefinition);\n    expansionCaseParser.build();\n    if (expansionCaseParser.errors.length > 0) {\n      this.errors = this.errors.concat(expansionCaseParser.errors);\n      return null;\n    }\n\n    const sourceSpan =\n        new ParseSourceSpan(value.sourceSpan.start, end.sourceSpan.end, value.sourceSpan.fullStart);\n    const expSourceSpan =\n        new ParseSourceSpan(start.sourceSpan.start, end.sourceSpan.end, start.sourceSpan.fullStart);\n    return new html.ExpansionCase(\n        value.parts[0], expansionCaseParser.rootNodes, sourceSpan, value.sourceSpan, expSourceSpan);\n  }\n\n  private _collectExpansionExpTokens(start: Token): Token[]|null {\n    const exp: Token[] = [];\n    const expansionFormStack = [TokenType.EXPANSION_CASE_EXP_START];\n\n    while (true) {\n      if (this._peek.type === TokenType.EXPANSION_FORM_START ||\n          this._peek.type === TokenType.EXPANSION_CASE_EXP_START) {\n        expansionFormStack.push(this._peek.type);\n      }\n\n      if (this._peek.type === TokenType.EXPANSION_CASE_EXP_END) {\n        if (lastOnStack(expansionFormStack, TokenType.EXPANSION_CASE_EXP_START)) {\n          expansionFormStack.pop();\n          if (expansionFormStack.length === 0) return exp;\n\n        } else {\n          this.errors.push(\n              TreeError.create(null, start.sourceSpan, `Invalid ICU message. Missing '}'.`));\n          return null;\n        }\n      }\n\n      if (this._peek.type === TokenType.EXPANSION_FORM_END) {\n        if (lastOnStack(expansionFormStack, TokenType.EXPANSION_FORM_START)) {\n          expansionFormStack.pop();\n        } else {\n          this.errors.push(\n              TreeError.create(null, start.sourceSpan, `Invalid ICU message. Missing '}'.`));\n          return null;\n        }\n      }\n\n      if (this._peek.type === TokenType.EOF) {\n        this.errors.push(\n            TreeError.create(null, start.sourceSpan, `Invalid ICU message. Missing '}'.`));\n        return null;\n      }\n\n      exp.push(this._advance());\n    }\n  }\n\n  private _consumeText(token: InterpolatedTextToken) {\n    const tokens = [token];\n    const startSpan = token.sourceSpan;\n    let text = token.parts[0];\n    if (text.length > 0 && text[0] === '\\n') {\n      const parent = this._getParentElement();\n      if (parent != null && parent.children.length === 0 &&\n          this.getTagDefinition(parent.name).ignoreFirstLf) {\n        text = text.substring(1);\n        tokens[0] = {type: token.type, sourceSpan: token.sourceSpan, parts: [text]} as typeof token;\n      }\n    }\n\n    while (this._peek.type === TokenType.INTERPOLATION || this._peek.type === TokenType.TEXT ||\n           this._peek.type === TokenType.ENCODED_ENTITY) {\n      token = this._advance();\n      tokens.push(token);\n      if (token.type === TokenType.INTERPOLATION) {\n        // For backward compatibility we decode HTML entities that appear in interpolation\n        // expressions. This is arguably a bug, but it could be a considerable breaking change to\n        // fix it. It should be addressed in a larger project to refactor the entire parser/lexer\n        // chain after View Engine has been removed.\n        text += token.parts.join('').replace(/&([^;]+);/g, decodeEntity);\n      } else if (token.type === TokenType.ENCODED_ENTITY) {\n        text += token.parts[0];\n      } else {\n        text += token.parts.join('');\n      }\n    }\n\n    if (text.length > 0) {\n      const endSpan = token.sourceSpan;\n      this._addToParent(new html.Text(\n          text,\n          new ParseSourceSpan(startSpan.start, endSpan.end, startSpan.fullStart, startSpan.details),\n          tokens));\n    }\n  }\n\n  private _closeVoidElement(): void {\n    const el = this._getParentElement();\n    if (el && this.getTagDefinition(el.name).isVoid) {\n      this._elementStack.pop();\n    }\n  }\n\n  private _consumeStartTag(startTagToken: TagOpenStartToken|IncompleteTagOpenToken) {\n    const [prefix, name] = startTagToken.parts;\n    const attrs: html.Attribute[] = [];\n    while (this._peek.type === TokenType.ATTR_NAME) {\n      attrs.push(this._consumeAttr(this._advance<AttributeNameToken>()));\n    }\n    const fullName = this._getElementFullName(prefix, name, this._getParentElement());\n    let selfClosing = false;\n    // Note: There could have been a tokenizer error\n    // so that we don't get a token for the end tag...\n    if (this._peek.type === TokenType.TAG_OPEN_END_VOID) {\n      this._advance();\n      selfClosing = true;\n      const tagDef = this.getTagDefinition(fullName);\n      if (!(tagDef.canSelfClose || getNsPrefix(fullName) !== null || tagDef.isVoid)) {\n        this.errors.push(TreeError.create(\n            fullName, startTagToken.sourceSpan,\n            `Only void and foreign elements can be self closed \"${startTagToken.parts[1]}\"`));\n      }\n    } else if (this._peek.type === TokenType.TAG_OPEN_END) {\n      this._advance();\n      selfClosing = false;\n    }\n    const end = this._peek.sourceSpan.fullStart;\n    const span = new ParseSourceSpan(\n        startTagToken.sourceSpan.start, end, startTagToken.sourceSpan.fullStart);\n    // Create a separate `startSpan` because `span` will be modified when there is an `end` span.\n    const startSpan = new ParseSourceSpan(\n        startTagToken.sourceSpan.start, end, startTagToken.sourceSpan.fullStart);\n    const el = new html.Element(fullName, attrs, [], span, startSpan, undefined);\n    this._pushElement(el);\n    if (selfClosing) {\n      // Elements that are self-closed have their `endSourceSpan` set to the full span, as the\n      // element start tag also represents the end tag.\n      this._popElement(fullName, span);\n    } else if (startTagToken.type === TokenType.INCOMPLETE_TAG_OPEN) {\n      // We already know the opening tag is not complete, so it is unlikely it has a corresponding\n      // close tag. Let's optimistically parse it as a full element and emit an error.\n      this._popElement(fullName, null);\n      this.errors.push(\n          TreeError.create(fullName, span, `Opening tag \"${fullName}\" not terminated.`));\n    }\n  }\n\n  private _pushElement(el: html.Element) {\n    const parentEl = this._getParentElement();\n\n    if (parentEl && this.getTagDefinition(parentEl.name).isClosedByChild(el.name)) {\n      this._elementStack.pop();\n    }\n\n    this._addToParent(el);\n    this._elementStack.push(el);\n  }\n\n  private _consumeEndTag(endTagToken: TagCloseToken) {\n    const fullName = this._getElementFullName(\n        endTagToken.parts[0], endTagToken.parts[1], this._getParentElement());\n\n    if (this.getTagDefinition(fullName).isVoid) {\n      this.errors.push(TreeError.create(\n          fullName, endTagToken.sourceSpan,\n          `Void elements do not have end tags \"${endTagToken.parts[1]}\"`));\n    } else if (!this._popElement(fullName, endTagToken.sourceSpan)) {\n      const errMsg = `Unexpected closing tag \"${\n          fullName}\". It may happen when the tag has already been closed by another tag. For more info see https://www.w3.org/TR/html5/syntax.html#closing-elements-that-have-implied-end-tags`;\n      this.errors.push(TreeError.create(fullName, endTagToken.sourceSpan, errMsg));\n    }\n  }\n\n  /**\n   * Closes the nearest element with the tag name `fullName` in the parse tree.\n   * `endSourceSpan` is the span of the closing tag, or null if the element does\n   * not have a closing tag (for example, this happens when an incomplete\n   * opening tag is recovered).\n   */\n  private _popElement(fullName: string, endSourceSpan: ParseSourceSpan|null): boolean {\n    let unexpectedCloseTagDetected = false;\n    for (let stackIndex = this._elementStack.length - 1; stackIndex >= 0; stackIndex--) {\n      const el = this._elementStack[stackIndex];\n      if (el.name === fullName) {\n        // Record the parse span with the element that is being closed. Any elements that are\n        // removed from the element stack at this point are closed implicitly, so they won't get\n        // an end source span (as there is no explicit closing element).\n        el.endSourceSpan = endSourceSpan;\n        el.sourceSpan.end = endSourceSpan !== null ? endSourceSpan.end : el.sourceSpan.end;\n\n        this._elementStack.splice(stackIndex, this._elementStack.length - stackIndex);\n        return !unexpectedCloseTagDetected;\n      }\n\n      if (!this.getTagDefinition(el.name).closedByParent) {\n        // Note that we encountered an unexpected close tag but continue processing the element\n        // stack so we can assign an `endSourceSpan` if there is a corresponding start tag for this\n        // end tag in the stack.\n        unexpectedCloseTagDetected = true;\n      }\n    }\n    return false;\n  }\n\n  private _consumeAttr(attrName: AttributeNameToken): html.Attribute {\n    const fullName = mergeNsAndName(attrName.parts[0], attrName.parts[1]);\n    let attrEnd = attrName.sourceSpan.end;\n\n    // Consume any quote\n    if (this._peek.type === TokenType.ATTR_QUOTE) {\n      this._advance();\n    }\n\n    // Consume the attribute value\n    let value = '';\n    const valueTokens: InterpolatedAttributeToken[] = [];\n    let valueStartSpan: ParseSourceSpan|undefined = undefined;\n    let valueEnd: ParseLocation|undefined = undefined;\n    // NOTE: We need to use a new variable `nextTokenType` here to hide the actual type of\n    // `_peek.type` from TS. Otherwise TS will narrow the type of `_peek.type` preventing it from\n    // being able to consider `ATTR_VALUE_INTERPOLATION` as an option. This is because TS is not\n    // able to see that `_advance()` will actually mutate `_peek`.\n    const nextTokenType = this._peek.type;\n    if (nextTokenType === TokenType.ATTR_VALUE_TEXT) {\n      valueStartSpan = this._peek.sourceSpan;\n      valueEnd = this._peek.sourceSpan.end;\n      while (this._peek.type === TokenType.ATTR_VALUE_TEXT ||\n             this._peek.type === TokenType.ATTR_VALUE_INTERPOLATION ||\n             this._peek.type === TokenType.ENCODED_ENTITY) {\n        const valueToken = this._advance<InterpolatedAttributeToken>();\n        valueTokens.push(valueToken);\n        if (valueToken.type === TokenType.ATTR_VALUE_INTERPOLATION) {\n          // For backward compatibility we decode HTML entities that appear in interpolation\n          // expressions. This is arguably a bug, but it could be a considerable breaking change to\n          // fix it. It should be addressed in a larger project to refactor the entire parser/lexer\n          // chain after View Engine has been removed.\n          value += valueToken.parts.join('').replace(/&([^;]+);/g, decodeEntity);\n        } else if (valueToken.type === TokenType.ENCODED_ENTITY) {\n          value += valueToken.parts[0];\n        } else {\n          value += valueToken.parts.join('');\n        }\n        valueEnd = attrEnd = valueToken.sourceSpan.end;\n      }\n    }\n\n    // Consume any quote\n    if (this._peek.type === TokenType.ATTR_QUOTE) {\n      const quoteToken = this._advance<AttributeQuoteToken>();\n      attrEnd = quoteToken.sourceSpan.end;\n    }\n\n    const valueSpan = valueStartSpan && valueEnd &&\n        new ParseSourceSpan(valueStartSpan.start, valueEnd, valueStartSpan.fullStart);\n    return new html.Attribute(\n        fullName, value,\n        new ParseSourceSpan(attrName.sourceSpan.start, attrEnd, attrName.sourceSpan.fullStart),\n        attrName.sourceSpan, valueSpan, valueTokens.length > 0 ? valueTokens : undefined,\n        undefined);\n  }\n\n  private _getParentElement(): html.Element|null {\n    return this._elementStack.length > 0 ? this._elementStack[this._elementStack.length - 1] : null;\n  }\n\n  private _addToParent(node: html.Node) {\n    const parent = this._getParentElement();\n    if (parent != null) {\n      parent.children.push(node);\n    } else {\n      this.rootNodes.push(node);\n    }\n  }\n\n  private _getElementFullName(prefix: string, localName: string, parentElement: html.Element|null):\n      string {\n    if (prefix === '') {\n      prefix = this.getTagDefinition(localName).implicitNamespacePrefix || '';\n      if (prefix === '' && parentElement != null) {\n        const parentTagName = splitNsName(parentElement.name)[1];\n        const parentTagDefinition = this.getTagDefinition(parentTagName);\n        if (!parentTagDefinition.preventNamespaceInheritance) {\n          prefix = getNsPrefix(parentElement.name);\n        }\n      }\n    }\n\n    return mergeNsAndName(prefix, localName);\n  }\n}\n\nfunction lastOnStack(stack: any[], element: any): boolean {\n  return stack.length > 0 && stack[stack.length - 1] === element;\n}\n\n/**\n * Decode the `entity` string, which we believe is the contents of an HTML entity.\n *\n * If the string is not actually a valid/known entity then just return the original `match` string.\n */\nfunction decodeEntity(match: string, entity: string): string {\n  if (NAMED_ENTITIES[entity] !== undefined) {\n    return NAMED_ENTITIES[entity] || match;\n  }\n  if (/^#x[a-f0-9]+$/i.test(entity)) {\n    return String.fromCodePoint(parseInt(entity.slice(2), 16));\n  }\n  if (/^#\\d+$/.test(entity)) {\n    return String.fromCodePoint(parseInt(entity.slice(1), 10));\n  }\n  return match;\n}\n"]} |
---|