From f71e16b5e001826dd6294be3cdfce59a523bc463 Mon Sep 17 00:00:00 2001 From: Ian Wahbe Date: Tue, 13 Aug 2024 14:11:29 +0200 Subject: [PATCH] Enforce .Elem.Fields (instead of .Fields) for object types This is a reversal of our previous stance that .Fields should be used. This aligns the fields argument with [schema generation](https://github.com/pulumi/pulumi-terraform-bridge/blob/72b0b7d33b6d9f0dce7c09b31951ade3711a025d/pkg/tfgen/generate.go#L395-L398). We *need* to choose `.Elem.Fields` as our 1 representation (instead of `.Fields`) because of an ambiguity in `shim.Schema`. From a comment in the PR: > To prevent confusion, users are barred from specifying > information on ps directly, they must set .Fields on ps.Elem: > ps.Elem.Fields. > > We need to make this choice (instead of having users set > information on .Fields (and forbidding ps.Elem.Fields) because > the [shim.Schema] representation is not capable of > distinguishing between Map[Object] and Object. SDKv{1,2} > providers are not able to express Map[Object], but Plugin > Framework based providers are. While this is technically a breaking change, I don't expect users to be broken. Any user that would break on this change would have been unable to addapt the last several bridge versions (for example: https://github.com/pulumiverse/pulumi-talos/pull/99). Fixes https://github.com/pulumi/pulumi-terraform-bridge/issues/2185 --- pkg/tfbridge/info/validate.go | 68 ++++++++++++++---------------- pkg/tfbridge/info/validate_test.go | 16 +++---- 2 files changed, 40 insertions(+), 44 deletions(-) diff --git a/pkg/tfbridge/info/validate.go b/pkg/tfbridge/info/validate.go index 7a7e49681..50406431a 100644 --- a/pkg/tfbridge/info/validate.go +++ b/pkg/tfbridge/info/validate.go @@ -20,7 +20,6 @@ import ( "fmt" shim "github.com/pulumi/pulumi-terraform-bridge/v3/pkg/tfshim" - "github.com/pulumi/pulumi-terraform-bridge/v3/pkg/tfshim/util" "github.com/pulumi/pulumi-terraform-bridge/v3/pkg/tfshim/walk" ) @@ -83,12 +82,12 @@ func (c *infoCheck) error(path walk.SchemaPath, err error) { } var ( - errNoCorrespondingField = fmt.Errorf("overriding non-existent field") - errNoElemToOverride = fmt.Errorf("overriding non-existent elem") - errCannotSpecifyFieldsOnListOrSet = fmt.Errorf("cannot specify .Fields on a List[T] or Set[T] type") - errCannotSpecifyNameOnElem = fmt.Errorf("cannot specify .Name on a List[T], Map[T] or Set[T] type") - errCannotSetMaxItemsOne = fmt.Errorf("cannot specify .MaxItemsOne on a scalar type") - errElemForObject = fmt.Errorf("cannot set .Elem on a singly nested object block") + errNoCorrespondingField = fmt.Errorf("overriding non-existent field") + errNoElemToOverride = fmt.Errorf("overriding non-existent elem") + errCannotSpecifyFieldsOnCollection = fmt.Errorf("cannot specify .Fields on a List[T], Set[T] or Map[T] type") + errCannotSpecifyNameOnElem = fmt.Errorf("cannot specify .Name on a List[T], Map[T] or Set[T] type") + errCannotSetMaxItemsOne = fmt.Errorf("can only specify .MaxItemsOne on List[T] or Set[T] type") + errFieldsShouldBeOnElem = fmt.Errorf(".Fields should be .Elem.Fields") ) func (c *infoCheck) checkResource(tfToken string, schema shim.SchemaMap, info map[string]*Schema) { @@ -107,32 +106,8 @@ func (c *infoCheck) checkProperty(path walk.SchemaPath, tfs shim.Schema, ps *Sch return } - // s.Elem() case 2 - // - // `path` represents a singly-nested Terraform block. - // - // Conceptually `path` and `path.Elem` represent the same object. - // - // To prevent confusion, users are barred from specifying information on - // the associated Elem. All information should be specified directly on - // this SchemaInfo. - if obj, ok := util.CastToTypeObject(tfs); ok { - if ps.Elem != nil { - c.error(path, errElemForObject) - } - - if ps.MaxItemsOne != nil { - c.error(path.Element(), errCannotSetMaxItemsOne) - } - - // Now check sub-fields - c.checkFields(path, obj, ps.Fields) - - return - } - - if len(ps.Fields) > 0 { - c.error(path, errCannotSpecifyFieldsOnListOrSet) + if t := tfs.Type(); t != shim.TypeSet && t != shim.TypeList && ps.MaxItemsOne != nil { + c.error(path, errCannotSetMaxItemsOne) } switch elem := tfs.Elem().(type) { @@ -141,14 +116,35 @@ func (c *infoCheck) checkProperty(path walk.SchemaPath, tfs shim.Schema, ps *Sch // // There is a simple element type here, so we just recursively validate that. case shim.Schema: + if len(ps.Fields) > 0 { + c.error(path, errCannotSpecifyFieldsOnCollection) + } + c.checkElem(path.Element(), elem, ps.Elem) // Either `path` represents an object nested under a list or set, or `path` is // itself an object, depending on the .Type() property. case shim.Resource: - // s.Elem() case 3 + // s.Elem() case 2 and 3 + // + // `path` represents a List[elem], Set[elem], Map[elem] or Object. // - // `path` represents a List[elem] or Set[elem]. + // The [shim.Schema] representation is unable to distinguish between + // Map[elem] and Object. + + if tfs.Type() == shim.TypeMap && len(ps.Fields) > 0 { + // To prevent confusion, users are barred from specifying + // information on ps directly, they must set .Fields on ps.Elem: + // ps.Elem.Fields. + // + // We need to make this choice (instead of having users set + // information on .Fields (and forbidding ps.Elem.Fields) because + // the [shim.Schema] representation is not capable of + // distinguishing between Map[Object] and Object. SDKv{1,2} + // providers are not able to express Map[Object], but Plugin + // Framework based providers are. + c.error(path, errFieldsShouldBeOnElem) + } // Check the nested fields if ps.Elem != nil { @@ -162,7 +158,7 @@ func (c *infoCheck) checkProperty(path walk.SchemaPath, tfs shim.Schema, ps *Sch switch tfs.Type() { // s.Elem() case 5 case shim.TypeMap, shim.TypeList, shim.TypeSet: - // The type is unknown, but specifying overrides is invalid. + // The type is unknown, but specifying overrides is valid. // // We can't check any deeper, so return return diff --git a/pkg/tfbridge/info/validate_test.go b/pkg/tfbridge/info/validate_test.go index 80bab1bef..716295b62 100644 --- a/pkg/tfbridge/info/validate_test.go +++ b/pkg/tfbridge/info/validate_test.go @@ -113,8 +113,10 @@ func TestValidateNameOverride(t *testing.T) { info: &Resource{ Fields: map[string]*Schema{ "object1": { - Fields: map[string]*Schema{ - "nest1": {Name: "Foo"}, + Elem: &Schema{ + Fields: map[string]*Schema{ + "nest1": {Name: "Foo"}, + }, }, }, }, @@ -125,15 +127,13 @@ func TestValidateNameOverride(t *testing.T) { info: &Resource{ Fields: map[string]*Schema{ "object1": { - Elem: &Schema{ - Fields: map[string]*Schema{ - "nest1": {Name: "Foo"}, - }, + Fields: map[string]*Schema{ + "nest1": {Name: "Foo"}, }, }, }, }, - expectedErr: errElemForObject, + expectedErr: errFieldsShouldBeOnElem, }, { name: "invalid fields on list", @@ -146,7 +146,7 @@ func TestValidateNameOverride(t *testing.T) { }, }, }, - expectedErr: errCannotSpecifyFieldsOnListOrSet, + expectedErr: errCannotSpecifyFieldsOnCollection, }, { name: "valid max items 1",