2020-03-18 16:28:57 +00:00
|
|
|
// Copyright 2016-2020, Pulumi Corporation.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
|
|
|
package model
|
|
|
|
|
|
|
|
import (
|
2020-04-03 04:23:12 +00:00
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
|
2020-03-18 16:28:57 +00:00
|
|
|
"github.com/hashicorp/hcl/v2"
|
|
|
|
"github.com/hashicorp/hcl/v2/hclsyntax"
|
2021-03-17 13:20:05 +00:00
|
|
|
"github.com/pulumi/pulumi/pkg/v3/codegen/hcl2/syntax"
|
2020-03-18 16:28:57 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Block represents an HCL2 block.
|
|
|
|
type Block struct {
|
|
|
|
// The syntax node for the block, if any.
|
|
|
|
Syntax *hclsyntax.Block
|
|
|
|
// The tokens for the block.
|
2020-04-03 03:01:14 +00:00
|
|
|
Tokens *syntax.BlockTokens
|
2020-03-18 16:28:57 +00:00
|
|
|
|
|
|
|
// The block's type.
|
|
|
|
Type string
|
|
|
|
// The block's labels.
|
|
|
|
Labels []string
|
|
|
|
|
|
|
|
// The block's body.
|
|
|
|
Body *Body
|
|
|
|
}
|
|
|
|
|
|
|
|
// SyntaxNode returns the syntax node of the block, and will either return an *hclsyntax.Block or syntax.None.
|
|
|
|
func (b *Block) SyntaxNode() hclsyntax.Node {
|
|
|
|
return syntaxOrNone(b.Syntax)
|
|
|
|
}
|
|
|
|
|
2020-04-09 23:37:52 +00:00
|
|
|
func (b *Block) HasLeadingTrivia() bool {
|
2020-04-03 04:23:12 +00:00
|
|
|
return b.Tokens != nil
|
|
|
|
}
|
|
|
|
|
2020-04-09 23:37:52 +00:00
|
|
|
func (b *Block) HasTrailingTrivia() bool {
|
2020-04-03 04:23:12 +00:00
|
|
|
return b.Tokens != nil
|
|
|
|
}
|
|
|
|
|
2020-07-07 20:46:26 +00:00
|
|
|
func (b *Block) GetLeadingTrivia() syntax.TriviaList {
|
|
|
|
return b.Tokens.GetType(b.Type).LeadingTrivia
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) GetTrailingTrivia() syntax.TriviaList {
|
|
|
|
return b.Tokens.GetCloseBrace().TrailingTrivia
|
|
|
|
}
|
|
|
|
|
2020-04-03 04:23:12 +00:00
|
|
|
func (b *Block) Format(f fmt.State, c rune) {
|
|
|
|
b.print(f, &printer{})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (b *Block) print(w io.Writer, p *printer) {
|
|
|
|
// Print the type.
|
2020-04-14 02:11:56 +00:00
|
|
|
p.fprintf(w, "%v", b.Tokens.GetType(b.Type))
|
2020-04-03 04:23:12 +00:00
|
|
|
|
|
|
|
// Print the labels with leading and trailing trivia.
|
2020-04-14 02:11:56 +00:00
|
|
|
labelTokens := b.Tokens.GetLabels(b.Labels)
|
2020-04-03 04:23:12 +00:00
|
|
|
for i, l := range b.Labels {
|
|
|
|
var t syntax.Token
|
|
|
|
if i < len(labelTokens) {
|
|
|
|
t = labelTokens[i]
|
|
|
|
}
|
|
|
|
if hclsyntax.ValidIdentifier(l) {
|
2020-04-14 02:11:56 +00:00
|
|
|
t = identToken(t, l)
|
2020-04-03 04:23:12 +00:00
|
|
|
} else {
|
2020-04-14 02:11:56 +00:00
|
|
|
l = fmt.Sprintf("%q", l)
|
|
|
|
if t.Raw.Type != hclsyntax.TokenQuotedLit || string(t.Raw.Bytes) != l {
|
|
|
|
t.Raw.Type = hclsyntax.TokenQuotedLit
|
|
|
|
t.Raw.Bytes = []byte(l)
|
|
|
|
}
|
2020-04-03 04:23:12 +00:00
|
|
|
}
|
|
|
|
p.fprintf(w, "% v", t)
|
|
|
|
}
|
2020-04-07 02:43:16 +00:00
|
|
|
if len(b.Labels) < len(labelTokens) {
|
|
|
|
for _, l := range labelTokens[len(b.Labels):] {
|
2020-04-03 04:23:12 +00:00
|
|
|
p.fprintf(w, "%v", syntax.Token{
|
|
|
|
LeadingTrivia: l.LeadingTrivia,
|
|
|
|
TrailingTrivia: l.TrailingTrivia,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Print the opening brace.
|
2020-04-14 02:11:56 +00:00
|
|
|
p.fprintf(w, "% v", b.Tokens.GetOpenBrace())
|
2020-04-03 04:23:12 +00:00
|
|
|
|
|
|
|
// Print the block contents.
|
|
|
|
p.indented(func() {
|
|
|
|
b.Body.print(w, p)
|
|
|
|
})
|
|
|
|
|
|
|
|
if b.Tokens != nil {
|
2020-04-14 02:11:56 +00:00
|
|
|
p.fprintf(w, "%v", b.Tokens.GetCloseBrace())
|
2020-04-03 04:23:12 +00:00
|
|
|
} else {
|
2020-04-07 02:43:16 +00:00
|
|
|
p.fprintf(w, "%s}", p.indent)
|
2020-04-03 04:23:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-18 16:28:57 +00:00
|
|
|
func (*Block) isBodyItem() {}
|
|
|
|
|
|
|
|
// BindBlock binds an HCL2 block using the given scopes and token map.
|
2020-04-17 15:24:44 +00:00
|
|
|
func BindBlock(block *hclsyntax.Block, scopes Scopes, tokens syntax.TokenMap,
|
2023-03-03 16:36:39 +00:00
|
|
|
opts ...BindOption,
|
|
|
|
) (*Block, hcl.Diagnostics) {
|
2020-04-17 15:24:44 +00:00
|
|
|
body, diagnostics := BindBody(block.Body, scopes, tokens, opts...)
|
2020-04-03 03:01:14 +00:00
|
|
|
blockTokens, _ := tokens.ForNode(block).(*syntax.BlockTokens)
|
2020-03-18 16:28:57 +00:00
|
|
|
return &Block{
|
|
|
|
Syntax: block,
|
|
|
|
Tokens: blockTokens,
|
|
|
|
Type: block.Type,
|
|
|
|
Labels: block.Labels,
|
|
|
|
Body: body,
|
|
|
|
}, diagnostics
|
|
|
|
}
|