hclwrite: add space between "in" keyword and expr in for expression
Our normal ruleset thinks that the "in" keyword here is a variable reference and so writes it as "in[y]". Since there's never any reason for a variable to appear immediately after another variable, we can check for a preceding identifier as a heuristic to recognize whether in is probably being used as a keyword rather than as a variable. This is not exact, but the only time this should be a false positive is if there were a syntax error in the input, and we don't make any guarantees about the result in that case anyway. This fixes #52.
This commit is contained in:
parent
bafa0c5ace
commit
002296d7bb
@ -4,6 +4,8 @@ import (
|
|||||||
"github.com/hashicorp/hcl2/hcl/hclsyntax"
|
"github.com/hashicorp/hcl2/hcl/hclsyntax"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var inKeyword = hclsyntax.Keyword([]byte{'i', 'n'})
|
||||||
|
|
||||||
// placeholder token used when we don't have a token but we don't want
|
// placeholder token used when we don't have a token but we don't want
|
||||||
// to pass a real "nil" and complicate things with nil pointer checks
|
// to pass a real "nil" and complicate things with nil pointer checks
|
||||||
var nilToken = &Token{
|
var nilToken = &Token{
|
||||||
@ -247,6 +249,15 @@ func spaceAfterToken(subject, before, after *Token) bool {
|
|||||||
// No extra spaces within templates
|
// No extra spaces within templates
|
||||||
return false
|
return false
|
||||||
|
|
||||||
|
case inKeyword.TokenMatches(subject.asHCLSyntax()) && before.Type == hclsyntax.TokenIdent:
|
||||||
|
// This is a special case for inside for expressions where a user
|
||||||
|
// might want to use a literal tuple constructor:
|
||||||
|
// [for x in [foo]: x]
|
||||||
|
// ... in that case, we would normally produce in[foo] thinking that
|
||||||
|
// in is a reference, but we'll recognize it as a keyword here instead
|
||||||
|
// to make the result less confusing.
|
||||||
|
return true
|
||||||
|
|
||||||
case after.Type == hclsyntax.TokenOBrack && (subject.Type == hclsyntax.TokenIdent || subject.Type == hclsyntax.TokenNumberLit || tokenBracketChange(subject) < 0):
|
case after.Type == hclsyntax.TokenOBrack && (subject.Type == hclsyntax.TokenIdent || subject.Type == hclsyntax.TokenNumberLit || tokenBracketChange(subject) < 0):
|
||||||
return false
|
return false
|
||||||
|
|
||||||
@ -283,7 +294,7 @@ func spaceAfterToken(subject, before, after *Token) bool {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
case subject.Type == hclsyntax.TokenOBrace || (after != nil && after.Type == hclsyntax.TokenCBrace):
|
case subject.Type == hclsyntax.TokenOBrace || after.Type == hclsyntax.TokenCBrace:
|
||||||
// Unlike other bracket types, braces have spaces on both sides of them,
|
// Unlike other bracket types, braces have spaces on both sides of them,
|
||||||
// both in single-line nested blocks foo { bar = baz } and in object
|
// both in single-line nested blocks foo { bar = baz } and in object
|
||||||
// constructor expressions foo = { bar = baz }.
|
// constructor expressions foo = { bar = baz }.
|
||||||
@ -294,6 +305,10 @@ func spaceAfterToken(subject, before, after *Token) bool {
|
|||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
|
|
||||||
|
case after.Type == hclsyntax.TokenColon:
|
||||||
|
// Never spaces before colons
|
||||||
|
return false
|
||||||
|
|
||||||
case tokenBracketChange(subject) > 0:
|
case tokenBracketChange(subject) > 0:
|
||||||
// No spaces after open brackets
|
// No spaces after open brackets
|
||||||
return false
|
return false
|
||||||
|
@ -111,6 +111,14 @@ foo(
|
|||||||
`[ [ ] ]`,
|
`[ [ ] ]`,
|
||||||
`[[]]`,
|
`[[]]`,
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
`[for x in y: x]`,
|
||||||
|
`[for x in y: x]`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
`[for x in [y]: x]`,
|
||||||
|
`[for x in [y]: x]`,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
`
|
`
|
||||||
[
|
[
|
||||||
|
@ -5,6 +5,7 @@ import (
|
|||||||
"io"
|
"io"
|
||||||
|
|
||||||
"github.com/apparentlymart/go-textseg/textseg"
|
"github.com/apparentlymart/go-textseg/textseg"
|
||||||
|
"github.com/hashicorp/hcl2/hcl"
|
||||||
"github.com/hashicorp/hcl2/hcl/hclsyntax"
|
"github.com/hashicorp/hcl2/hcl/hclsyntax"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -22,6 +23,23 @@ type Token struct {
|
|||||||
SpacesBefore int
|
SpacesBefore int
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// asHCLSyntax returns the receiver expressed as an incomplete hclsyntax.Token.
|
||||||
|
// A complete token is not possible since we don't have source location
|
||||||
|
// information here, and so this method is unexported so we can be sure it will
|
||||||
|
// only be used for internal purposes where we know the range isn't important.
|
||||||
|
//
|
||||||
|
// This is primarily intended to allow us to re-use certain functionality from
|
||||||
|
// hclsyntax rather than re-implementing it against our own token type here.
|
||||||
|
func (t *Token) asHCLSyntax() hclsyntax.Token {
|
||||||
|
return hclsyntax.Token{
|
||||||
|
Type: t.Type,
|
||||||
|
Bytes: t.Bytes,
|
||||||
|
Range: hcl.Range{
|
||||||
|
Filename: "<invalid>",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Tokens is a flat list of tokens.
|
// Tokens is a flat list of tokens.
|
||||||
type Tokens []*Token
|
type Tokens []*Token
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user