hcl/hclsyntax: Parsing of for expressions while newline-sensitive
The fact that object constructors are newline-sensitive while object for expressions are not requires some special consideration in the parser. We previously make a small fix here to delay turning on newline-sensitive scanning before peeking ahead for a "for" keyword, but that was sufficient only when the for expression was not already in a newline-sensitive context. Now we force newline-sensitive parsing off while we scan for the keyword, and also again once we begin parsing the for expression, ensuring that the for expression is always scanned properly regardless of what context it appears in.
This commit is contained in:
parent
fb2bc46cdb
commit
7e26f2f346
@ -507,6 +507,20 @@ upper(
|
|||||||
}),
|
}),
|
||||||
0,
|
0,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
// This one is different than the previous because the extra level of
|
||||||
|
// object constructor causes the inner for expression to begin parsing
|
||||||
|
// in newline-sensitive mode, which it must then properly disable in
|
||||||
|
// order to peek the "for" keyword.
|
||||||
|
"{\n a = {\n for k, v in {hello: \"world\"}:\nk => v\n }\n}",
|
||||||
|
nil,
|
||||||
|
cty.ObjectVal(map[string]cty.Value{
|
||||||
|
"a": cty.ObjectVal(map[string]cty.Value{
|
||||||
|
"hello": cty.StringVal("world"),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
0,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
`{for k, v in {hello: "world"}: k => v if k == "hello"}`,
|
`{for k, v in {hello: "world"}: k => v if k == "hello"}`,
|
||||||
nil,
|
nil,
|
||||||
|
@ -1242,7 +1242,13 @@ func (p *parser) parseObjectCons() (Expression, hcl.Diagnostics) {
|
|||||||
panic("parseObjectCons called without peeker pointing to open brace")
|
panic("parseObjectCons called without peeker pointing to open brace")
|
||||||
}
|
}
|
||||||
|
|
||||||
if forKeyword.TokenMatches(p.Peek()) {
|
// We must temporarily stop looking at newlines here while we check for
|
||||||
|
// a "for" keyword, since for expressions are _not_ newline-sensitive,
|
||||||
|
// even though object constructors are.
|
||||||
|
p.PushIncludeNewlines(false)
|
||||||
|
isFor := forKeyword.TokenMatches(p.Peek())
|
||||||
|
p.PopIncludeNewlines()
|
||||||
|
if isFor {
|
||||||
return p.finishParsingForExpr(open)
|
return p.finishParsingForExpr(open)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1377,6 +1383,8 @@ func (p *parser) parseObjectCons() (Expression, hcl.Diagnostics) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (p *parser) finishParsingForExpr(open Token) (Expression, hcl.Diagnostics) {
|
func (p *parser) finishParsingForExpr(open Token) (Expression, hcl.Diagnostics) {
|
||||||
|
p.PushIncludeNewlines(false)
|
||||||
|
defer p.PopIncludeNewlines()
|
||||||
introducer := p.Read()
|
introducer := p.Read()
|
||||||
if !forKeyword.TokenMatches(introducer) {
|
if !forKeyword.TokenMatches(introducer) {
|
||||||
// Should never happen if callers are behaving
|
// Should never happen if callers are behaving
|
||||||
|
Loading…
Reference in New Issue
Block a user