2017-09-11 23:00:31 +00:00
|
|
|
package hclwrite
|
2017-05-29 23:05:34 +00:00
|
|
|
|
|
|
|
import (
|
2017-06-10 21:11:47 +00:00
|
|
|
"bytes"
|
2017-06-07 14:06:23 +00:00
|
|
|
"io"
|
|
|
|
|
2017-06-10 21:11:47 +00:00
|
|
|
"github.com/apparentlymart/go-textseg/textseg"
|
2017-09-11 23:40:37 +00:00
|
|
|
"github.com/hashicorp/hcl2/hcl/hclsyntax"
|
2017-05-29 23:05:34 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Token is a single sequence of bytes annotated with a type. It is similar
|
2018-01-24 05:54:38 +00:00
|
|
|
// in purpose to hclsyntax.Token, but discards the source position information
|
2017-05-29 23:05:34 +00:00
|
|
|
// since that is not useful in code generation.
|
|
|
|
type Token struct {
|
2017-09-11 23:40:37 +00:00
|
|
|
Type hclsyntax.TokenType
|
2017-05-29 23:05:34 +00:00
|
|
|
Bytes []byte
|
|
|
|
|
|
|
|
// We record the number of spaces before each token so that we can
|
|
|
|
// reproduce the exact layout of the original file when we're making
|
|
|
|
// surgical changes in-place. When _new_ code is created it will always
|
|
|
|
// be in the canonical style, but we preserve layout of existing code.
|
2017-05-29 23:59:20 +00:00
|
|
|
SpacesBefore int
|
2017-05-29 23:05:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Tokens is a flat list of tokens.
|
|
|
|
type Tokens []*Token
|
|
|
|
|
2017-06-10 21:11:47 +00:00
|
|
|
func (ts Tokens) Bytes() []byte {
|
|
|
|
buf := &bytes.Buffer{}
|
|
|
|
ts.WriteTo(buf)
|
|
|
|
return buf.Bytes()
|
|
|
|
}
|
|
|
|
|
2018-08-01 15:45:22 +00:00
|
|
|
func (ts Tokens) testValue() string {
|
|
|
|
return string(ts.Bytes())
|
|
|
|
}
|
|
|
|
|
2017-06-10 21:11:47 +00:00
|
|
|
// Columns returns the number of columns (grapheme clusters) the token sequence
|
|
|
|
// occupies. The result is not meaningful if there are newline or single-line
|
|
|
|
// comment tokens in the sequence.
|
|
|
|
func (ts Tokens) Columns() int {
|
|
|
|
ret := 0
|
|
|
|
for _, token := range ts {
|
|
|
|
ret += token.SpacesBefore // spaces are always worth one column each
|
|
|
|
ct, _ := textseg.TokenCount(token.Bytes, textseg.ScanGraphemeClusters)
|
|
|
|
ret += ct
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|
|
|
|
|
2017-06-07 14:06:23 +00:00
|
|
|
// WriteTo takes an io.Writer and writes the bytes for each token to it,
|
|
|
|
// along with the spacing that separates each token. In other words, this
|
|
|
|
// allows serializing the tokens to a file or other such byte stream.
|
2018-08-01 15:45:22 +00:00
|
|
|
func (ts Tokens) WriteTo(wr io.Writer) (int, error) {
|
2017-06-07 14:06:23 +00:00
|
|
|
// We know we're going to be writing a lot of small chunks of repeated
|
|
|
|
// space characters, so we'll prepare a buffer of these that we can
|
|
|
|
// easily pass to wr.Write without any further allocation.
|
|
|
|
spaces := make([]byte, 40)
|
|
|
|
for i := range spaces {
|
|
|
|
spaces[i] = ' '
|
|
|
|
}
|
|
|
|
|
|
|
|
var n int
|
|
|
|
var err error
|
2018-08-01 15:45:22 +00:00
|
|
|
for _, token := range ts {
|
2017-06-07 14:06:23 +00:00
|
|
|
if err != nil {
|
2018-08-01 15:45:22 +00:00
|
|
|
return n, err
|
2017-06-07 14:06:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for spacesBefore := token.SpacesBefore; spacesBefore > 0; spacesBefore -= len(spaces) {
|
|
|
|
thisChunk := spacesBefore
|
|
|
|
if thisChunk > len(spaces) {
|
|
|
|
thisChunk = len(spaces)
|
|
|
|
}
|
|
|
|
var thisN int
|
|
|
|
thisN, err = wr.Write(spaces[:thisChunk])
|
|
|
|
n += thisN
|
|
|
|
if err != nil {
|
2018-08-01 15:45:22 +00:00
|
|
|
return n, err
|
2017-06-07 14:06:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var thisN int
|
|
|
|
thisN, err = wr.Write(token.Bytes)
|
|
|
|
n += thisN
|
2018-08-01 15:45:22 +00:00
|
|
|
}
|
2017-06-07 14:06:23 +00:00
|
|
|
|
|
|
|
return n, err
|
2017-05-29 23:05:34 +00:00
|
|
|
}
|
|
|
|
|
2018-08-01 15:45:22 +00:00
|
|
|
func (ts Tokens) walkChildNodes(w internalWalkFunc) {
|
|
|
|
// Unstructured tokens have no child nodes
|
2017-06-12 01:24:15 +00:00
|
|
|
}
|
|
|
|
|
2018-08-01 15:45:22 +00:00
|
|
|
func (ts Tokens) BuildTokens(to Tokens) Tokens {
|
|
|
|
return append(to, ts...)
|
2017-06-12 01:24:15 +00:00
|
|
|
}
|
2018-08-09 15:44:48 +00:00
|
|
|
|
|
|
|
func newIdentToken(name string) *Token {
|
|
|
|
return &Token{
|
|
|
|
Type: hclsyntax.TokenIdent,
|
|
|
|
Bytes: []byte(name),
|
|
|
|
}
|
|
|
|
}
|