zcl: Factor the index operation out into its own function
Indexing is pretty fundamental and it's also non-trivial, so having this exposed will make it easier for it to be implemented consistently across many different callers, including within calling applications.
This commit is contained in:
parent
0266334560
commit
36eacf5110
114
zcl/ops.go
Normal file
114
zcl/ops.go
Normal file
@ -0,0 +1,114 @@
|
|||||||
|
package zcl
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/zclconf/go-cty/cty"
|
||||||
|
)
|
||||||
|
|
||||||
|
// Index is a helper function that performs the same operation as the index
|
||||||
|
// operator in the zcl expression language. That is, the result is the
|
||||||
|
// same as it would be for collection[key] in a configuration expression.
|
||||||
|
//
|
||||||
|
// This is exported so that applications can perform indexing in a manner
|
||||||
|
// consistent with how the language does it, including handling of null and
|
||||||
|
// unknown values, etc.
|
||||||
|
//
|
||||||
|
// Diagnostics are produced if the given combination of values is not valid.
|
||||||
|
// Therefore a pointer to a source range must be provided to use in diagnostics,
|
||||||
|
// though nil can be provided if the calling application is going to
|
||||||
|
// ignore the subject of the returned diagnostics anyway.
|
||||||
|
func Index(collection, key cty.Value, srcRange *Range) (cty.Value, Diagnostics) {
|
||||||
|
if collection.IsNull() {
|
||||||
|
return cty.DynamicVal, Diagnostics{
|
||||||
|
{
|
||||||
|
Severity: DiagError,
|
||||||
|
Summary: "Attempt to index null value",
|
||||||
|
Detail: "This value is null, so it does not have any indices.",
|
||||||
|
Subject: srcRange,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if key.IsNull() {
|
||||||
|
return cty.DynamicVal, Diagnostics{
|
||||||
|
{
|
||||||
|
Severity: DiagError,
|
||||||
|
Summary: "Invalid index",
|
||||||
|
Detail: "Can't use a null value as an indexing key.",
|
||||||
|
Subject: srcRange,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
ty := collection.Type()
|
||||||
|
kty := key.Type()
|
||||||
|
if kty == cty.DynamicPseudoType || ty == cty.DynamicPseudoType {
|
||||||
|
return cty.DynamicVal, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
switch {
|
||||||
|
|
||||||
|
case ty.IsListType() || ty.IsTupleType() || ty.IsMapType():
|
||||||
|
has := collection.HasIndex(key)
|
||||||
|
if !has.IsKnown() {
|
||||||
|
if ty.IsTupleType() {
|
||||||
|
return cty.DynamicVal, nil
|
||||||
|
} else {
|
||||||
|
return cty.UnknownVal(ty.ElementType()), nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if has.False() {
|
||||||
|
return cty.DynamicVal, Diagnostics{
|
||||||
|
{
|
||||||
|
Severity: DiagError,
|
||||||
|
Summary: "Invalid index",
|
||||||
|
Detail: "The given index value does not identify an element in this collection value.",
|
||||||
|
Subject: srcRange,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return collection.Index(key), nil
|
||||||
|
|
||||||
|
case ty.IsObjectType():
|
||||||
|
if kty != cty.String {
|
||||||
|
return cty.DynamicVal, Diagnostics{
|
||||||
|
{
|
||||||
|
Severity: DiagError,
|
||||||
|
Summary: "Invalid index",
|
||||||
|
Detail: "The given key does not identify an element in this collection value. A string key is required.",
|
||||||
|
Subject: srcRange,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !collection.IsKnown() {
|
||||||
|
return cty.DynamicVal, nil
|
||||||
|
}
|
||||||
|
if !key.IsKnown() {
|
||||||
|
return cty.DynamicVal, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
attrName := key.AsString()
|
||||||
|
|
||||||
|
if !ty.HasAttribute(attrName) {
|
||||||
|
return cty.DynamicVal, Diagnostics{
|
||||||
|
{
|
||||||
|
Severity: DiagError,
|
||||||
|
Summary: "Invalid index",
|
||||||
|
Detail: "The given index value does not identify an element in this collection value.",
|
||||||
|
Subject: srcRange,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return collection.GetAttr(attrName), nil
|
||||||
|
|
||||||
|
default:
|
||||||
|
return cty.DynamicVal, Diagnostics{
|
||||||
|
{
|
||||||
|
Severity: DiagError,
|
||||||
|
Summary: "Invalid index",
|
||||||
|
Detail: "This value does not have any indices.",
|
||||||
|
Subject: srcRange,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -252,98 +252,7 @@ type TraverseIndex struct {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (tn TraverseIndex) TraversalStep(val cty.Value) (cty.Value, Diagnostics) {
|
func (tn TraverseIndex) TraversalStep(val cty.Value) (cty.Value, Diagnostics) {
|
||||||
if val.IsNull() {
|
return Index(val, tn.Key, &tn.SrcRange)
|
||||||
return cty.DynamicVal, Diagnostics{
|
|
||||||
{
|
|
||||||
Severity: DiagError,
|
|
||||||
Summary: "Attempt to index null value",
|
|
||||||
Detail: "This value is null, so it does not have any indices.",
|
|
||||||
Subject: &tn.SrcRange,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if tn.Key.IsNull() {
|
|
||||||
return cty.DynamicVal, Diagnostics{
|
|
||||||
{
|
|
||||||
Severity: DiagError,
|
|
||||||
Summary: "Invalid index",
|
|
||||||
Detail: "Can't use a null value as an index.",
|
|
||||||
Subject: &tn.SrcRange,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
ty := val.Type()
|
|
||||||
kty := tn.Key.Type()
|
|
||||||
if kty == cty.DynamicPseudoType {
|
|
||||||
return cty.DynamicVal, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
switch {
|
|
||||||
case ty.IsListType() || ty.IsTupleType() || ty.IsMapType():
|
|
||||||
has := val.HasIndex(tn.Key)
|
|
||||||
if !has.IsKnown() {
|
|
||||||
if ty.IsTupleType() {
|
|
||||||
return cty.DynamicVal, nil
|
|
||||||
} else {
|
|
||||||
return cty.UnknownVal(ty.ElementType()), nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if has.False() {
|
|
||||||
return cty.DynamicVal, Diagnostics{
|
|
||||||
{
|
|
||||||
Severity: DiagError,
|
|
||||||
Summary: "Invalid index",
|
|
||||||
Detail: "The given index value does not identify an element in this collection value.",
|
|
||||||
Subject: &tn.SrcRange,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return val.Index(tn.Key), nil
|
|
||||||
case ty.IsObjectType():
|
|
||||||
if kty != cty.String {
|
|
||||||
return cty.DynamicVal, Diagnostics{
|
|
||||||
{
|
|
||||||
Severity: DiagError,
|
|
||||||
Summary: "Invalid index",
|
|
||||||
Detail: "The given index value does not identify an element in this collection value.",
|
|
||||||
Subject: &tn.SrcRange,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if !val.IsKnown() {
|
|
||||||
return cty.DynamicVal, nil
|
|
||||||
}
|
|
||||||
if !tn.Key.IsKnown() {
|
|
||||||
return cty.DynamicVal, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
attrName := tn.Key.AsString()
|
|
||||||
|
|
||||||
if !ty.HasAttribute(attrName) {
|
|
||||||
return cty.DynamicVal, Diagnostics{
|
|
||||||
{
|
|
||||||
Severity: DiagError,
|
|
||||||
Summary: "Invalid index",
|
|
||||||
Detail: "The given index value does not identify an element in this collection value.",
|
|
||||||
Subject: &tn.SrcRange,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return val.GetAttr(attrName), nil
|
|
||||||
case ty == cty.DynamicPseudoType:
|
|
||||||
return cty.DynamicVal, nil
|
|
||||||
default:
|
|
||||||
return cty.DynamicVal, Diagnostics{
|
|
||||||
{
|
|
||||||
Severity: DiagError,
|
|
||||||
Summary: "Invalid index",
|
|
||||||
Detail: "This value does not have any indices.",
|
|
||||||
Subject: &tn.SrcRange,
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// TraverseSplat applies the splat operation to its initial value.
|
// TraverseSplat applies the splat operation to its initial value.
|
||||||
|
Loading…
x
Reference in New Issue
Block a user