You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
227 lines
8.0 KiB
Go
227 lines
8.0 KiB
Go
package convert
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"sort"
|
|
|
|
"github.com/zclconf/go-cty/cty"
|
|
)
|
|
|
|
// MismatchMessage is a helper to return an English-language description of
|
|
// the differences between got and want, phrased as a reason why got does
|
|
// not conform to want.
|
|
//
|
|
// This function does not itself attempt conversion, and so it should generally
|
|
// be used only after a conversion has failed, to report the conversion failure
|
|
// to an English-speaking user. The result will be confusing got is actually
|
|
// conforming to or convertable to want.
|
|
//
|
|
// The shorthand helper function Convert uses this function internally to
|
|
// produce its error messages, so callers of that function do not need to
|
|
// also use MismatchMessage.
|
|
//
|
|
// This function is similar to Type.TestConformance, but it is tailored to
|
|
// describing conversion failures and so the messages it generates relate
|
|
// specifically to the conversion rules implemented in this package.
|
|
func MismatchMessage(got, want cty.Type) string {
|
|
switch {
|
|
|
|
case got.IsObjectType() && want.IsObjectType():
|
|
// If both types are object types then we may be able to say something
|
|
// about their respective attributes.
|
|
return mismatchMessageObjects(got, want)
|
|
|
|
case got.IsTupleType() && want.IsListType() && want.ElementType() == cty.DynamicPseudoType:
|
|
// If conversion from tuple to list failed then it's because we couldn't
|
|
// find a common type to convert all of the tuple elements to.
|
|
return "all list elements must have the same type"
|
|
|
|
case got.IsTupleType() && want.IsSetType() && want.ElementType() == cty.DynamicPseudoType:
|
|
// If conversion from tuple to set failed then it's because we couldn't
|
|
// find a common type to convert all of the tuple elements to.
|
|
return "all set elements must have the same type"
|
|
|
|
case got.IsObjectType() && want.IsMapType() && want.ElementType() == cty.DynamicPseudoType:
|
|
// If conversion from object to map failed then it's because we couldn't
|
|
// find a common type to convert all of the object attributes to.
|
|
return "all map elements must have the same type"
|
|
|
|
case (got.IsTupleType() || got.IsObjectType()) && want.IsCollectionType():
|
|
return mismatchMessageCollectionsFromStructural(got, want)
|
|
|
|
case got.IsCollectionType() && want.IsCollectionType():
|
|
return mismatchMessageCollectionsFromCollections(got, want)
|
|
|
|
default:
|
|
// If we have nothing better to say, we'll just state what was required.
|
|
return want.FriendlyNameForConstraint() + " required"
|
|
}
|
|
}
|
|
|
|
func mismatchMessageObjects(got, want cty.Type) string {
|
|
// Per our conversion rules, "got" is allowed to be a superset of "want",
|
|
// and so we'll produce error messages here under that assumption.
|
|
gotAtys := got.AttributeTypes()
|
|
wantAtys := want.AttributeTypes()
|
|
|
|
// If we find missing attributes then we'll report those in preference,
|
|
// but if not then we will report a maximum of one non-conforming
|
|
// attribute, just to keep our messages relatively terse.
|
|
// We'll also prefer to report a recursive type error from an _unsafe_
|
|
// conversion over a safe one, because these are subjectively more
|
|
// "serious".
|
|
var missingAttrs []string
|
|
var unsafeMismatchAttr string
|
|
var safeMismatchAttr string
|
|
|
|
for name, wantAty := range wantAtys {
|
|
gotAty, exists := gotAtys[name]
|
|
if !exists {
|
|
if !want.AttributeOptional(name) {
|
|
missingAttrs = append(missingAttrs, name)
|
|
}
|
|
continue
|
|
}
|
|
|
|
if gotAty.Equals(wantAty) {
|
|
continue // exact match, so no problem
|
|
}
|
|
|
|
// We'll now try to convert these attributes in isolation and
|
|
// see if we have a nested conversion error to report.
|
|
// We'll try an unsafe conversion first, and then fall back on
|
|
// safe if unsafe is possible.
|
|
|
|
// If we already have an unsafe mismatch attr error then we won't bother
|
|
// hunting for another one.
|
|
if unsafeMismatchAttr != "" {
|
|
continue
|
|
}
|
|
if conv := GetConversionUnsafe(gotAty, wantAty); conv == nil {
|
|
unsafeMismatchAttr = fmt.Sprintf("attribute %q: %s", name, MismatchMessage(gotAty, wantAty))
|
|
}
|
|
|
|
// If we already have a safe mismatch attr error then we won't bother
|
|
// hunting for another one.
|
|
if safeMismatchAttr != "" {
|
|
continue
|
|
}
|
|
if conv := GetConversion(gotAty, wantAty); conv == nil {
|
|
safeMismatchAttr = fmt.Sprintf("attribute %q: %s", name, MismatchMessage(gotAty, wantAty))
|
|
}
|
|
}
|
|
|
|
// We should now have collected at least one problem. If we have more than
|
|
// one then we'll use our preference order to decide what is most important
|
|
// to report.
|
|
switch {
|
|
|
|
case len(missingAttrs) != 0:
|
|
sort.Strings(missingAttrs)
|
|
switch len(missingAttrs) {
|
|
case 1:
|
|
return fmt.Sprintf("attribute %q is required", missingAttrs[0])
|
|
case 2:
|
|
return fmt.Sprintf("attributes %q and %q are required", missingAttrs[0], missingAttrs[1])
|
|
default:
|
|
sort.Strings(missingAttrs)
|
|
var buf bytes.Buffer
|
|
for _, name := range missingAttrs[:len(missingAttrs)-1] {
|
|
fmt.Fprintf(&buf, "%q, ", name)
|
|
}
|
|
fmt.Fprintf(&buf, "and %q", missingAttrs[len(missingAttrs)-1])
|
|
return fmt.Sprintf("attributes %s are required", buf.Bytes())
|
|
}
|
|
|
|
case unsafeMismatchAttr != "":
|
|
return unsafeMismatchAttr
|
|
|
|
case safeMismatchAttr != "":
|
|
return safeMismatchAttr
|
|
|
|
default:
|
|
// We should never get here, but if we do then we'll return
|
|
// just a generic message.
|
|
return "incorrect object attributes"
|
|
}
|
|
}
|
|
|
|
func mismatchMessageCollectionsFromStructural(got, want cty.Type) string {
|
|
// First some straightforward cases where the kind is just altogether wrong.
|
|
switch {
|
|
case want.IsListType() && !got.IsTupleType():
|
|
return want.FriendlyNameForConstraint() + " required"
|
|
case want.IsSetType() && !got.IsTupleType():
|
|
return want.FriendlyNameForConstraint() + " required"
|
|
case want.IsMapType() && !got.IsObjectType():
|
|
return want.FriendlyNameForConstraint() + " required"
|
|
}
|
|
|
|
// If the kinds are matched well enough then we'll move on to checking
|
|
// individual elements.
|
|
wantEty := want.ElementType()
|
|
switch {
|
|
case got.IsTupleType():
|
|
for i, gotEty := range got.TupleElementTypes() {
|
|
if gotEty.Equals(wantEty) {
|
|
continue // exact match, so no problem
|
|
}
|
|
if conv := getConversion(gotEty, wantEty, true); conv != nil {
|
|
continue // conversion is available, so no problem
|
|
}
|
|
return fmt.Sprintf("element %d: %s", i, MismatchMessage(gotEty, wantEty))
|
|
}
|
|
|
|
// If we get down here then something weird is going on but we'll
|
|
// return a reasonable fallback message anyway.
|
|
return fmt.Sprintf("all elements must be %s", wantEty.FriendlyNameForConstraint())
|
|
|
|
case got.IsObjectType():
|
|
for name, gotAty := range got.AttributeTypes() {
|
|
if gotAty.Equals(wantEty) {
|
|
continue // exact match, so no problem
|
|
}
|
|
if conv := getConversion(gotAty, wantEty, true); conv != nil {
|
|
continue // conversion is available, so no problem
|
|
}
|
|
return fmt.Sprintf("element %q: %s", name, MismatchMessage(gotAty, wantEty))
|
|
}
|
|
|
|
// If we get down here then something weird is going on but we'll
|
|
// return a reasonable fallback message anyway.
|
|
return fmt.Sprintf("all elements must be %s", wantEty.FriendlyNameForConstraint())
|
|
|
|
default:
|
|
// Should not be possible to get here since we only call this function
|
|
// with got as structural types, but...
|
|
return want.FriendlyNameForConstraint() + " required"
|
|
}
|
|
}
|
|
|
|
func mismatchMessageCollectionsFromCollections(got, want cty.Type) string {
|
|
// First some straightforward cases where the kind is just altogether wrong.
|
|
switch {
|
|
case want.IsListType() && !(got.IsListType() || got.IsSetType()):
|
|
return want.FriendlyNameForConstraint() + " required"
|
|
case want.IsSetType() && !(got.IsListType() || got.IsSetType()):
|
|
return want.FriendlyNameForConstraint() + " required"
|
|
case want.IsMapType() && !got.IsMapType():
|
|
return want.FriendlyNameForConstraint() + " required"
|
|
}
|
|
|
|
// If the kinds are matched well enough then we'll check the element types.
|
|
gotEty := got.ElementType()
|
|
wantEty := want.ElementType()
|
|
noun := "element type"
|
|
switch {
|
|
case want.IsListType():
|
|
noun = "list element type"
|
|
case want.IsSetType():
|
|
noun = "set element type"
|
|
case want.IsMapType():
|
|
noun = "map element type"
|
|
}
|
|
return fmt.Sprintf("incorrect %s: %s", noun, MismatchMessage(gotEty, wantEty))
|
|
}
|