1
2
3
4
5
6
7
8
9 package parse
10
11 import (
12 "bytes"
13 "fmt"
14 "runtime"
15 "strconv"
16 "strings"
17 )
18
19
20 type Tree struct {
21 Name string
22 ParseName string
23 Root *ListNode
24 Mode Mode
25 text string
26
27 funcs []map[string]any
28 lex *lexer
29 token [3]item
30 peekCount int
31 vars []string
32 treeSet map[string]*Tree
33 actionLine int
34 rangeDepth int
35 }
36
37
38 type Mode uint
39
40 const (
41 ParseComments Mode = 1 << iota
42 SkipFuncCheck
43 )
44
45
46 func (t *Tree) Copy() *Tree {
47 if t == nil {
48 return nil
49 }
50 return &Tree{
51 Name: t.Name,
52 ParseName: t.ParseName,
53 Root: t.Root.CopyList(),
54 text: t.text,
55 }
56 }
57
58
59
60
61
62 func Parse(name, text, leftDelim, rightDelim string, funcs ...map[string]any) (map[string]*Tree, error) {
63 treeSet := make(map[string]*Tree)
64 t := New(name)
65 t.text = text
66 _, err := t.Parse(text, leftDelim, rightDelim, treeSet, funcs...)
67 return treeSet, err
68 }
69
70
71 func (t *Tree) next() item {
72 if t.peekCount > 0 {
73 t.peekCount--
74 } else {
75 t.token[0] = t.lex.nextItem()
76 }
77 return t.token[t.peekCount]
78 }
79
80
81 func (t *Tree) backup() {
82 t.peekCount++
83 }
84
85
86
87 func (t *Tree) backup2(t1 item) {
88 t.token[1] = t1
89 t.peekCount = 2
90 }
91
92
93
94 func (t *Tree) backup3(t2, t1 item) {
95 t.token[1] = t1
96 t.token[2] = t2
97 t.peekCount = 3
98 }
99
100
101 func (t *Tree) peek() item {
102 if t.peekCount > 0 {
103 return t.token[t.peekCount-1]
104 }
105 t.peekCount = 1
106 t.token[0] = t.lex.nextItem()
107 return t.token[0]
108 }
109
110
111 func (t *Tree) nextNonSpace() (token item) {
112 for {
113 token = t.next()
114 if token.typ != itemSpace {
115 break
116 }
117 }
118 return token
119 }
120
121
122 func (t *Tree) peekNonSpace() item {
123 token := t.nextNonSpace()
124 t.backup()
125 return token
126 }
127
128
129
130
131 func New(name string, funcs ...map[string]any) *Tree {
132 return &Tree{
133 Name: name,
134 funcs: funcs,
135 }
136 }
137
138
139
140
141 func (t *Tree) ErrorContext(n Node) (location, context string) {
142 pos := int(n.Position())
143 tree := n.tree()
144 if tree == nil {
145 tree = t
146 }
147 text := tree.text[:pos]
148 byteNum := strings.LastIndex(text, "\n")
149 if byteNum == -1 {
150 byteNum = pos
151 } else {
152 byteNum++
153 byteNum = pos - byteNum
154 }
155 lineNum := 1 + strings.Count(text, "\n")
156 context = n.String()
157 return fmt.Sprintf("%s:%d:%d", tree.ParseName, lineNum, byteNum), context
158 }
159
160
161 func (t *Tree) errorf(format string, args ...any) {
162 t.Root = nil
163 format = fmt.Sprintf("template: %s:%d: %s", t.ParseName, t.token[0].line, format)
164 panic(fmt.Errorf(format, args...))
165 }
166
167
168 func (t *Tree) error(err error) {
169 t.errorf("%s", err)
170 }
171
172
173 func (t *Tree) expect(expected itemType, context string) item {
174 token := t.nextNonSpace()
175 if token.typ != expected {
176 t.unexpected(token, context)
177 }
178 return token
179 }
180
181
182 func (t *Tree) expectOneOf(expected1, expected2 itemType, context string) item {
183 token := t.nextNonSpace()
184 if token.typ != expected1 && token.typ != expected2 {
185 t.unexpected(token, context)
186 }
187 return token
188 }
189
190
191 func (t *Tree) unexpected(token item, context string) {
192 if token.typ == itemError {
193 extra := ""
194 if t.actionLine != 0 && t.actionLine != token.line {
195 extra = fmt.Sprintf(" in action started at %s:%d", t.ParseName, t.actionLine)
196 if strings.HasSuffix(token.val, " action") {
197 extra = extra[len(" in action"):]
198 }
199 }
200 t.errorf("%s%s", token, extra)
201 }
202 t.errorf("unexpected %s in %s", token, context)
203 }
204
205
206 func (t *Tree) recover(errp *error) {
207 e := recover()
208 if e != nil {
209 if _, ok := e.(runtime.Error); ok {
210 panic(e)
211 }
212 if t != nil {
213 t.stopParse()
214 }
215 *errp = e.(error)
216 }
217 }
218
219
220 func (t *Tree) startParse(funcs []map[string]any, lex *lexer, treeSet map[string]*Tree) {
221 t.Root = nil
222 t.lex = lex
223 t.vars = []string{"$"}
224 t.funcs = funcs
225 t.treeSet = treeSet
226 lex.options = lexOptions{
227 emitComment: t.Mode&ParseComments != 0,
228 breakOK: !t.hasFunction("break"),
229 continueOK: !t.hasFunction("continue"),
230 }
231 }
232
233
234 func (t *Tree) stopParse() {
235 t.lex = nil
236 t.vars = nil
237 t.funcs = nil
238 t.treeSet = nil
239 }
240
241
242
243
244
245 func (t *Tree) Parse(text, leftDelim, rightDelim string, treeSet map[string]*Tree, funcs ...map[string]any) (tree *Tree, err error) {
246 defer t.recover(&err)
247 t.ParseName = t.Name
248 lexer := lex(t.Name, text, leftDelim, rightDelim)
249 t.startParse(funcs, lexer, treeSet)
250 t.text = text
251 t.parse()
252 t.add()
253 t.stopParse()
254 return t, nil
255 }
256
257
258 func (t *Tree) add() {
259 tree := t.treeSet[t.Name]
260 if tree == nil || IsEmptyTree(tree.Root) {
261 t.treeSet[t.Name] = t
262 return
263 }
264 if !IsEmptyTree(t.Root) {
265 t.errorf("template: multiple definition of template %q", t.Name)
266 }
267 }
268
269
270 func IsEmptyTree(n Node) bool {
271 switch n := n.(type) {
272 case nil:
273 return true
274 case *ActionNode:
275 case *CommentNode:
276 return true
277 case *IfNode:
278 case *ListNode:
279 for _, node := range n.Nodes {
280 if !IsEmptyTree(node) {
281 return false
282 }
283 }
284 return true
285 case *RangeNode:
286 case *TemplateNode:
287 case *TextNode:
288 return len(bytes.TrimSpace(n.Text)) == 0
289 case *WithNode:
290 default:
291 panic("unknown node: " + n.String())
292 }
293 return false
294 }
295
296
297
298
299 func (t *Tree) parse() {
300 t.Root = t.newList(t.peek().pos)
301 for t.peek().typ != itemEOF {
302 if t.peek().typ == itemLeftDelim {
303 delim := t.next()
304 if t.nextNonSpace().typ == itemDefine {
305 newT := New("definition")
306 newT.text = t.text
307 newT.Mode = t.Mode
308 newT.ParseName = t.ParseName
309 newT.startParse(t.funcs, t.lex, t.treeSet)
310 newT.parseDefinition()
311 continue
312 }
313 t.backup2(delim)
314 }
315 switch n := t.textOrAction(); n.Type() {
316 case nodeEnd, nodeElse:
317 t.errorf("unexpected %s", n)
318 default:
319 t.Root.append(n)
320 }
321 }
322 }
323
324
325
326
327 func (t *Tree) parseDefinition() {
328 const context = "define clause"
329 name := t.expectOneOf(itemString, itemRawString, context)
330 var err error
331 t.Name, err = strconv.Unquote(name.val)
332 if err != nil {
333 t.error(err)
334 }
335 t.expect(itemRightDelim, context)
336 var end Node
337 t.Root, end = t.itemList()
338 if end.Type() != nodeEnd {
339 t.errorf("unexpected %s in %s", end, context)
340 }
341 t.add()
342 t.stopParse()
343 }
344
345
346
347
348
349
350 func (t *Tree) itemList() (list *ListNode, next Node) {
351 list = t.newList(t.peekNonSpace().pos)
352 for t.peekNonSpace().typ != itemEOF {
353 n := t.textOrAction()
354 switch n.Type() {
355 case nodeEnd, nodeElse:
356 return list, n
357 }
358 list.append(n)
359 }
360 t.errorf("unexpected EOF")
361 return
362 }
363
364
365
366
367 func (t *Tree) textOrAction() Node {
368 switch token := t.nextNonSpace(); token.typ {
369 case itemText:
370 return t.newText(token.pos, token.val)
371 case itemLeftDelim:
372 t.actionLine = token.line
373 defer t.clearActionLine()
374 return t.action()
375 case itemComment:
376 return t.newComment(token.pos, token.val)
377 default:
378 t.unexpected(token, "input")
379 }
380 return nil
381 }
382
383 func (t *Tree) clearActionLine() {
384 t.actionLine = 0
385 }
386
387
388
389
390
391
392
393
394 func (t *Tree) action() (n Node) {
395 switch token := t.nextNonSpace(); token.typ {
396 case itemBlock:
397 return t.blockControl()
398 case itemBreak:
399 return t.breakControl(token.pos, token.line)
400 case itemContinue:
401 return t.continueControl(token.pos, token.line)
402 case itemElse:
403 return t.elseControl()
404 case itemEnd:
405 return t.endControl()
406 case itemIf:
407 return t.ifControl()
408 case itemRange:
409 return t.rangeControl()
410 case itemTemplate:
411 return t.templateControl()
412 case itemWith:
413 return t.withControl()
414 }
415 t.backup()
416 token := t.peek()
417
418 return t.newAction(token.pos, token.line, t.pipeline("command", itemRightDelim))
419 }
420
421
422
423
424
425
426 func (t *Tree) breakControl(pos Pos, line int) Node {
427 if token := t.nextNonSpace(); token.typ != itemRightDelim {
428 t.unexpected(token, "{{break}}")
429 }
430 if t.rangeDepth == 0 {
431 t.errorf("{{break}} outside {{range}}")
432 }
433 return t.newBreak(pos, line)
434 }
435
436
437
438
439
440
441 func (t *Tree) continueControl(pos Pos, line int) Node {
442 if token := t.nextNonSpace(); token.typ != itemRightDelim {
443 t.unexpected(token, "{{continue}}")
444 }
445 if t.rangeDepth == 0 {
446 t.errorf("{{continue}} outside {{range}}")
447 }
448 return t.newContinue(pos, line)
449 }
450
451
452
453
454 func (t *Tree) pipeline(context string, end itemType) (pipe *PipeNode) {
455 token := t.peekNonSpace()
456 pipe = t.newPipeline(token.pos, token.line, nil)
457
458 decls:
459 if v := t.peekNonSpace(); v.typ == itemVariable {
460 t.next()
461
462
463
464
465 tokenAfterVariable := t.peek()
466 next := t.peekNonSpace()
467 switch {
468 case next.typ == itemAssign, next.typ == itemDeclare:
469 pipe.IsAssign = next.typ == itemAssign
470 t.nextNonSpace()
471 pipe.Decl = append(pipe.Decl, t.newVariable(v.pos, v.val))
472 t.vars = append(t.vars, v.val)
473 case next.typ == itemChar && next.val == ",":
474 t.nextNonSpace()
475 pipe.Decl = append(pipe.Decl, t.newVariable(v.pos, v.val))
476 t.vars = append(t.vars, v.val)
477 if context == "range" && len(pipe.Decl) < 2 {
478 switch t.peekNonSpace().typ {
479 case itemVariable, itemRightDelim, itemRightParen:
480
481 goto decls
482 default:
483 t.errorf("range can only initialize variables")
484 }
485 }
486 t.errorf("too many declarations in %s", context)
487 case tokenAfterVariable.typ == itemSpace:
488 t.backup3(v, tokenAfterVariable)
489 default:
490 t.backup2(v)
491 }
492 }
493 for {
494 switch token := t.nextNonSpace(); token.typ {
495 case end:
496
497 t.checkPipeline(pipe, context)
498 return
499 case itemBool, itemCharConstant, itemComplex, itemDot, itemField, itemIdentifier,
500 itemNumber, itemNil, itemRawString, itemString, itemVariable, itemLeftParen:
501 t.backup()
502 pipe.append(t.command())
503 default:
504 t.unexpected(token, context)
505 }
506 }
507 }
508
509 func (t *Tree) checkPipeline(pipe *PipeNode, context string) {
510
511 if len(pipe.Cmds) == 0 {
512 t.errorf("missing value for %s", context)
513 }
514
515 for i, c := range pipe.Cmds[1:] {
516 switch c.Args[0].Type() {
517 case NodeBool, NodeDot, NodeNil, NodeNumber, NodeString:
518
519 t.errorf("non executable command in pipeline stage %d", i+2)
520 }
521 }
522 }
523
524 func (t *Tree) parseControl(allowElseIf bool, context string) (pos Pos, line int, pipe *PipeNode, list, elseList *ListNode) {
525 defer t.popVars(len(t.vars))
526 pipe = t.pipeline(context, itemRightDelim)
527 if context == "range" {
528 t.rangeDepth++
529 }
530 var next Node
531 list, next = t.itemList()
532 if context == "range" {
533 t.rangeDepth--
534 }
535 switch next.Type() {
536 case nodeEnd:
537 case nodeElse:
538 if allowElseIf {
539
540
541
542
543
544
545
546
547 if t.peek().typ == itemIf {
548 t.next()
549 elseList = t.newList(next.Position())
550 elseList.append(t.ifControl())
551
552 break
553 }
554 }
555 elseList, next = t.itemList()
556 if next.Type() != nodeEnd {
557 t.errorf("expected end; found %s", next)
558 }
559 }
560 return pipe.Position(), pipe.Line, pipe, list, elseList
561 }
562
563
564
565
566
567
568
569 func (t *Tree) ifControl() Node {
570 return t.newIf(t.parseControl(true, "if"))
571 }
572
573
574
575
576
577
578
579 func (t *Tree) rangeControl() Node {
580 r := t.newRange(t.parseControl(false, "range"))
581 return r
582 }
583
584
585
586
587
588
589
590 func (t *Tree) withControl() Node {
591 return t.newWith(t.parseControl(false, "with"))
592 }
593
594
595
596
597
598
599 func (t *Tree) endControl() Node {
600 return t.newEnd(t.expect(itemRightDelim, "end").pos)
601 }
602
603
604
605
606
607
608 func (t *Tree) elseControl() Node {
609
610 peek := t.peekNonSpace()
611 if peek.typ == itemIf {
612
613 return t.newElse(peek.pos, peek.line)
614 }
615 token := t.expect(itemRightDelim, "else")
616 return t.newElse(token.pos, token.line)
617 }
618
619
620
621
622
623
624
625
626 func (t *Tree) blockControl() Node {
627 const context = "block clause"
628
629 token := t.nextNonSpace()
630 name := t.parseTemplateName(token, context)
631 pipe := t.pipeline(context, itemRightDelim)
632
633 block := New(name)
634 block.text = t.text
635 block.Mode = t.Mode
636 block.ParseName = t.ParseName
637 block.startParse(t.funcs, t.lex, t.treeSet)
638 var end Node
639 block.Root, end = block.itemList()
640 if end.Type() != nodeEnd {
641 t.errorf("unexpected %s in %s", end, context)
642 }
643 block.add()
644 block.stopParse()
645
646 return t.newTemplate(token.pos, token.line, name, pipe)
647 }
648
649
650
651
652
653
654
655 func (t *Tree) templateControl() Node {
656 const context = "template clause"
657 token := t.nextNonSpace()
658 name := t.parseTemplateName(token, context)
659 var pipe *PipeNode
660 if t.nextNonSpace().typ != itemRightDelim {
661 t.backup()
662
663 pipe = t.pipeline(context, itemRightDelim)
664 }
665 return t.newTemplate(token.pos, token.line, name, pipe)
666 }
667
668 func (t *Tree) parseTemplateName(token item, context string) (name string) {
669 switch token.typ {
670 case itemString, itemRawString:
671 s, err := strconv.Unquote(token.val)
672 if err != nil {
673 t.error(err)
674 }
675 name = s
676 default:
677 t.unexpected(token, context)
678 }
679 return
680 }
681
682
683
684
685
686
687
688 func (t *Tree) command() *CommandNode {
689 cmd := t.newCommand(t.peekNonSpace().pos)
690 for {
691 t.peekNonSpace()
692 operand := t.operand()
693 if operand != nil {
694 cmd.append(operand)
695 }
696 switch token := t.next(); token.typ {
697 case itemSpace:
698 continue
699 case itemRightDelim, itemRightParen:
700 t.backup()
701 case itemPipe:
702
703 default:
704 t.unexpected(token, "operand")
705 }
706 break
707 }
708 if len(cmd.Args) == 0 {
709 t.errorf("empty command")
710 }
711 return cmd
712 }
713
714
715
716
717
718
719
720
721 func (t *Tree) operand() Node {
722 node := t.term()
723 if node == nil {
724 return nil
725 }
726 if t.peek().typ == itemField {
727 chain := t.newChain(t.peek().pos, node)
728 for t.peek().typ == itemField {
729 chain.Add(t.next().val)
730 }
731
732
733
734
735
736 switch node.Type() {
737 case NodeField:
738 node = t.newField(chain.Position(), chain.String())
739 case NodeVariable:
740 node = t.newVariable(chain.Position(), chain.String())
741 case NodeBool, NodeString, NodeNumber, NodeNil, NodeDot:
742 t.errorf("unexpected . after term %q", node.String())
743 default:
744 node = chain
745 }
746 }
747 return node
748 }
749
750
751
752
753
754
755
756
757
758
759
760
761 func (t *Tree) term() Node {
762 switch token := t.nextNonSpace(); token.typ {
763 case itemIdentifier:
764 checkFunc := t.Mode&SkipFuncCheck == 0
765 if checkFunc && !t.hasFunction(token.val) {
766 t.errorf("function %q not defined", token.val)
767 }
768 return NewIdentifier(token.val).SetTree(t).SetPos(token.pos)
769 case itemDot:
770 return t.newDot(token.pos)
771 case itemNil:
772 return t.newNil(token.pos)
773 case itemVariable:
774 return t.useVar(token.pos, token.val)
775 case itemField:
776 return t.newField(token.pos, token.val)
777 case itemBool:
778 return t.newBool(token.pos, token.val == "true")
779 case itemCharConstant, itemComplex, itemNumber:
780 number, err := t.newNumber(token.pos, token.val, token.typ)
781 if err != nil {
782 t.error(err)
783 }
784 return number
785 case itemLeftParen:
786 return t.pipeline("parenthesized pipeline", itemRightParen)
787 case itemString, itemRawString:
788 s, err := strconv.Unquote(token.val)
789 if err != nil {
790 t.error(err)
791 }
792 return t.newString(token.pos, token.val, s)
793 }
794 t.backup()
795 return nil
796 }
797
798
799 func (t *Tree) hasFunction(name string) bool {
800 for _, funcMap := range t.funcs {
801 if funcMap == nil {
802 continue
803 }
804 if funcMap[name] != nil {
805 return true
806 }
807 }
808 return false
809 }
810
811
812 func (t *Tree) popVars(n int) {
813 t.vars = t.vars[:n]
814 }
815
816
817
818 func (t *Tree) useVar(pos Pos, name string) Node {
819 v := t.newVariable(pos, name)
820 for _, varName := range t.vars {
821 if varName == v.Ident[0] {
822 return v
823 }
824 }
825 t.errorf("undefined variable %q", v.Ident[0])
826 return nil
827 }
828
View as plain text