Skip to content

Commit

Permalink
[BREAKING] DefaultCheck now accepts a context.Context
Browse files Browse the repository at this point in the history
Following up on #252, this allows `DefaultCheck` to safely and accurately handle secrets
application. The new signature for `DefaultCheck` is easier to extend without additional
breaking changes. The downside to this design is that this makes `DefaultCheck` special. I
think it's worth living with that until #212 is resolved.

I'd like to merge shortly after #252 (and before #252 is released) so that user's don't
rely on #252 injecting secrets when `CustomCheck` is implemented and `DefaultCheck` is not
called.
  • Loading branch information
iwahbe committed Jul 24, 2024
1 parent e71fef2 commit a47da29
Show file tree
Hide file tree
Showing 6 changed files with 89 additions and 39 deletions.
2 changes: 1 addition & 1 deletion examples/file/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -112,7 +112,7 @@ func (*File) Check(ctx context.Context, name string, oldInputs, newInputs resour
if _, ok := newInputs["path"]; !ok {
newInputs["path"] = resource.NewStringProperty(name)
}
return infer.DefaultCheck[FileArgs](newInputs)
return infer.DefaultCheck[FileArgs](ctx, newInputs)
}

func (*File) Update(ctx context.Context, id string, olds FileState, news FileArgs, preview bool) (FileState, error) {
Expand Down
2 changes: 1 addition & 1 deletion infer/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -231,7 +231,7 @@ func (*File) Check(ctx context.Context, name string, oldInputs, newInputs resour
if _, ok := newInputs["path"]; !ok {
newInputs["path"] = resource.NewStringProperty(name)
}
return infer.DefaultCheck[FileArgs](newInputs)
return infer.DefaultCheck[FileArgs](ctx, newInputs)
}
```

Expand Down
8 changes: 7 additions & 1 deletion infer/configuration.go
Original file line number Diff line number Diff line change
Expand Up @@ -93,17 +93,23 @@ func (c *config[T]) checkConfig(ctx context.Context, req p.CheckRequest) (p.Chec
if req.Urn != "" {
req.Urn.Name()
}
defaultCheckEncoder := new(defaultCheckEncoderValue)
ctx = context.WithValue(ctx, defaultCheckEncoderKey{}, defaultCheckEncoder)
i, failures, err := t.Check(ctx, name, req.Olds, req.News)
if err != nil {
return p.CheckResponse{}, err
}

if defaultCheckEncoder.enc != nil {
encoder = *defaultCheckEncoder.enc
}

inputs, err := encoder.Encode(i)
if err != nil {
return p.CheckResponse{}, err
}
return p.CheckResponse{
Inputs: applySecrets[T](inputs),
Inputs: inputs,
Failures: failures,
}, nil
}
Expand Down
31 changes: 24 additions & 7 deletions infer/resource.go
Original file line number Diff line number Diff line change
Expand Up @@ -877,12 +877,18 @@ func (rc *derivedResourceController[R, I, O]) Check(ctx context.Context, req p.C
if r, ok := ((interface{})(r)).(CustomCheck[I]); ok {
// The user implemented check manually, so call that.
//
// We do not apply defaults if the user has implemented Check
// themselves. Defaults are applied by [DefaultCheck].
// We do not apply defaults or secrets if the user has implemented Check
// themselves. Defaults and secrets are applied by [DefaultCheck].
defaultCheckEncoder := new(defaultCheckEncoderValue)
ctx = context.WithValue(ctx, defaultCheckEncoderKey{}, defaultCheckEncoder)
i, failures, err := r.Check(ctx, req.Urn.Name(), req.Olds, req.News)
if err != nil {
return p.CheckResponse{}, err
}
if defaultCheckEncoder.enc != nil {
encoder = *defaultCheckEncoder.enc
}

inputs, err := encoder.Encode(i)
return p.CheckResponse{
Inputs: inputs,
Expand All @@ -899,12 +905,23 @@ func (rc *derivedResourceController[R, I, O]) Check(ctx context.Context, req p.C
return p.CheckResponse{Inputs: applySecrets[I](inputs)}, err
}

// DefaultCheck verifies that `inputs` can deserialize cleanly into `I`. This is the default
// validation that is performed when leaving `Check` unimplemented.
// It also adds defaults to `inputs` as necessary, as defined by `Annotator.SetDefault“.
func DefaultCheck[I any](inputs resource.PropertyMap) (I, []p.CheckFailure, error) {
type (
defaultCheckEncoderKey struct{}
defaultCheckEncoderValue struct{ enc *ende.Encoder }
)

// DefaultCheck verifies that inputs can deserialize cleanly into I. This is the default
// validation that is performed when leaving Check unimplemented.
//
// It also adds defaults to inputs as necessary, as defined by [Annotator.SetDefault].
func DefaultCheck[I any](ctx context.Context, inputs resource.PropertyMap) (I, []p.CheckFailure, error) {
inputs = applySecrets[I](inputs)
_, i, failures, err := decodeCheckingMapErrors[I](inputs)
enc, i, failures, err := decodeCheckingMapErrors[I](inputs)

if v, ok := ctx.Value(defaultCheckEncoderKey{}).(*defaultCheckEncoderValue); ok {
v.enc = &enc
}

if err != nil || len(failures) > 0 {
return i, failures, err
}
Expand Down
2 changes: 1 addition & 1 deletion infer/resource_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -528,7 +528,7 @@ func TestCheck(t *testing.T) {

t.Run("DefaultCheck "+tcName, func(t *testing.T) {
t.Parallel()
in, failures, err := DefaultCheck[checkResource](tc.input.Copy())
in, failures, err := DefaultCheck[checkResource](context.Background(), tc.input.Copy())
require.NoError(t, err)
assert.Empty(t, failures)
assert.Equal(t, tc.expected, in.P1)
Expand Down
83 changes: 55 additions & 28 deletions tests/config_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -123,12 +123,9 @@ func TestInferCheckConfigSecrets(t *testing.T) {
}

type config struct {
Field string `pulumi:"field" provider:"secret"`
Nested struct {
Int int `pulumi:"int" provider:"secret"`
NotSecret string `pulumi:"not-nested"`
} `pulumi:"nested"`
NotSecret string `pulumi:"not"`
Field string `pulumi:"field" provider:"secret"`
NotSecret string `pulumi:"not"`
ApplyDefaults bool `pulumi:"applyDefaults,optional"`
}

var _ infer.CustomCheck[*config] = &config{}
Expand All @@ -140,33 +137,63 @@ func (c *config) Check(
return c, nil, fmt.Errorf("found secrets")
}

d, f, err := infer.DefaultCheck[config](newInputs)
return &d, f, err
if v, ok := newInputs["applyDefaults"]; ok && v.IsBool() && v.BoolValue() {
d, f, err := infer.DefaultCheck[config](ctx, newInputs)
*c = d
return &d, f, err
}

// No defaults, so apply manually
if v := newInputs["field"]; v.IsString() {
c.Field = v.StringValue()
}
if v := newInputs["not"]; v.IsString() {
c.NotSecret = v.StringValue()
}
if v := newInputs["apply-defaults"]; v.IsBool() {
c.ApplyDefaults = v.BoolValue()
}
return c, nil, nil
}

func TestInferCustomCheckConfig(t *testing.T) {
t.Parallel()

resp, err := integration.NewServer("test", semver.MustParse("0.0.0"), infer.Provider(infer.Options{
s := integration.NewServer("test", semver.MustParse("0.0.0"), infer.Provider(infer.Options{
Config: infer.Config[*config](),
})).CheckConfig(p.CheckRequest{
News: resource.PropertyMap{
"field": resource.NewProperty("value"),
"nested": resource.NewProperty(resource.PropertyMap{
"int": resource.NewProperty(1.0),
"not-nested": resource.NewProperty("not-secret"),
}),
"not": resource.NewProperty("not-secret"),
},
}))

t.Run("with-defaults", func(t *testing.T) {
resp, err := s.CheckConfig(p.CheckRequest{
News: resource.PropertyMap{
"field": resource.NewProperty("value"),
"not": resource.NewProperty("not-secret"),
"applyDefaults": resource.NewProperty(true),
},
})
require.NoError(t, err)
require.Empty(t, resp.Failures)
assert.Equal(t, resource.PropertyMap{
"field": resource.MakeSecret(resource.NewProperty("value")),
"not": resource.NewProperty("not-secret"),
"applyDefaults": resource.NewProperty(true),
}, resp.Inputs)
})

t.Run("no-defaults", func(t *testing.T) {
resp, err := s.CheckConfig(p.CheckRequest{
News: resource.PropertyMap{
"field": resource.NewProperty("value"),
"not": resource.NewProperty("not-secret"),
"applyDefaults": resource.NewProperty(false),
},
})
require.NoError(t, err)
require.Empty(t, resp.Failures)
assert.Equal(t, resource.PropertyMap{
"field": resource.NewProperty("value"),
"not": resource.NewProperty("not-secret"),
"applyDefaults": resource.NewProperty(false),
}, resp.Inputs)
})
require.NoError(t, err)
require.Empty(t, resp.Failures)
assert.Equal(t, resource.PropertyMap{
"field": resource.MakeSecret(resource.NewProperty("value")),
"nested": resource.NewProperty(resource.PropertyMap{
"int": resource.MakeSecret(resource.NewProperty(1.0)),
"not-nested": resource.NewProperty("not-secret"),
}),
"not": resource.NewProperty("not-secret"),
}, resp.Inputs)
}

0 comments on commit a47da29

Please sign in to comment.