• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

nette / latte / 22369510456

24 Feb 2026 08:49PM UTC coverage: 94.841% (+0.02%) from 94.825%
22369510456

push

github

dg
phpstan

134 of 145 new or added lines in 39 files covered. (92.41%)

36 existing lines in 13 files now uncovered.

5533 of 5834 relevant lines covered (94.84%)

0.95 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

98.04
/src/Latte/Compiler/TemplateParserHtml.php
1
<?php declare(strict_types=1);
2

3
/**
4
 * This file is part of the Latte (https://latte.nette.org)
5
 * Copyright (c) 2008 David Grudl (https://davidgrudl.com)
6
 */
7

8
namespace Latte\Compiler;
9

10
use Latte;
11
use Latte\CompileException;
12
use Latte\Compiler\Nodes\AreaNode;
13
use Latte\Compiler\Nodes\FragmentNode;
14
use Latte\Compiler\Nodes\Html;
15
use Latte\ContentType;
16
use Latte\Helpers;
17
use Latte\SecurityViolationException;
18
use function array_keys, array_pop, end, implode, in_array, key, preg_replace, str_starts_with, strlen, substr;
19

20

21
/**
22
 * Template parser extension for HTML.
23
 */
24
final class TemplateParserHtml
25
{
26
        private ?Html\ElementNode $element = null;
27

28
        /** @var array{string, ?Nodes\Php\ExpressionNode}|null */
29
        private ?array $endName = null;
30

31
        /** @var \WeakMap<Html\ElementNode, object{tag: mixed, textualName: string, unclosedTags?: array<string>}> */
32
        private \WeakMap $elementData;
33

34

35
        public function __construct(
1✔
36
                private readonly TemplateParser $parser,
37
                /** @var array<string, \Closure(Tag, TemplateParser): (Node|\Generator|void)> */
38
                private readonly array $attrParsers,
39
        ) {
40
                $this->elementData = new \WeakMap;
1✔
41
        }
1✔
42

43

44
        public function getElement(): ?Html\ElementNode
45
        {
46
                return $this->element;
1✔
47
        }
48

49

50
        public function inTextResolve(): ?Node
51
        {
52
                $stream = $this->parser->getStream();
1✔
53
                $token = $stream->peek();
1✔
54
                return match ($token->type) {
1✔
55
                        Token::Html_TagOpen => $this->parseTag(),
1✔
56
                        Token::Html_CommentOpen => $this->parseComment(),
1✔
57
                        Token::Html_BogusOpen => $this->parseBogusTag(),
1✔
58
                        default => $this->parser->inTextResolve(),
1✔
59
                };
60
        }
61

62

63
        public function inTagResolve(FragmentNode $fragment): ?Node
1✔
64
        {
65
                $stream = $this->parser->getStream();
1✔
66
                $token = $stream->peek();
1✔
67
                return match ($token->type) {
1✔
68
                        Token::Html_Name => str_starts_with($token->text, TemplateLexer::NPrefix)
1✔
69
                                ? $this->parseNAttribute()
1✔
70
                                : $this->parseAttribute($fragment),
1✔
71
                        Token::Latte_TagOpen => $this->parseAttribute($fragment),
1✔
72
                        Token::Whitespace => $this->parseAttributeWhitespace(),
1✔
73
                        Token::Html_TagClose => null,
1✔
74
                        default => $this->parser->inTextResolve(),
1✔
75
                };
76
        }
77

78

79
        private function parseTag(): ?Node
80
        {
81
                $stream = $this->parser->getStream();
1✔
82
                $lexer = $this->parser->getLexer();
1✔
83
                $lexer->pushState(TemplateLexer::StateHtmlTag);
1✔
84
                $closing = $stream->tryPeek(1)?->is(Token::Slash);
1✔
85
                $lexer->popState();
1✔
86
                if (!$closing) {
1✔
87
                        return $this->parseElement();
1✔
88
                }
89

90
                if ($this->element
1✔
91
                        && $this->parser->peekTag() === $this->elementData[$this->element]->tag // is directly in the element
1✔
92
                ) {
93
                        $save = $stream->getIndex();
1✔
94
                        $this->endName = [$endText] = $this->parseEndTag();
1✔
95
                        if ($this->element->is($endText) || $this->elementData[$this->element]->textualName === $endText) {
1✔
96
                                return null; // go to parseElement() one level up to close the element
1✔
97
                        }
98
                        $stream->seek($save);
1✔
99
                        if (!in_array($endText, $this->elementData[$this->element]->unclosedTags ?? [], strict: true)) {
1✔
100
                                return null; // go to parseElement() one level up to collapse
1✔
101
                        }
102
                }
103

104
                if ($this->parser->strict) {
1✔
105
                        $stream->throwUnexpectedException(excerpt: '/');
1✔
106
                }
107
                return $this->parseBogusEndTag();
1✔
108
        }
109

110

111
        private function parseElement(): Node
112
        {
113
                $res = new FragmentNode;
1✔
114
                $res->append($this->extractIndentation());
1✔
115
                $res->append($this->parseStartTag($this->element));
1✔
116
                $elem = $this->element;
1✔
117

118
                $stream = $this->parser->getStream();
1✔
119
                $void = $this->resolveVoidness($elem);
1✔
120
                $attrs = $this->prepareNAttrs($elem->nAttributes, $void);
1✔
121
                $outerNodes = $this->openNAttrNodes($attrs[Tag::PrefixNone] ?? []);
1✔
122
                $tagNodes = $this->openNAttrNodes($attrs[Tag::PrefixTag] ?? []);
1✔
123
                if ($tagNodes) {
1✔
124
                        $elem->dynamicTag = $this->finishNAttrNodes($elem->dynamicTag ?? new Nodes\Html\TagNode($elem), $tagNodes);
1✔
125
                }
126

127
                if (!$void) {
1✔
128
                        if ($elem->isRawText()) {
1✔
129
                                $this->parser->getLexer()->pushState(TemplateLexer::StateHtmlRawText, $elem->name);
1✔
130
                        }
131
                        $content = new FragmentNode;
1✔
132
                        if ($token = $stream->tryConsume(Token::Newline)) {
1✔
133
                                $content->append(new Nodes\TextNode($token->text, $token->position));
1✔
134
                        }
135

136
                        $innerNodes = $this->openNAttrNodes($attrs[Tag::PrefixInner] ?? []);
1✔
137
                        $this->elementData[$elem]->tag = $this->parser->peekTag();
1✔
138
                        $frag = $this->parser->parseFragment($this->inTextResolve(...));
1✔
139
                        $content->append($this->finishNAttrNodes($frag, $innerNodes));
1✔
140
                        if ($elem->isRawText()) {
1✔
141
                                $this->parser->getLexer()->popState();
1✔
142
                        }
143

144
                        [$endText, $endVariable] = $this->endName ?? [null, null];
1✔
145
                        $this->endName = null;
1✔
146
                        if ($endText && ($this->element->is($endText) || $this->elementData[$this->element]->textualName === $endText)) {
1✔
147
                                $elem->content = $content;
1✔
148
                                $elem->content->append($this->extractIndentation());
1✔
149

150
                        } elseif ($outerNodes || $innerNodes
1✔
151
                                || $elem->dynamicTag
1✔
152
                                || $this->parser->strict
1✔
153
                                || $endVariable
1✔
154
                                || $elem->isRawText()
1✔
155
                        ) {
156
                                $stream->throwUnexpectedException(
1✔
157
                                        addendum: ", expecting </{$this->elementData[$elem]->textualName}> for element started $elem->position",
1✔
158
                                        excerpt: $endText ? "/{$endText}>" : ($stream->tryPeek(1)->text ?? '') . ($stream->tryPeek(2)->text ?? ''),
1✔
159
                                );
160
                        } else { // element collapsed to tags
161
                                $res->append($content);
1✔
162
                                $this->element = $elem->parent;
1✔
163
                                if ($this->element && !$stream->is(Token::Html_TagOpen)) {
1✔
164
                                        $this->elementData[$this->element]->unclosedTags[] = $elem->name;
1✔
165
                                }
166
                                return $res;
1✔
167
                        }
168
                }
169

170
                if ($token = $stream->tryConsume(Token::Newline)) {
1✔
171
                        $res->append(new Nodes\TextNode($token->text, $token->position));
1✔
172
                }
173

174
                $res = $this->finishNAttrNodes($res, $outerNodes);
1✔
175
                $this->element = $elem->parent;
1✔
176
                return $res;
1✔
177
        }
178

179

180
        private function extractIndentation(): AreaNode
181
        {
182
                if ($this->parser->lastIndentation) {
1✔
183
                        $dolly = clone $this->parser->lastIndentation;
1✔
184
                        $this->parser->lastIndentation->content = '';
1✔
185
                        return $dolly;
1✔
186
                } else {
187
                        return new Nodes\NopNode;
1✔
188
                }
189
        }
190

191

192
        /** @param-out Html\ElementNode $elem */
193
        private function parseStartTag(&$elem = null): Html\ElementNode
194
        {
195
                $stream = $this->parser->getStream();
1✔
196
                $lexer = $this->parser->getLexer();
1✔
197
                $openToken = $stream->consume(Token::Html_TagOpen);
1✔
198
                $lexer->pushState(TemplateLexer::StateHtmlTag);
1✔
199

200
                [$textual, $variable] = $this->parseTagName($this->parser->strict);
1✔
201
                if (($this->parser->strict || $variable)
1✔
202
                        && !$stream->is(Token::Whitespace, Token::Slash, Token::Html_TagClose)
1✔
203
                ) {
204
                        throw $stream->throwUnexpectedException();
×
205
                }
206

207
                $this->parser->lastIndentation = null;
1✔
208
                $this->parser->inHead = false;
1✔
209
                $elem = new Html\ElementNode(
1✔
210
                        name: $variable ? '' : $textual,
1✔
211
                        position: $openToken->position,
1✔
212
                        parent: $this->element,
1✔
213
                        contentType: $this->parser->getContentType(),
1✔
214
                );
215
                $this->elementData[$elem] = (object) [
1✔
216
                        'tag' => $this->parser->peekTag(),
1✔
217
                        'textualName' => $textual,
1✔
218
                ];
219
                $elem->attributes = $this->parser->parseFragment($this->inTagResolve(...));
1✔
220
                $elem->selfClosing = (bool) $stream->tryConsume(Token::Slash);
1✔
221
                if ($variable) {
1✔
222
                        $elem->dynamicTag = new Nodes\Html\TagNode($elem, $variable);
1✔
223
                }
224
                $stream->consume(Token::Html_TagClose);
1✔
225
                $lexer->popState();
1✔
226
                return $elem;
1✔
227
        }
228

229

230
        /** @return array{string, ?Nodes\Php\ExpressionNode} */
231
        private function parseEndTag(): array
232
        {
233
                $stream = $this->parser->getStream();
1✔
234
                $lexer = $this->parser->getLexer();
1✔
235
                $stream->consume(Token::Html_TagOpen);
1✔
236
                $lexer->pushState(TemplateLexer::StateHtmlTag);
1✔
237
                $stream->consume(Token::Slash);
1✔
238
                if (isset($this->element->nAttributes['syntax'])) {  // hardcoded
1✔
239
                        $lexer->popSyntax();
1✔
240
                }
241
                $name = $this->parseTagName();
1✔
242
                $stream->tryConsume(Token::Whitespace);
1✔
243
                $stream->consume(Token::Html_TagClose);
1✔
244
                $lexer->popState();
1✔
245
                return $name;
1✔
246
        }
247

248

249
        /** @return array{string, ?Nodes\Php\ExpressionNode} */
250
        private function parseTagName(bool $strict = true): array
1✔
251
        {
252
                $variable = null;
1✔
253
                $text = '';
1✔
254
                $parts = [];
1✔
255
                $stream = $this->parser->getStream();
1✔
256
                do {
257
                        if ($stream->is(Token::Latte_TagOpen)) {
1✔
258
                                $save = $stream->getIndex();
1✔
259
                                $statement = $this->parser->parseLatteStatement($this->inTagResolve(...));
1✔
260
                                if (!$statement instanceof Nodes\PrintNode) {
1✔
261
                                        if (!$parts || $strict) {
1✔
262
                                                throw new CompileException('Only expression can be used as a HTML tag name.', $statement?->position);
1✔
263
                                        }
264
                                        $stream->seek($save);
1✔
265
                                        break;
1✔
266
                                }
267
                                $parts[] = $statement->expression;
1✔
268
                                $save -= $stream->getIndex();
1✔
269
                                while ($save < 0) {
1✔
270
                                        $text .= $stream->peek($save++)->text;
1✔
271
                                }
272
                                $variable = true;
1✔
273

274
                        } elseif ($token = $stream->tryConsume(Token::Html_Name)) {
1✔
275
                                $parts[] = new Latte\Compiler\Nodes\Php\Scalar\StringNode($token->text, $token->position);
1✔
276
                                $text .= $token->text;
1✔
277

278
                        } elseif (!$parts) {
1✔
UNCOV
279
                                throw $stream->throwUnexpectedException([Token::Html_Name, Token::Latte_TagOpen]);
×
280
                        } else {
281
                                break;
1✔
282
                        }
283
                } while (true);
1✔
284

285
                $variable = $variable
1✔
286
                        ? Latte\Compiler\Nodes\Php\Expression\BinaryOpNode::nest('.', ...$parts)
1✔
287
                        : null;
1✔
288
                return [$text, $variable];
1✔
289
        }
290

291

292
        private function parseBogusEndTag(): Html\BogusTagNode
293
        {
294
                $stream = $this->parser->getStream();
1✔
295
                $lexer = $this->parser->getLexer();
1✔
296
                $openToken = $stream->consume(Token::Html_TagOpen);
1✔
297
                $lexer->pushState(TemplateLexer::StateHtmlTag);
1✔
298
                $this->parser->lastIndentation = null;
1✔
299
                $this->parser->inHead = false;
1✔
300
                $node = new Html\BogusTagNode(
1✔
301
                        openDelimiter: $openToken->text . $stream->consume(Token::Slash)->text . $stream->consume(Token::Html_Name)->text,
1✔
302
                        content: new Nodes\TextNode($stream->tryConsume(Token::Whitespace)->text ?? ''),
1✔
303
                        endDelimiter: $stream->consume(Token::Html_TagClose)->text,
1✔
304
                        position: $openToken->position,
1✔
305
                );
306
                $lexer->popState();
1✔
307
                return $node;
1✔
308
        }
309

310

311
        private function parseBogusTag(): Html\BogusTagNode
312
        {
313
                $stream = $this->parser->getStream();
1✔
314
                $lexer = $this->parser->getLexer();
1✔
315
                $openToken = $stream->consume(Token::Html_BogusOpen);
1✔
316
                $lexer->pushState(TemplateLexer::StateHtmlBogus);
1✔
317
                $this->parser->lastIndentation = null;
1✔
318
                $this->parser->inHead = false;
1✔
319
                $content = $this->parser->parseFragment($this->parser->inTextResolve(...));
1✔
320
                $lexer->popState();
1✔
321
                return new Html\BogusTagNode(
1✔
322
                        openDelimiter: $openToken->text,
1✔
323
                        content: $content,
324
                        endDelimiter: $stream->consume(Token::Html_TagClose)->text,
1✔
325
                        position: $openToken->position,
1✔
326
                );
327
        }
328

329

330
        private function resolveVoidness(Html\ElementNode $elem): bool
1✔
331
        {
332
                if ($elem->contentType !== ContentType::Html) {
1✔
333
                        return $elem->selfClosing;
1✔
334
                } elseif (Latte\Runtime\HtmlHelpers::isVoidElement($elem->name)) {
1✔
335
                        return true;
1✔
336
                } elseif ($elem->selfClosing) { // auto-correct
1✔
337
                        $elem->content = new Nodes\NopNode;
1✔
338
                        $elem->selfClosing = false;
1✔
339
                        $last = end($elem->attributes->children);
1✔
340
                        if ($last instanceof Nodes\TextNode && $last->isWhitespace()) {
1✔
341
                                array_pop($elem->attributes->children);
1✔
342
                        }
343
                        return true;
1✔
344
                }
345

346
                return $elem->selfClosing;
1✔
347
        }
348

349

350
        private function parseAttributeWhitespace(): Node
351
        {
352
                $stream = $this->parser->getStream();
1✔
353
                $token = $stream->consume(Token::Whitespace);
1✔
354
                return $stream->is(Token::Html_Name) && str_starts_with($stream->peek()->text, TemplateLexer::NPrefix)
1✔
355
                        ? new Nodes\NopNode
1✔
356
                        : new Nodes\TextNode($token->text, $token->position);
1✔
357
        }
358

359

360
        private function parseAttribute(FragmentNode $fragment): ?Node
1✔
361
        {
362
                $stream = $this->parser->getStream();
1✔
363
                if ($stream->is(Token::Latte_TagOpen)) {
1✔
364
                        $name = $this->parser->parseLatteStatement();
1✔
365
                        if (!$name instanceof Nodes\PrintNode) {
1✔
366
                                return $name; // value like '<span {if true}attr1=val{/if}>'
1✔
367
                        }
368
                } else {
369
                        $name = $this->parser->parseText();
1✔
370
                }
371

372
                [$value, $quote] = $this->parseAttributeValue() ?? [null, null];
1✔
373
                if ($name instanceof Nodes\TextNode && $value instanceof Nodes\PrintNode && $value->modifier->escape) {
1✔
374
                        if (($indent = end($fragment->children)) instanceof Nodes\TextNode && $indent->isWhitespace()) {
1✔
375
                                array_pop($fragment->children);
1✔
376
                        }
377
                        $value->modifier->escape = false;
1✔
378
                        return new Html\ExpressionAttributeNode(
1✔
379
                                name: $name->content,
1✔
380
                                value: $value->expression,
1✔
381
                                modifier: $value->modifier,
1✔
382
                                indentation: $indent->content ?? null,
1✔
383
                                position: $name->position,
1✔
384
                        );
385
                }
386

387
                return new Html\AttributeNode(
1✔
388
                        name: $name,
1✔
389
                        value: $value,
390
                        quote: $quote,
391
                        position: $name->position,
1✔
392
                );
393
        }
394

395

396
        /** @return ?array{AreaNode|Nodes\PrintNode, ?string} */
397
        private function parseAttributeValue(): ?array
398
        {
399
                $stream = $this->parser->getStream();
1✔
400
                $save = $stream->getIndex();
1✔
401
                $this->consumeIgnored();
1✔
402
                if (!$stream->tryConsume(Token::Equals)) {
1✔
403
                        $stream->seek($save);
1✔
404
                        return null;
1✔
405
                }
406

407
                $this->consumeIgnored();
1✔
408
                if ($quoteToken = $stream->tryConsume(Token::Quote)) {
1✔
409
                        $lexer = $this->parser->getLexer();
1✔
410
                        $lexer->pushState(TemplateLexer::StateHtmlQuotedValue, $quoteToken->text);
1✔
411
                        $value = $this->parser->parseFragment($this->parser->inTextResolve(...))->simplify(allowsNull: false);
1✔
412
                        assert($value !== null);
413
                        $stream->tryConsume(Token::Quote) || $stream->throwUnexpectedException([$quoteToken->text], addendum: ", end of HTML attribute started $quoteToken->position");
1✔
414
                        $lexer->popState();
1✔
415
                        return [$value, $quoteToken->text];
1✔
416
                }
417

418
                $value = $this->parser->parseFragment(
1✔
419
                        fn() => match ($stream->peek()->type) {
1✔
420
                                Token::Html_Name => $this->parser->parseText(),
1✔
421
                                Token::Latte_TagOpen => $this->parser->parseLatteStatement(),
1✔
422
                                Token::Latte_CommentOpen => $this->parser->parseLatteComment(),
423
                                default => null,
1✔
424
                        },
1✔
425
                )->simplify() ?? $stream->throwUnexpectedException();
1✔
426
                return [$value, null];
1✔
427
        }
428

429

430
        private function parseNAttribute(): Nodes\TextNode
431
        {
432
                assert($this->element !== null);
433
                $stream = $this->parser->getStream();
1✔
434
                $nameToken = $stream->consume(Token::Html_Name);
1✔
435
                $save = $stream->getIndex();
1✔
436
                $pos = $stream->peek()->position;
1✔
437
                $name = substr($nameToken->text, strlen(TemplateLexer::NPrefix));
1✔
438
                if ($this->parser->peekTag() !== $this->elementData[$this->element]->tag) {
1✔
439
                        throw new CompileException("Attribute n:$name must not appear inside {tags}", $nameToken->position);
1✔
440

441
                } elseif (isset($this->element->nAttributes[$name])) {
1✔
442
                        throw new CompileException("Found multiple attributes n:$name.", $nameToken->position);
1✔
443
                }
444

445
                $this->consumeIgnored();
1✔
446
                if ($stream->tryConsume(Token::Equals)) {
1✔
447
                        $lexer = $this->parser->getLexer();
1✔
448
                        $this->consumeIgnored();
1✔
449
                        if ($quoteToken = $stream->tryConsume(Token::Quote)) {
1✔
450
                                $lexer->pushState(TemplateLexer::StateHtmlQuotedNAttrValue, $quoteToken->text);
1✔
451
                                $valueToken = $stream->tryConsume(Token::Text);
1✔
452
                                $pos = $stream->peek()->position;
1✔
453
                                $stream->tryConsume(Token::Quote) || $stream->throwUnexpectedException([$quoteToken->text], addendum: ", end of n:attribute started $quoteToken->position");
1✔
454
                                $lexer->popState();
1✔
455
                                $tokens = $valueToken ? (new TagLexer)->tokenize($valueToken->text, $valueToken->position) : null;
1✔
456

457
                        } elseif ($openToken = $stream->tryConsume(Token::Latte_TagOpen)) {
1✔
458
                                $lexer->pushState(TemplateLexer::StateLatteContent);
1✔
459
                                $tokens = $this->parser->consumeTag();
1✔
460
                                $stream->tryConsume(Token::Latte_TagClose) || $stream->throwUnexpectedException([Token::Latte_TagClose], addendum: " started $openToken->position");
1✔
461
                                $lexer->popState();
1✔
462

463
                        } else {
464
                                $valueToken = $stream->consume(Token::Html_Name);
1✔
465
                                $tokens = (new TagLexer)->tokenize($valueToken->text, $valueToken->position);
1✔
466
                        }
467
                } else {
468
                        $stream->seek($save);
1✔
469
                }
470
                $tokens ??= [new Token(Token::End, '', $pos)];
1✔
471

472
                assert($nameToken->position !== null);
473
                assert($this->element !== null);
474
                $this->element->nAttributes[$name] = new Tag(
1✔
475
                        name: preg_replace('~(inner-|tag-|)~', '', $name),
1✔
476
                        tokens: $tokens,
477
                        position: $nameToken->position,
1✔
478
                        prefix: $this->getPrefix($name),
1✔
479
                        inTag: true,
1✔
480
                        htmlElement: $this->element,
1✔
481
                        nAttribute: $node = new Nodes\TextNode(''),
1✔
482
                );
483
                return $node;
1✔
484
        }
485

486

487
        private function parseComment(): Html\CommentNode
488
        {
489
                $lexer = $this->parser->getLexer();
1✔
490
                $this->parser->lastIndentation = null;
1✔
491
                $this->parser->inHead = false;
1✔
492
                $lexer->pushState(TemplateLexer::StateHtmlComment);
1✔
493
                $stream = $this->parser->getStream();
1✔
494
                $openToken = $stream->consume(Token::Html_CommentOpen);
1✔
495
                $node = new Html\CommentNode(
1✔
496
                        position: $openToken->position,
1✔
497
                        content: $this->parser->parseFragment($this->parser->inTextResolve(...)),
1✔
498
                );
499
                $stream->tryConsume(Token::Html_CommentClose) || $stream->throwUnexpectedException([Token::Html_CommentClose], addendum: " started $openToken->position");
1✔
500
                $lexer->popState();
1✔
501
                return $node;
1✔
502
        }
503

504

505
        private function consumeIgnored(): void
506
        {
507
                $stream = $this->parser->getStream();
1✔
508
                $lexer = $this->parser->getLexer();
1✔
509
                do {
510
                        if ($stream->tryConsume(Token::Whitespace)) {
1✔
511
                                continue;
1✔
512
                        }
513
                        if ($stream->tryConsume(Token::Latte_CommentOpen)) {
1✔
514
                                $lexer->pushState(TemplateLexer::StateLatteComment);
1✔
515
                                $stream->consume(Token::Text);
1✔
516
                                $stream->consume(Token::Latte_CommentClose);
1✔
517
                                $lexer->popState();
1✔
518
                                $stream->tryConsume(Token::Newline);
1✔
519
                                continue;
1✔
520
                        }
521
                        return;
1✔
522
                } while (true);
1✔
523
        }
524

525

526
        /**
527
         * @param  array<Tag>  $attrs
528
         * @return array<string, list<Tag>>
529
         */
530
        private function prepareNAttrs(array $attrs, bool $void): array
1✔
531
        {
532
                $res = [];
1✔
533
                foreach ($this->attrParsers as $name => $foo) {
1✔
534
                        if ($tag = $attrs[$name] ?? null) {
1✔
535
                                $prefix = $this->getPrefix($name);
1✔
536
                                if (!$prefix || !$void) {
1✔
537
                                        $res[$prefix][] = $tag;
1✔
538
                                        unset($attrs[$name]);
1✔
539
                                }
540
                        }
541
                }
542

543
                if ($attrs) {
1✔
544
                        $hint = Helpers::getSuggestion(array_keys($this->attrParsers), $k = key($attrs));
1✔
545
                        throw new CompileException('Unexpected attribute n:'
1✔
546
                                . ($hint ? "$k, did you mean n:$hint?" : implode(' and n:', array_keys($attrs))), $attrs[$k]->position);
1✔
547
                }
548

549
                return $res;
1✔
550
        }
551

552

553
        /**
554
         * @param  array<Tag>  $toOpen
555
         * @return array<array{\Generator, Tag}>
556
         */
557
        private function openNAttrNodes(array $toOpen): array
1✔
558
        {
559
                $toClose = [];
1✔
560
                foreach ($toOpen as $tag) {
1✔
561
                        $parser = $this->getAttrParser($tag->name, $tag->position);
1✔
562
                        $this->parser->pushTag($tag);
1✔
563
                        $res = $parser($tag, $this->parser);
1✔
564
                        if ($res instanceof \Generator && $res->valid()) {
1✔
565
                                $toClose[] = [$res, $tag];
1✔
566

567
                        } elseif ($res instanceof AreaNode) {
1✔
568
                                $this->parser->ensureIsConsumed($tag);
1✔
569
                                $res->position = $tag->position;
1✔
570
                                $tag->replaceNAttribute($res);
1✔
571
                                $this->parser->popTag();
1✔
572

573
                        } elseif (!$res) {
1✔
574
                                $this->parser->ensureIsConsumed($tag);
1✔
575
                                $this->parser->popTag();
1✔
576

577
                        } else {
UNCOV
578
                                throw new CompileException("Unexpected value returned by {$tag->getNotation()} parser.", $tag->position);
×
579
                        }
580
                }
581

582
                return $toClose;
1✔
583
        }
584

585

586
        /** @param  array<array{\Generator, Tag}>  $toClose */
587
        private function finishNAttrNodes(AreaNode $node, array $toClose): AreaNode
1✔
588
        {
589
                while ([$gen, $tag] = array_pop($toClose)) {
1✔
590
                        $gen->send([$node, null]);
1✔
591
                        $node = $gen->getReturn();
1✔
592
                        $node->position = $tag->position;
1✔
593
                        $this->parser->popTag();
1✔
594
                        $this->parser->ensureIsConsumed($tag);
1✔
595
                }
596

597
                return $node;
1✔
598
        }
599

600

601
        /** @return \Closure(Tag, TemplateParser): (Node|\Generator|void) */
602
        private function getAttrParser(string $name, Position $pos): \Closure
1✔
603
        {
604
                if (!isset($this->attrParsers[$name])) {
1✔
UNCOV
605
                        $hint = ($t = Helpers::getSuggestion(array_keys($this->attrParsers), $name))
×
UNCOV
606
                                ? ", did you mean n:$t?"
×
UNCOV
607
                                : '';
×
UNCOV
608
                        throw new CompileException("Unknown n:{$name}{$hint}", $pos);
×
609
                } elseif (!$this->parser->isTagAllowed($name)) {
1✔
610
                        throw new SecurityViolationException("Attribute n:$name is not allowed", $pos);
1✔
611
                }
612
                return $this->attrParsers[$name];
1✔
613
        }
614

615

616
        private function getPrefix(string $name): string
1✔
617
        {
618
                return match (true) {
619
                        str_starts_with($name, 'inner-') => Tag::PrefixInner,
1✔
620
                        str_starts_with($name, 'tag-') => Tag::PrefixTag,
1✔
621
                        default => Tag::PrefixNone,
1✔
622
                };
623
        }
624
}
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc