2017-09-11 23:00:31 +00:00
|
|
|
package hcltest
|
2017-07-27 22:59:32 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
"github.com/hashicorp/hcl2/hcl"
|
2017-07-27 22:59:32 +00:00
|
|
|
"github.com/zclconf/go-cty/cty"
|
|
|
|
)
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
// MockBody returns a hcl.Body implementation that works in terms of a
|
|
|
|
// caller-constructed hcl.BodyContent, thus avoiding the need to parse
|
2017-07-27 22:59:32 +00:00
|
|
|
// a "real" zcl config file to use as input to a test.
|
2017-09-11 23:40:37 +00:00
|
|
|
func MockBody(content *hcl.BodyContent) hcl.Body {
|
2017-07-27 22:59:32 +00:00
|
|
|
return mockBody{content}
|
|
|
|
}
|
|
|
|
|
|
|
|
type mockBody struct {
|
2017-09-11 23:40:37 +00:00
|
|
|
C *hcl.BodyContent
|
2017-07-27 22:59:32 +00:00
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
func (b mockBody) Content(schema *hcl.BodySchema) (*hcl.BodyContent, hcl.Diagnostics) {
|
2017-07-27 22:59:32 +00:00
|
|
|
content, remainI, diags := b.PartialContent(schema)
|
|
|
|
remain := remainI.(mockBody)
|
|
|
|
for _, attr := range remain.C.Attributes {
|
2017-09-11 23:40:37 +00:00
|
|
|
diags = append(diags, &hcl.Diagnostic{
|
|
|
|
Severity: hcl.DiagError,
|
2017-07-27 22:59:32 +00:00
|
|
|
Summary: "Extraneous attribute in mock body",
|
|
|
|
Detail: fmt.Sprintf("Mock body has extraneous attribute %q.", attr.Name),
|
|
|
|
Subject: &attr.NameRange,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
for _, block := range remain.C.Blocks {
|
2017-09-11 23:40:37 +00:00
|
|
|
diags = append(diags, &hcl.Diagnostic{
|
|
|
|
Severity: hcl.DiagError,
|
2017-07-27 22:59:32 +00:00
|
|
|
Summary: "Extraneous block in mock body",
|
|
|
|
Detail: fmt.Sprintf("Mock body has extraneous block of type %q.", block.Type),
|
|
|
|
Subject: &block.DefRange,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return content, diags
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
func (b mockBody) PartialContent(schema *hcl.BodySchema) (*hcl.BodyContent, hcl.Body, hcl.Diagnostics) {
|
|
|
|
ret := &hcl.BodyContent{
|
|
|
|
Attributes: map[string]*hcl.Attribute{},
|
|
|
|
Blocks: []*hcl.Block{},
|
2017-07-27 22:59:32 +00:00
|
|
|
MissingItemRange: b.C.MissingItemRange,
|
|
|
|
}
|
2017-09-11 23:40:37 +00:00
|
|
|
remain := &hcl.BodyContent{
|
|
|
|
Attributes: map[string]*hcl.Attribute{},
|
|
|
|
Blocks: []*hcl.Block{},
|
2017-07-27 22:59:32 +00:00
|
|
|
MissingItemRange: b.C.MissingItemRange,
|
|
|
|
}
|
2017-09-11 23:40:37 +00:00
|
|
|
var diags hcl.Diagnostics
|
2017-07-27 22:59:32 +00:00
|
|
|
|
|
|
|
if len(schema.Attributes) != 0 {
|
|
|
|
for _, attrS := range schema.Attributes {
|
|
|
|
name := attrS.Name
|
|
|
|
attr, ok := b.C.Attributes[name]
|
|
|
|
if !ok {
|
|
|
|
if attrS.Required {
|
2017-09-11 23:40:37 +00:00
|
|
|
diags = append(diags, &hcl.Diagnostic{
|
|
|
|
Severity: hcl.DiagError,
|
2017-07-27 22:59:32 +00:00
|
|
|
Summary: "Missing required attribute",
|
|
|
|
Detail: fmt.Sprintf("Mock body doesn't have attribute %q", name),
|
|
|
|
Subject: b.C.MissingItemRange.Ptr(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
ret.Attributes[name] = attr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for attrN, attr := range b.C.Attributes {
|
|
|
|
if _, ok := ret.Attributes[attrN]; !ok {
|
|
|
|
remain.Attributes[attrN] = attr
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
wantedBlocks := map[string]hcl.BlockHeaderSchema{}
|
2017-07-27 22:59:32 +00:00
|
|
|
for _, blockS := range schema.Blocks {
|
|
|
|
wantedBlocks[blockS.Type] = blockS
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, block := range b.C.Blocks {
|
|
|
|
if blockS, ok := wantedBlocks[block.Type]; ok {
|
|
|
|
if len(block.Labels) != len(blockS.LabelNames) {
|
2017-09-11 23:40:37 +00:00
|
|
|
diags = append(diags, &hcl.Diagnostic{
|
|
|
|
Severity: hcl.DiagError,
|
2017-07-27 22:59:32 +00:00
|
|
|
Summary: "Wrong number of block labels",
|
|
|
|
Detail: fmt.Sprintf("Block of type %q requires %d labels, but got %d", blockS.Type, len(blockS.LabelNames), len(block.Labels)),
|
|
|
|
Subject: b.C.MissingItemRange.Ptr(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
ret.Blocks = append(ret.Blocks, block)
|
|
|
|
} else {
|
|
|
|
remain.Blocks = append(remain.Blocks, block)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret, mockBody{remain}, diags
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
func (b mockBody) JustAttributes() (hcl.Attributes, hcl.Diagnostics) {
|
|
|
|
var diags hcl.Diagnostics
|
2017-07-27 22:59:32 +00:00
|
|
|
if len(b.C.Blocks) != 0 {
|
2017-09-11 23:40:37 +00:00
|
|
|
diags = append(diags, &hcl.Diagnostic{
|
|
|
|
Severity: hcl.DiagError,
|
2017-07-27 22:59:32 +00:00
|
|
|
Summary: "Mock body has blocks",
|
|
|
|
Detail: "Can't use JustAttributes on a mock body with blocks.",
|
|
|
|
Subject: b.C.MissingItemRange.Ptr(),
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return b.C.Attributes, diags
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
func (b mockBody) MissingItemRange() hcl.Range {
|
2017-07-27 22:59:32 +00:00
|
|
|
return b.C.MissingItemRange
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
// MockExprLiteral returns a hcl.Expression that evaluates to the given literal
|
2017-07-27 22:59:32 +00:00
|
|
|
// value.
|
2017-09-11 23:40:37 +00:00
|
|
|
func MockExprLiteral(val cty.Value) hcl.Expression {
|
2017-07-27 22:59:32 +00:00
|
|
|
return mockExprLiteral{val}
|
|
|
|
}
|
|
|
|
|
|
|
|
type mockExprLiteral struct {
|
|
|
|
V cty.Value
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
func (e mockExprLiteral) Value(ctx *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
2017-07-27 22:59:32 +00:00
|
|
|
return e.V, nil
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
func (e mockExprLiteral) Variables() []hcl.Traversal {
|
2017-07-27 22:59:32 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
func (e mockExprLiteral) Range() hcl.Range {
|
|
|
|
return hcl.Range{
|
2017-07-27 22:59:32 +00:00
|
|
|
Filename: "MockExprLiteral",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
func (e mockExprLiteral) StartRange() hcl.Range {
|
2017-07-27 22:59:32 +00:00
|
|
|
return e.Range()
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
// MockExprVariable returns a hcl.Expression that evaluates to the value of
|
2017-07-27 22:59:32 +00:00
|
|
|
// the variable with the given name.
|
2017-09-11 23:40:37 +00:00
|
|
|
func MockExprVariable(name string) hcl.Expression {
|
2017-07-27 22:59:32 +00:00
|
|
|
return mockExprVariable(name)
|
|
|
|
}
|
|
|
|
|
|
|
|
type mockExprVariable string
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
func (e mockExprVariable) Value(ctx *hcl.EvalContext) (cty.Value, hcl.Diagnostics) {
|
2017-07-27 22:59:32 +00:00
|
|
|
name := string(e)
|
|
|
|
for ctx != nil {
|
|
|
|
if val, ok := ctx.Variables[name]; ok {
|
|
|
|
return val, nil
|
|
|
|
}
|
|
|
|
ctx = ctx.Parent()
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we fall out here then there is no variable with the given name
|
2017-09-11 23:40:37 +00:00
|
|
|
return cty.DynamicVal, hcl.Diagnostics{
|
2017-07-27 22:59:32 +00:00
|
|
|
{
|
2017-09-11 23:40:37 +00:00
|
|
|
Severity: hcl.DiagError,
|
2017-07-27 22:59:32 +00:00
|
|
|
Summary: "Reference to undefined variable",
|
|
|
|
Detail: fmt.Sprintf("Variable %q is not defined.", name),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
func (e mockExprVariable) Variables() []hcl.Traversal {
|
|
|
|
return []hcl.Traversal{
|
2017-07-27 22:59:32 +00:00
|
|
|
{
|
2017-09-11 23:40:37 +00:00
|
|
|
hcl.TraverseRoot{
|
2017-07-27 22:59:32 +00:00
|
|
|
Name: string(e),
|
|
|
|
SrcRange: e.Range(),
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
func (e mockExprVariable) Range() hcl.Range {
|
|
|
|
return hcl.Range{
|
2017-07-27 22:59:32 +00:00
|
|
|
Filename: "MockExprVariable",
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
func (e mockExprVariable) StartRange() hcl.Range {
|
2017-07-27 22:59:32 +00:00
|
|
|
return e.Range()
|
|
|
|
}
|
|
|
|
|
2017-09-11 23:40:37 +00:00
|
|
|
// MockAttrs constructs and returns a hcl.Attributes map with attributes
|
2017-07-27 22:59:32 +00:00
|
|
|
// derived from the given expression map.
|
|
|
|
//
|
|
|
|
// Each entry in the map becomes an attribute whose name is the key and
|
|
|
|
// whose expression is the value.
|
2017-09-11 23:40:37 +00:00
|
|
|
func MockAttrs(exprs map[string]hcl.Expression) hcl.Attributes {
|
|
|
|
ret := make(hcl.Attributes)
|
2017-07-27 22:59:32 +00:00
|
|
|
for name, expr := range exprs {
|
2017-09-11 23:40:37 +00:00
|
|
|
ret[name] = &hcl.Attribute{
|
2017-07-27 22:59:32 +00:00
|
|
|
Name: name,
|
|
|
|
Expr: expr,
|
2017-09-11 23:40:37 +00:00
|
|
|
Range: hcl.Range{
|
2017-07-27 22:59:32 +00:00
|
|
|
Filename: "MockAttrs",
|
|
|
|
},
|
2017-09-11 23:40:37 +00:00
|
|
|
NameRange: hcl.Range{
|
2017-07-27 22:59:32 +00:00
|
|
|
Filename: "MockAttrs",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret
|
|
|
|
}
|