Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

tftypes: Introduce unknown value refinement support for Value #448

Draft
wants to merge 18 commits into
base: main
Choose a base branch
from

Conversation

austinvalle
Copy link
Member

@austinvalle austinvalle commented Nov 25, 2024

Ref: hashicorp/terraform#30937
Ref: hashicorp/terraform-plugin-framework#869


This PR introduces value refinement support to the tftypes package. This allows Terraform providers to communicate unknown value refinements to/from Terraform core (Terraform 1.6+).

Value refinements are additional constraints that can be applied to unknown values in Terraform that can be used to produce known results from unknown values. For example, with value refinements, providers can now indicate specific attributes will "definitely not be null" during PlanResourceChange, which would allow Terraform core to successfully evaluate expressions like count during plan, rather than returning an error like:

│ Error: Invalid count argument
│
│ ...
│
│ The "count" value depends on resource attributes that cannot be
│ determined until apply, so Terraform cannot predict how many
│ instances will be created. To work around this, use the -target
│ argument to first apply only the resources that the count depends
│ on.

No changes to existing provider code will be necessary as partially unknown values will still evaluate the same as a wholly unknown value today.

TODOs left on this PR

  • Design is still under review
  • Changelogs

@austinvalle austinvalle added the enhancement New feature or request label Nov 25, 2024
Comment on lines +1901 to +1953
"unknown-bool": {
in: NewValue(Bool, UnknownValue),
expected: "tftypes.Bool<unknown>",
},
"unknown-bool-with-nullness-refinement": {
in: NewValue(Bool, UnknownValue).Refine(refinement.Refinements{
refinement.KeyNullness: refinement.NewNullness(false),
}),
expected: `tftypes.Bool<unknown, not null>`,
},
"unknown-string": {
in: NewValue(String, UnknownValue),
expected: "tftypes.String<unknown>",
},
"unknown-string-with-multiple-refinements": {
in: NewValue(String, UnknownValue).Refine(refinement.Refinements{
refinement.KeyNullness: refinement.NewNullness(false),
refinement.KeyStringPrefix: refinement.NewStringPrefix("str://"),
}),
expected: `tftypes.String<unknown, not null, prefix = "str://">`,
},
"unknown-number": {
in: NewValue(Number, UnknownValue),
expected: "tftypes.Number<unknown>",
},
"unknown-number-with-multiple-refinements": {
in: NewValue(Number, UnknownValue).Refine(refinement.Refinements{
refinement.KeyNullness: refinement.NewNullness(false),
refinement.KeyNumberLowerBound: refinement.NewNumberLowerBound(big.NewFloat(5), true),
refinement.KeyNumberUpperBound: refinement.NewNumberUpperBound(big.NewFloat(10), true),
}),
expected: `tftypes.Number<unknown, not null, lower bound = 5 (inclusive), upper bound = 10 (inclusive)>`,
},
"unknown-number-float-with-multiple-refinements": {
in: NewValue(Number, UnknownValue).Refine(refinement.Refinements{
refinement.KeyNullness: refinement.NewNullness(false),
refinement.KeyNumberLowerBound: refinement.NewNumberLowerBound(big.NewFloat(5.67), true),
refinement.KeyNumberUpperBound: refinement.NewNumberUpperBound(big.NewFloat(10.123), true),
}),
expected: `tftypes.Number<unknown, not null, lower bound = 5.67 (inclusive), upper bound = 10.123 (inclusive)>`,
},
"unknown-list": {
in: NewValue(List{ElementType: String}, UnknownValue),
expected: "tftypes.List[tftypes.String]<unknown>",
},
"unknown-list-with-multiple-refinements": {
in: NewValue(List{ElementType: String}, UnknownValue).Refine(refinement.Refinements{
refinement.KeyNullness: refinement.NewNullness(false),
refinement.KeyCollectionLengthLowerBound: refinement.NewCollectionLengthLowerBound(2),
refinement.KeyCollectionLengthUpperBound: refinement.NewCollectionLengthUpperBound(7),
}),
expected: `tftypes.List[tftypes.String]<unknown, not null, length lower bound = 2, length upper bound = 7>`,
},
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We can bikeshed the display of these attributes since there is no established pattern ATM (core displays the cty methods when showing data consistency errors, but doesn't have a "user friendly" version of the data defined). Other thoughts:

tftypes.String<unknown, refinements = [not null, prefix = "str://"]>
tftypes.String<unknown, will not be null, will have prefix "str://">

Comment on lines +35 to +42
// NewStringPrefix returns the StringPrefix unknown value refinement that indicates the final value will be prefixed with the specified
// string value. String prefixes that exceed 256 characters in length will be truncated and empty string prefixes will not be encoded. This
// refinement can only be applied to the String type.
func NewStringPrefix(value string) Refinement {
return StringPrefix{
value: value,
}
}
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is a potential subtle hazard here that you can read more about in the code that deals with the hazard over in the cty implementation:

https://github.com/zclconf/go-cty/blob/main/cty/ctystrings/prefix.go#L29

To reframe that in more practical terms, the hazard is that if the promised string prefix is "hi" and then the final string is "hí" (that is: h, i, followed by U+0301 combining acute accent) then cty's model of strings as a sequence of grapheme clusters (as opposed to unicode scalar values) will mean that the final value isn't considered to conform to the refinement, which will probably cause Terraform to complain that the refinements were inconsistent.

I don't recall whether Terraform actually checks this as part of its plan/apply consistency checks, but even if Terraform doesn't immediately catch and reject it the broken promise could still cause problems downstream with anything that was relying on the initial refinement.

In cty I implemented a rather heavy rule that discards any trailing character that could potentially turn out to be the start of a multi-scalar-value sequence. I think that's probably overkill for the Terraform plugin libraries because these prefixes are presumably going to be chosen directly by the provider developer rather than the end-user anyway, and so it'd probably be sufficient to just warn about this in the documentation here.

To again make that more practical/concrete: it's fine and good to say NewStringPrefix("ami-") because - is not the start of any valid combining sequence. It's also fine to announce NewStringPrefix("ami") if the provider dev knows that the final string is definitely not going to have a combining diacritic after it, which would be true for an AWS AMI ID. Where things get messy is if the character immediately after the prefix is end-user-controlled and could potentially begin with a combining character.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for sharing that @apparentlymart! I've put this work down ATM, but I'll make a note for later to come back and decide what we want to do here. Like you said, at the very least documentation should be added.

I did run a quick test and Terraform does check the refinements during it's plan/apply data consistency check, so that's good!

image
image

--- FAIL: TestSchemaResource_basic (0.89s)
    /Users/austin.valle/code/terraform-provider-corner/internal/framework6provider/schema_resource_test.go:19: Step 1/1 error: Error running apply: exit status 1
        
        Error: Provider produced inconsistent result after apply
        
        When applying changes to framework_schema.test, provider
        "provider[\"registry.terraform.io/hashicorp/framework\"]" produced an
        unexpected new value: .prefix_test: final value cty.StringVal("") does not
        conform to planning placeholder
        cty.UnknownVal(cty.String).Refine().NotNull().StringPrefixFull("hi").NewValue().
        
        This is a bug in the provider, which should be reported in the provider's own
        issue tracker.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants