7d8f0ff870
All of the other subdivisions of a block were already nodes, but we'd represented the labels as an undifferentiated set of nodes belonging directly to the block's child node list. Now that we support replacing the labels in the public API, that's a good excuse to refactor this slightly to make the labels their own node. As well as being consistent with everything else in Block, this also makes it easier to implement the Block.SetLabels operation because we can just change the children of the labels node, rather than having to carefully identify and extract the individual child nodes of the block that happen to represent labels. Internally this models the labels in a similar sort of way as the content of a body, although we've kept the public API directly on the Block type here because that's a more straightforward model for the use-cases we currently know and matches better with the API of hcl.Block. This is just an internal change for consistency. I also added a few tests for having comments interspersed with labels while I was here, because that helped to better exercise the new parseBlockLabels function.
172 lines
4.6 KiB
Go
172 lines
4.6 KiB
Go
package hclwrite
|
|
|
|
import (
|
|
"github.com/hashicorp/hcl/v2/hclsyntax"
|
|
"github.com/zclconf/go-cty/cty"
|
|
)
|
|
|
|
type Block struct {
|
|
inTree
|
|
|
|
leadComments *node
|
|
typeName *node
|
|
labels *node
|
|
open *node
|
|
body *node
|
|
close *node
|
|
}
|
|
|
|
func newBlock() *Block {
|
|
return &Block{
|
|
inTree: newInTree(),
|
|
}
|
|
}
|
|
|
|
// NewBlock constructs a new, empty block with the given type name and labels.
|
|
func NewBlock(typeName string, labels []string) *Block {
|
|
block := newBlock()
|
|
block.init(typeName, labels)
|
|
return block
|
|
}
|
|
|
|
func (b *Block) init(typeName string, labels []string) {
|
|
nameTok := newIdentToken(typeName)
|
|
nameObj := newIdentifier(nameTok)
|
|
b.leadComments = b.children.Append(newComments(nil))
|
|
b.typeName = b.children.Append(nameObj)
|
|
labelsObj := newBlockLabels(labels)
|
|
b.labels = b.children.Append(labelsObj)
|
|
b.open = b.children.AppendUnstructuredTokens(Tokens{
|
|
{
|
|
Type: hclsyntax.TokenOBrace,
|
|
Bytes: []byte{'{'},
|
|
},
|
|
{
|
|
Type: hclsyntax.TokenNewline,
|
|
Bytes: []byte{'\n'},
|
|
},
|
|
})
|
|
body := newBody() // initially totally empty; caller can append to it subsequently
|
|
b.body = b.children.Append(body)
|
|
b.close = b.children.AppendUnstructuredTokens(Tokens{
|
|
{
|
|
Type: hclsyntax.TokenCBrace,
|
|
Bytes: []byte{'}'},
|
|
},
|
|
{
|
|
Type: hclsyntax.TokenNewline,
|
|
Bytes: []byte{'\n'},
|
|
},
|
|
})
|
|
}
|
|
|
|
// Body returns the body that represents the content of the receiving block.
|
|
//
|
|
// Appending to or otherwise modifying this body will make changes to the
|
|
// tokens that are generated between the blocks open and close braces.
|
|
func (b *Block) Body() *Body {
|
|
return b.body.content.(*Body)
|
|
}
|
|
|
|
// Type returns the type name of the block.
|
|
func (b *Block) Type() string {
|
|
typeNameObj := b.typeName.content.(*identifier)
|
|
return string(typeNameObj.token.Bytes)
|
|
}
|
|
|
|
// SetType updates the type name of the block to a given name.
|
|
func (b *Block) SetType(typeName string) {
|
|
nameTok := newIdentToken(typeName)
|
|
nameObj := newIdentifier(nameTok)
|
|
b.typeName.ReplaceWith(nameObj)
|
|
}
|
|
|
|
// Labels returns the labels of the block.
|
|
func (b *Block) Labels() []string {
|
|
return b.labelsObj().Current()
|
|
}
|
|
|
|
// SetLabels updates the labels of the block to given labels.
|
|
// Since we cannot assume that old and new labels are equal in length,
|
|
// remove old labels and insert new ones before TokenOBrace.
|
|
func (b *Block) SetLabels(labels []string) {
|
|
b.labelsObj().Replace(labels)
|
|
}
|
|
|
|
// labelsObj returns the internal node content representation of the block
|
|
// labels. This is not part of the public API because we're intentionally
|
|
// exposing only a limited API to get/set labels on the block itself in a
|
|
// manner similar to the main hcl.Block type, but our block accessors all
|
|
// use this to get the underlying node content to work with.
|
|
func (b *Block) labelsObj() *blockLabels {
|
|
return b.labels.content.(*blockLabels)
|
|
}
|
|
|
|
type blockLabels struct {
|
|
inTree
|
|
|
|
items nodeSet
|
|
}
|
|
|
|
func newBlockLabels(labels []string) *blockLabels {
|
|
ret := &blockLabels{
|
|
inTree: newInTree(),
|
|
items: newNodeSet(),
|
|
}
|
|
|
|
ret.Replace(labels)
|
|
return ret
|
|
}
|
|
|
|
func (bl *blockLabels) Replace(newLabels []string) {
|
|
bl.inTree.children.Clear()
|
|
bl.items.Clear()
|
|
|
|
for _, label := range newLabels {
|
|
labelToks := TokensForValue(cty.StringVal(label))
|
|
// Force a new label to use the quoted form, which is the idiomatic
|
|
// form. The unquoted form is supported in HCL 2 only for compatibility
|
|
// with historical use in HCL 1.
|
|
labelObj := newQuoted(labelToks)
|
|
labelNode := bl.children.Append(labelObj)
|
|
bl.items.Add(labelNode)
|
|
}
|
|
}
|
|
|
|
func (bl *blockLabels) Current() []string {
|
|
labelNames := make([]string, 0, len(bl.items))
|
|
list := bl.items.List()
|
|
|
|
for _, label := range list {
|
|
switch labelObj := label.content.(type) {
|
|
case *identifier:
|
|
if labelObj.token.Type == hclsyntax.TokenIdent {
|
|
labelString := string(labelObj.token.Bytes)
|
|
labelNames = append(labelNames, labelString)
|
|
}
|
|
|
|
case *quoted:
|
|
tokens := labelObj.tokens
|
|
if len(tokens) == 3 &&
|
|
tokens[0].Type == hclsyntax.TokenOQuote &&
|
|
tokens[1].Type == hclsyntax.TokenQuotedLit &&
|
|
tokens[2].Type == hclsyntax.TokenCQuote {
|
|
// Note that TokenQuotedLit may contain escape sequences.
|
|
labelString, diags := hclsyntax.ParseStringLiteralToken(tokens[1].asHCLSyntax())
|
|
|
|
// If parsing the string literal returns error diagnostics
|
|
// then we can just assume the label doesn't match, because it's invalid in some way.
|
|
if !diags.HasErrors() {
|
|
labelNames = append(labelNames, labelString)
|
|
}
|
|
}
|
|
|
|
default:
|
|
// If neither of the previous cases are true (should be impossible)
|
|
// then we can just ignore it, because it's invalid too.
|
|
}
|
|
}
|
|
|
|
return labelNames
|
|
}
|