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

ephemeral: support write-only attributes #36031

Merged
merged 12 commits into from
Dec 12, 2024
Merged
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Prev Previous commit
Next Next commit
ephemeral: remove write only paths
we might need them later on, but it will be easier to find the right place and abstraction when the need to use them arises
  • Loading branch information
DanielMSchmidt committed Dec 12, 2024
commit ea19aa14f8b0726987961d27df1b6930df8a25e5
6 changes: 0 additions & 6 deletions internal/plans/changes.go
Original file line number Diff line number Diff line change
Expand Up @@ -582,10 +582,6 @@ type Change struct {
// collections/structures.
Before, After cty.Value

// BeforeWriteOnlyPaths and AfterWriteOnlyPaths are paths for any values
// in Before or After (respectively) that are considered to be write-only.
BeforeWriteOnlyPaths, AfterWriteOnlyPaths []cty.Path

// Importing is present if the resource is being imported as part of this
// change.
//
Expand Down Expand Up @@ -649,8 +645,6 @@ func (c *Change) Encode(ty cty.Type) (*ChangeSrc, error) {
After: afterDV,
BeforeSensitivePaths: sensitiveAttrsBefore,
AfterSensitivePaths: sensitiveAttrsAfter,
BeforeWriteOnlyPaths: c.BeforeWriteOnlyPaths,
AfterWriteOnlyPaths: c.AfterWriteOnlyPaths,
Importing: c.Importing.Encode(),
GeneratedConfig: c.GeneratedConfig,
}, nil
Expand Down
10 changes: 0 additions & 10 deletions internal/plans/changes_src.go
Original file line number Diff line number Diff line change
Expand Up @@ -130,12 +130,6 @@ func (c *ChangesSrc) Decode(schemas *schemarepo.Schemas) (*Changes, error) {
rc.Before = marks.MarkPaths(rc.Before, marks.Sensitive, rcs.BeforeSensitivePaths)
rc.After = marks.MarkPaths(rc.After, marks.Sensitive, rcs.AfterSensitivePaths)

rc.Before = marks.MarkPaths(rc.Before, marks.Ephemeral, rcs.BeforeWriteOnlyPaths)
rc.After = marks.MarkPaths(rc.After, marks.Ephemeral, rcs.BeforeWriteOnlyPaths)

rc.BeforeWriteOnlyPaths = rcs.BeforeWriteOnlyPaths
rc.AfterWriteOnlyPaths = rcs.AfterWriteOnlyPaths

changes.Resources = append(changes.Resources, rc)
}

Expand Down Expand Up @@ -394,10 +388,6 @@ type ChangeSrc struct {
// the serialized change.
BeforeSensitivePaths, AfterSensitivePaths []cty.Path

// BeforeWriteOnlyPaths and AfterWriteOnlyPaths are paths for any values
// in Before or After (respectively) that are considered to be write-only.
BeforeWriteOnlyPaths, AfterWriteOnlyPaths []cty.Path

// Importing is present if the resource is being imported as part of this
// change.
//
Expand Down
5 changes: 0 additions & 5 deletions internal/states/instance_object.go
Original file line number Diff line number Diff line change
Expand Up @@ -50,10 +50,6 @@ type ResourceInstanceObject struct {
// destroy operations, we need to record the status to ensure a resource
// removed from the config will still be destroyed in the same manner.
CreateBeforeDestroy bool

// AttrWriteOnlyPaths is an array of paths to mark as ephemeral coming out of
// state, or to save as write_only paths when saving state
AttrWriteOnlyPaths []cty.Path
}

// ObjectStatus represents the status of a RemoteObject.
Expand Down Expand Up @@ -139,7 +135,6 @@ func (o *ResourceInstanceObject) Encode(ty cty.Type, schemaVersion uint64) (*Res
SchemaVersion: schemaVersion,
AttrsJSON: src,
AttrSensitivePaths: sensitivePaths,
AttrWriteOnlyPaths: o.AttrWriteOnlyPaths,
Private: o.Private,
Status: o.Status,
Dependencies: dependencies,
Expand Down
6 changes: 0 additions & 6 deletions internal/states/instance_object_src.go
Original file line number Diff line number Diff line change
Expand Up @@ -57,10 +57,6 @@ type ResourceInstanceObjectSrc struct {
// state, or to save as sensitive paths when saving state
AttrSensitivePaths []cty.Path

// AttrWriteOnlyPaths is an array of paths to mark as ephemeral coming out of
// state, or to save as write_only paths when saving state
AttrWriteOnlyPaths []cty.Path

// These fields all correspond to the fields of the same name on
// ResourceInstanceObject.
Private []byte
Expand Down Expand Up @@ -100,7 +96,6 @@ func (os *ResourceInstanceObjectSrc) Decode(ty cty.Type) (*ResourceInstanceObjec
default:
val, err = ctyjson.Unmarshal(os.AttrsJSON, ty)
val = marks.MarkPaths(val, marks.Sensitive, os.AttrSensitivePaths)
val = marks.MarkPaths(val, marks.Ephemeral, os.AttrWriteOnlyPaths)
if err != nil {
return nil, err
}
Expand All @@ -112,7 +107,6 @@ func (os *ResourceInstanceObjectSrc) Decode(ty cty.Type) (*ResourceInstanceObjec
Dependencies: os.Dependencies,
Private: os.Private,
CreateBeforeDestroy: os.CreateBeforeDestroy,
AttrWriteOnlyPaths: os.AttrWriteOnlyPaths,
}, nil
}

Expand Down
10 changes: 4 additions & 6 deletions internal/states/remote/state_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -129,9 +129,8 @@ func TestStatePersist(t *testing.T) {
"attributes_flat": map[string]interface{}{
"filename": "file.txt",
},
"schema_version": 0.0,
"sensitive_attributes": []interface{}{},
"write_only_attributes": []interface{}{},
"schema_version": 0.0,
"sensitive_attributes": []interface{}{},
},
},
"mode": "managed",
Expand Down Expand Up @@ -168,9 +167,8 @@ func TestStatePersist(t *testing.T) {
"attributes_flat": map[string]interface{}{
"filename": "file.txt",
},
"schema_version": 0.0,
"sensitive_attributes": []interface{}{},
"write_only_attributes": []interface{}{},
"schema_version": 0.0,
"sensitive_attributes": []interface{}{},
},
},
"mode": "managed",
Expand Down
7 changes: 0 additions & 7 deletions internal/states/state_deepcopy.go
Original file line number Diff line number Diff line change
Expand Up @@ -148,12 +148,6 @@ func (os *ResourceInstanceObjectSrc) DeepCopy() *ResourceInstanceObjectSrc {
copy(sensitiveAttrPaths, os.AttrSensitivePaths)
}

var writeOnlyAttrPaths []cty.Path
if os.AttrWriteOnlyPaths != nil {
writeOnlyAttrPaths = make([]cty.Path, len(os.AttrWriteOnlyPaths))
copy(writeOnlyAttrPaths, os.AttrWriteOnlyPaths)
}

var private []byte
if os.Private != nil {
private = make([]byte, len(os.Private))
Expand All @@ -175,7 +169,6 @@ func (os *ResourceInstanceObjectSrc) DeepCopy() *ResourceInstanceObjectSrc {
AttrsFlat: attrsFlat,
AttrsJSON: attrsJSON,
AttrSensitivePaths: sensitiveAttrPaths,
AttrWriteOnlyPaths: writeOnlyAttrPaths,
Dependencies: dependencies,
CreateBeforeDestroy: os.CreateBeforeDestroy,
decodeValueCache: os.decodeValueCache,
Expand Down
14 changes: 0 additions & 14 deletions internal/states/statefile/version4.go
Original file line number Diff line number Diff line change
Expand Up @@ -166,16 +166,6 @@ func prepareStateV4(sV4 *stateV4) (*File, tfdiags.Diagnostics) {
obj.AttrSensitivePaths = paths
}

// write-only paths
if isV4.AttributeWriteOnlyPaths != nil {
paths, pathsDiags := unmarshalPaths([]byte(isV4.AttributeWriteOnlyPaths))
diags = diags.Append(pathsDiags)
if pathsDiags.HasErrors() {
continue
}
obj.AttrWriteOnlyPaths = paths
}

{
// Status
raw := isV4.Status
Expand Down Expand Up @@ -492,8 +482,6 @@ func appendInstanceObjectStateV4(rs *states.Resource, is *states.ResourceInstanc
// Marshal paths to JSON
attributeSensitivePaths, pathsDiags := marshalPaths(obj.AttrSensitivePaths)
diags = diags.Append(pathsDiags)
attributeWriteOnlyPaths, pathsDiags := marshalPaths(obj.AttrWriteOnlyPaths)
diags = diags.Append(pathsDiags)

return append(isV4s, instanceObjectStateV4{
IndexKey: rawKey,
Expand All @@ -503,7 +491,6 @@ func appendInstanceObjectStateV4(rs *states.Resource, is *states.ResourceInstanc
AttributesFlat: obj.AttrsFlat,
AttributesRaw: obj.AttrsJSON,
AttributeSensitivePaths: attributeSensitivePaths,
AttributeWriteOnlyPaths: attributeWriteOnlyPaths,
PrivateRaw: privateRaw,
Dependencies: deps,
CreateBeforeDestroy: obj.CreateBeforeDestroy,
Expand Down Expand Up @@ -714,7 +701,6 @@ type instanceObjectStateV4 struct {
AttributesRaw json.RawMessage `json:"attributes,omitempty"`
AttributesFlat map[string]string `json:"attributes_flat,omitempty"`
AttributeSensitivePaths json.RawMessage `json:"sensitive_attributes,omitempty"`
AttributeWriteOnlyPaths json.RawMessage `json:"write_only_attributes,omitempty"`

PrivateRaw []byte `json:"private,omitempty"`

Expand Down
36 changes: 0 additions & 36 deletions internal/terraform/context_apply_ephemeral_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -420,10 +420,6 @@ resource "ephem_write_only" "wo" {
t.Fatalf("Expected 1 resource change, got %d", len(plan.Changes.Resources))
}

if len(plan.Changes.Resources[0].AfterWriteOnlyPaths) != 1 {
t.Fatalf("Expected 1 write-only attribute in plan, got %d", len(plan.Changes.Resources[0].AfterWriteOnlyPaths))
}

schemas, schemaDiags := ctx.Schemas(m, plan.PriorState)
assertNoDiagnostics(t, schemaDiags)
planChanges, err := plan.Changes.Decode(schemas)
Expand Down Expand Up @@ -475,14 +471,6 @@ resource "ephem_write_only" "wo" {
if !attrs.Value.GetAttr("write_only").IsNull() {
t.Fatalf("write_only attribute should be null")
}

if len(resourceInstance.Current.AttrWriteOnlyPaths) != 1 {
t.Fatalf("Expected 1 write only attribute in apply")
}

if !resourceInstance.Current.AttrWriteOnlyPaths[0].Equals(cty.GetAttrPath("write_only")) {
t.Fatalf("Expected write_only to be a write only attribute")
}
}

func TestContext2Apply_update_write_only_attribute_not_in_plan_and_state(t *testing.T) {
Expand Down Expand Up @@ -560,10 +548,6 @@ resource "ephem_write_only" "wo" {
t.Fatalf("Expected 1 resource change, got %d", len(plan.Changes.Resources))
}

if len(plan.Changes.Resources[0].AfterWriteOnlyPaths) != 1 {
t.Fatalf("Expected 1 write-only attribute in plan, got %d", len(plan.Changes.Resources[0].AfterWriteOnlyPaths))
}

schemas, schemaDiags := ctx.Schemas(m, plan.PriorState)
assertNoDiagnostics(t, schemaDiags)
planChanges, err := plan.Changes.Decode(schemas)
Expand Down Expand Up @@ -615,14 +599,6 @@ resource "ephem_write_only" "wo" {
if !attrs.Value.GetAttr("write_only").IsNull() {
t.Fatalf("write_only attribute should be null")
}

if len(resourceInstance.Current.AttrWriteOnlyPaths) != 1 {
t.Fatalf("Expected 1 write only attribute in apply")
}

if !resourceInstance.Current.AttrWriteOnlyPaths[0].Equals(cty.GetAttrPath("write_only")) {
t.Fatalf("Expected write_only to be a write only attribute")
}
}

func TestContext2Apply_normal_attributes_becomes_write_only_attribute(t *testing.T) {
Expand Down Expand Up @@ -701,10 +677,6 @@ resource "ephem_write_only" "wo" {
t.Fatalf("Expected 1 resource change, got %d", len(plan.Changes.Resources))
}

if len(plan.Changes.Resources[0].AfterWriteOnlyPaths) != 1 {
t.Fatalf("Expected 1 write-only attribute in plan, got %d", len(plan.Changes.Resources[0].AfterWriteOnlyPaths))
}

schemas, schemaDiags := ctx.Schemas(m, plan.PriorState)
assertNoDiagnostics(t, schemaDiags)
planChanges, err := plan.Changes.Decode(schemas)
Expand Down Expand Up @@ -756,12 +728,4 @@ resource "ephem_write_only" "wo" {
if !attrs.Value.GetAttr("write_only").IsNull() {
t.Fatalf("write_only attribute should be null")
}

if len(resourceInstance.Current.AttrWriteOnlyPaths) != 1 {
t.Fatalf("Expected 1 write only attribute in apply")
}

if !resourceInstance.Current.AttrWriteOnlyPaths[0].Equals(cty.GetAttrPath("write_only")) {
t.Fatalf("Expected write_only to be a write only attribute")
}
}
10 changes: 3 additions & 7 deletions internal/terraform/node_resource_abstract_instance.go
Original file line number Diff line number Diff line change
Expand Up @@ -1202,9 +1202,6 @@ func (n *NodeAbstractResourceInstance) plan(
// Marks will be removed when encoding.
After: plannedNewVal,
GeneratedConfig: n.generatedConfigHCL,

BeforeWriteOnlyPaths: ephemeral.EphemeralValuePaths(priorVal),
AfterWriteOnlyPaths: ephemeralValuePaths,
},
ActionReason: actionReason,
RequiredReplace: reqRep,
Expand All @@ -1218,10 +1215,9 @@ func (n *NodeAbstractResourceInstance) plan(
// must _also_ record the returned change in the active plan,
// which the expression evaluator will use in preference to this
// incomplete value recorded in the state.
Status: states.ObjectPlanned,
Value: plannedNewVal,
Private: plannedPrivate,
AttrWriteOnlyPaths: ephemeralValuePaths,
Status: states.ObjectPlanned,
Value: plannedNewVal,
Private: plannedPrivate,
}

return plan, state, deferred, keyData, diags
Expand Down
5 changes: 0 additions & 5 deletions internal/terraform/node_resource_apply_instance.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@ import (
"github.com/hashicorp/terraform/internal/addrs"
"github.com/hashicorp/terraform/internal/configs"
"github.com/hashicorp/terraform/internal/instances"
"github.com/hashicorp/terraform/internal/lang/ephemeral"
"github.com/hashicorp/terraform/internal/plans"
"github.com/hashicorp/terraform/internal/plans/objchange"
"github.com/hashicorp/terraform/internal/providers"
Expand Down Expand Up @@ -324,10 +323,6 @@ func (n *NodeApplyableResourceInstance) managedResourceExecute(ctx EvalContext)
if state != nil {
// dependencies are always updated to match the configuration during apply
state.Dependencies = n.Dependencies

// The value is updated to remove ephemeral values
state.Value = ephemeral.RemoveEphemeralValues(state.Value)
state.AttrWriteOnlyPaths = diffApply.AfterWriteOnlyPaths
}
err = n.writeResourceInstanceState(ctx, state, workingState)
if err != nil {
Expand Down