pkg/cfgparser: Require a new line after }

Makes the following configuration invalid:
aaa {
    bbb
} ccc

Previously it would have been parsed as:
aaa {
    bbb
}
ccc
This commit is contained in:
fox.cpp 2020-03-04 01:34:04 +03:00
parent 296d67488c
commit 18fc48c6e6
No known key found for this signature in database
GPG key ID: E76D97CCEDE90B6C
2 changed files with 49 additions and 20 deletions

View file

@ -120,6 +120,11 @@ func (ctx *parseContext) readNode() (Node, error) {
node.Args = append(node.Args, ctx.Val())
}
// Continue reading the same Node if the \ was used to escape the newline.
// E.g.
// name arg0 arg1 \
// arg2 arg3
if len(node.Args) != 0 && node.Args[len(node.Args)-1] == `\` {
last := len(node.Args) - 1
node.Args[last] = node.Args[last][:len(node.Args[last])-1]
@ -197,14 +202,53 @@ func (ctx *parseContext) readNodes() ([]Node, error) {
// but non-nil Children slice for empty braces.
res := []Node{}
// Refuse to continue is nesting is too big.
if ctx.nesting > 255 {
return res, ctx.Err("nesting limit reached")
}
ctx.nesting++
for ctx.Next() {
var requireNewLine bool
// This loop iterates over logical lines.
// Here are some examples, '#' is placed before token where cursor is when
// another iteration of this loop starts.
//
// #a
// #a b
// #a b {
// #ac aa
// #}
// #aa bbb bbb \
// ccc ccc
// #a b { #ac aa }
//
// As can be seen by the latest example, sometimes such logical line might
// not be terminated by an actual LF character and so this needs to be
// handled carefully.
//
// Note that if the '}' is on the same physical line, it is currently
// included as the part of the logical line, that is:
// #a b { #ac aa }
// ^------- that's the logical line
// #c d
// ^--- that's the next logical line
// This is handled by the "edge case" branch inside the loop.
for {
if requireNewLine {
if !ctx.NextLine() {
// If we can't advance cursor even without Line constraint -
// that's EOF.
if !ctx.Next() {
return res, nil
}
return res, ctx.Err("newline is required after closing brace")
}
} else if !ctx.Next() {
break
}
requireNewLine = false
// name arg0 arg1 {
// c0
// c1
@ -230,6 +274,7 @@ func (ctx *parseContext) readNodes() ([]Node, error) {
if err != nil {
return res, err
}
requireNewLine = true
shouldStop := false

View file

@ -276,27 +276,11 @@ var cases = []struct {
true,
},
{
// TODO: Should we make it fail?
"closing brace in next block header",
`a {
} b b1`,
[]Node{
{
Name: "a",
Args: []string{},
Children: []Node{},
File: "test",
Line: 1,
},
{
Name: "b",
Args: []string{"b1"},
Children: nil,
File: "test",
Line: 2,
},
},
false,
nil,
true,
},
{
"environment variable expansion",