From 0f1206a6de376e443c7a58f7a6a2f1b3edf629a2 Mon Sep 17 00:00:00 2001 From: Venelin Date: Wed, 11 Dec 2024 16:30:57 +0000 Subject: [PATCH 1/4] Ensure all sdkv2 test providers have an id and an update method --- pkg/internal/tests/pulcheck/pulcheck.go | 26 ++++++++++++++++++++++++- 1 file changed, 25 insertions(+), 1 deletion(-) diff --git a/pkg/internal/tests/pulcheck/pulcheck.go b/pkg/internal/tests/pulcheck/pulcheck.go index 94c00e781..66dd04fb9 100644 --- a/pkg/internal/tests/pulcheck/pulcheck.go +++ b/pkg/internal/tests/pulcheck/pulcheck.go @@ -40,6 +40,8 @@ func propNeedsUpdate(prop *schema.Schema) bool { return true } +// resourceNeedsUpdate returns true if the TF SDK schema validation would consider the +// resource to need an update method. func resourceNeedsUpdate(res *schema.Resource) bool { // If any of the properties need an update, then the resource needs an update. for _, s := range res.Schema { @@ -53,6 +55,26 @@ func resourceNeedsUpdate(res *schema.Resource) bool { // This is an experimental API. func EnsureProviderValid(t T, tfp *schema.Provider) { for _, r := range tfp.ResourcesMap { + if r.Schema["id"] == nil { + r.Schema["id"] = &schema.Schema{ + Type: schema.TypeString, + Computed: true, + } + } + + // If the resource will be flagged as not needing an Update method by the TF SDK schema + // validation then add a no-op update_prop property that will instead force the resource + // to need an Update method. + // This is necessary to work around a bug in the TF SDK schema validation where some resources which + // do need an Update method will instead be flagged as not needing an Update method. + // See https://github.com/pulumi/pulumi-terraform-bridge/pull/2723#issuecomment-2541518646 + if !resourceNeedsUpdate(r) { + r.Schema["update_prop"] = &schema.Schema{ + Type: schema.TypeString, + Optional: true, + } + } + if r.ReadContext == nil { r.ReadContext = func(_ context.Context, d *schema.ResourceData, m interface{}) diag.Diagnostics { return nil @@ -75,7 +97,9 @@ func EnsureProviderValid(t T, tfp *schema.Provider) { } } - if resourceNeedsUpdate(r) && r.UpdateContext == nil { + // Because of the no-op update_prop property added above, all resources will now + // need an Update method. + if r.UpdateContext == nil { r.UpdateContext = func( ctx context.Context, rd *schema.ResourceData, i interface{}, ) diag.Diagnostics { From 43fe49d2e4a7dcfda5e0752e25d44179ecd9350f Mon Sep 17 00:00:00 2001 From: Venelin Date: Wed, 11 Dec 2024 16:31:33 +0000 Subject: [PATCH 2/4] SDKv2 detailed diff set tests for replacement From d168ffb562f85edab84aa5c92c1b973441dda639 Mon Sep 17 00:00:00 2001 From: Venelin Date: Wed, 11 Dec 2024 16:31:58 +0000 Subject: [PATCH 3/4] SDKv2 detailed diff set tests for replacement --- pkg/tests/detailed_diff_set_test.go | 89 +++++++++++++++---- .../attribute_force_new/added.golden | 0 .../attribute_force_new/added_end.golden | 0 .../added_end_unordered.golden | 0 .../attribute_force_new/added_front.golden | 0 .../added_front_unordered.golden | 0 .../attribute_force_new/added_middle.golden | 0 .../added_middle_unordered.golden | 0 .../changed_empty_to_null.golden | 0 .../changed_non-null.golden | 0 .../changed_non-null_to_null.golden | 0 .../changed_null_to_empty.golden | 0 .../changed_null_to_non-null.golden | 0 .../attribute_force_new/removed.golden | 0 .../attribute_force_new/removed_end.golden | 0 .../removed_end_unordered.golden | 0 .../attribute_force_new/removed_front.golden | 0 .../removed_front_unordered.golden | 0 .../attribute_force_new/removed_middle.golden | 0 .../removed_middle_unordered.golden | 0 .../same_element_updated.golden | 0 .../same_element_updated_unordered.golden | 0 .../attribute_force_new/shuffled.golden | 0 .../shuffled_added_end.golden | 0 .../shuffled_added_front.golden | 0 .../shuffled_added_middle.golden | 0 .../shuffled_removed_end.golden | 0 .../shuffled_removed_front.golden | 0 .../shuffled_removed_middle.golden | 0 .../shuffled_unordered.golden | 0 .../shuffled_with_duplicates.golden | 0 .../shuffled_with_duplicates_unordered.golden | 0 .../attribute_force_new/two_added.golden | 0 .../two_added_and_two_removed.golden | 0 ...d_two_removed_shuffled,_no_overlaps.golden | 0 ..._two_removed_shuffled,_one_overlaps.golden | 0 ...o_removed_shuffled,_with_duplicates.golden | 0 .../attribute_force_new/two_removed.golden | 0 .../unchanged_empty.golden | 0 .../unchanged_non-empty.golden | 0 .../attribute_force_new/unchanged_null.golden | 0 .../attribute_no_force_new/added.golden | 0 .../attribute_no_force_new/added_end.golden | 0 .../added_end_unordered.golden | 0 .../attribute_no_force_new/added_front.golden | 0 .../added_front_unordered.golden | 0 .../added_middle.golden | 0 .../added_middle_unordered.golden | 0 .../changed_empty_to_null.golden | 0 .../changed_non-null.golden | 0 .../changed_non-null_to_null.golden | 0 .../changed_null_to_empty.golden | 0 .../changed_null_to_non-null.golden | 0 .../attribute_no_force_new/removed.golden | 0 .../attribute_no_force_new/removed_end.golden | 0 .../removed_end_unordered.golden | 0 .../removed_front.golden | 0 .../removed_front_unordered.golden | 0 .../removed_middle.golden | 0 .../removed_middle_unordered.golden | 0 .../same_element_updated.golden | 0 .../same_element_updated_unordered.golden | 0 .../attribute_no_force_new/shuffled.golden | 0 .../shuffled_added_end.golden | 0 .../shuffled_added_front.golden | 0 .../shuffled_added_middle.golden | 0 .../shuffled_removed_end.golden | 0 .../shuffled_removed_front.golden | 0 .../shuffled_removed_middle.golden | 0 .../shuffled_unordered.golden | 0 .../shuffled_with_duplicates.golden | 0 .../shuffled_with_duplicates_unordered.golden | 0 .../attribute_no_force_new/two_added.golden | 0 .../two_added_and_two_removed.golden | 0 ...d_two_removed_shuffled,_no_overlaps.golden | 0 ..._two_removed_shuffled,_one_overlaps.golden | 0 ...o_removed_shuffled,_with_duplicates.golden | 0 .../attribute_no_force_new/two_removed.golden | 0 .../unchanged_empty.golden | 0 .../unchanged_non-empty.golden | 0 .../unchanged_null.golden | 0 .../block_nested_force_new}/added.golden | 0 .../block_nested_force_new}/added_end.golden | 0 .../added_end_unordered.golden | 0 .../added_front.golden | 0 .../added_front_unordered.golden | 0 .../added_middle.golden | 0 .../added_middle_unordered.golden | 0 .../changed_empty_to_null.golden | 0 .../changed_non-null.golden | 0 .../changed_non-null_to_null.golden | 0 .../changed_null_to_empty.golden | 0 .../changed_null_to_non-null.golden | 0 .../block_nested_force_new}/removed.golden | 0 .../removed_end.golden | 0 .../removed_end_unordered.golden | 0 .../removed_front.golden | 0 .../removed_front_unordered.golden | 0 .../removed_middle.golden | 0 .../removed_middle_unordered.golden | 0 .../same_element_updated.golden | 0 .../same_element_updated_unordered.golden | 0 .../block_nested_force_new}/shuffled.golden | 0 .../shuffled_added_end.golden | 0 .../shuffled_added_front.golden | 0 .../shuffled_added_middle.golden | 0 .../shuffled_removed_end.golden | 0 .../shuffled_removed_front.golden | 0 .../shuffled_removed_middle.golden | 0 .../shuffled_unordered.golden | 0 .../shuffled_with_duplicates.golden | 0 .../shuffled_with_duplicates_unordered.golden | 0 .../block_nested_force_new}/two_added.golden | 0 .../two_added_and_two_removed.golden | 0 ...d_two_removed_shuffled,_no_overlaps.golden | 0 ..._two_removed_shuffled,_one_overlaps.golden | 0 ...o_removed_shuffled,_with_duplicates.golden | 0 .../two_removed.golden | 0 .../unchanged_empty.golden | 0 .../unchanged_non-empty.golden | 0 .../unchanged_null.golden | 0 .../block_no_force_new/added.golden | 0 .../block_no_force_new/added_end.golden | 0 .../added_end_unordered.golden | 0 .../block_no_force_new/added_front.golden | 0 .../added_front_unordered.golden | 0 .../block_no_force_new/added_middle.golden | 0 .../added_middle_unordered.golden | 0 .../changed_empty_to_null.golden | 0 .../changed_non-null.golden | 0 .../changed_non-null_to_null.golden | 0 .../changed_null_to_empty.golden | 0 .../changed_null_to_non-null.golden | 0 .../block_no_force_new/removed.golden | 0 .../block_no_force_new/removed_end.golden | 0 .../removed_end_unordered.golden | 0 .../block_no_force_new/removed_front.golden | 0 .../removed_front_unordered.golden | 0 .../block_no_force_new/removed_middle.golden | 0 .../removed_middle_unordered.golden | 0 .../same_element_updated.golden | 0 .../same_element_updated_unordered.golden | 0 .../block_no_force_new/shuffled.golden | 0 .../shuffled_added_end.golden | 0 .../shuffled_added_front.golden | 0 .../shuffled_added_middle.golden | 0 .../shuffled_removed_end.golden | 0 .../shuffled_removed_front.golden | 0 .../shuffled_removed_middle.golden | 0 .../shuffled_unordered.golden | 0 .../shuffled_with_duplicates.golden | 0 .../shuffled_with_duplicates_unordered.golden | 0 .../block_no_force_new/two_added.golden | 0 .../two_added_and_two_removed.golden | 0 ...d_two_removed_shuffled,_no_overlaps.golden | 0 ..._two_removed_shuffled,_one_overlaps.golden | 0 ...o_removed_shuffled,_with_duplicates.golden | 0 .../block_no_force_new/two_removed.golden | 0 .../block_no_force_new/unchanged_empty.golden | 0 .../unchanged_non-empty.golden | 0 .../block_no_force_new/unchanged_null.golden | 0 .../block_top_level_force_new/added.golden | 39 ++++++++ .../added_end.golden | 48 ++++++++++ .../added_end_unordered.golden | 48 ++++++++++ .../added_front.golden | 48 ++++++++++ .../added_front_unordered.golden | 48 ++++++++++ .../added_middle.golden | 48 ++++++++++ .../added_middle_unordered.golden | 48 ++++++++++ .../changed_empty_to_null.golden | 15 ++++ .../changed_non-null.golden | 44 +++++++++ .../changed_non-null_to_null.golden | 40 +++++++++ .../changed_null_to_empty.golden | 15 ++++ .../changed_null_to_non-null.golden | 40 +++++++++ .../block_top_level_force_new/removed.golden | 41 +++++++++ .../removed_end.golden | 48 ++++++++++ .../removed_end_unordered.golden | 48 ++++++++++ .../removed_front.golden | 48 ++++++++++ .../removed_front_unordered.golden | 48 ++++++++++ .../removed_middle.golden | 48 ++++++++++ .../removed_middle_unordered.golden | 48 ++++++++++ .../same_element_updated.golden | 52 +++++++++++ .../same_element_updated_unordered.golden | 58 ++++++++++++ .../block_top_level_force_new/shuffled.golden | 24 +++++ .../shuffled_added_end.golden | 48 ++++++++++ .../shuffled_added_front.golden | 48 ++++++++++ .../shuffled_added_middle.golden | 48 ++++++++++ .../shuffled_removed_end.golden | 48 ++++++++++ .../shuffled_removed_front.golden | 48 ++++++++++ .../shuffled_removed_middle.golden | 48 ++++++++++ .../shuffled_unordered.golden | 24 +++++ .../shuffled_with_duplicates.golden | 25 ++++++ .../shuffled_with_duplicates_unordered.golden | 25 ++++++ .../two_added.golden | 58 ++++++++++++ .../two_added_and_two_removed.golden | 66 ++++++++++++++ ...d_two_removed_shuffled,_no_overlaps.golden | 74 +++++++++++++++ ..._two_removed_shuffled,_one_overlaps.golden | 70 +++++++++++++++ ...o_removed_shuffled,_with_duplicates.golden | 72 +++++++++++++++ .../two_removed.golden | 58 ++++++++++++ .../unchanged_empty.golden | 16 ++++ .../unchanged_non-empty.golden | 18 ++++ .../unchanged_null.golden | 11 +++ 201 files changed, 1820 insertions(+), 18 deletions(-) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/added.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/added_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/added_end_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/added_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/added_front_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/added_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/added_middle_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/changed_empty_to_null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/changed_non-null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/changed_non-null_to_null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/changed_null_to_empty.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/changed_null_to_non-null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/removed.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/removed_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/removed_end_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/removed_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/removed_front_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/removed_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/removed_middle_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/same_element_updated.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/same_element_updated_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/shuffled.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/shuffled_added_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/shuffled_added_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/shuffled_added_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/shuffled_removed_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/shuffled_removed_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/shuffled_removed_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/shuffled_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/shuffled_with_duplicates.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/shuffled_with_duplicates_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/two_added.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/two_added_and_two_removed.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/two_removed.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/unchanged_empty.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/unchanged_non-empty.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_force_new/unchanged_null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/added.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/added_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/added_end_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/added_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/added_front_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/added_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/added_middle_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/changed_empty_to_null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/changed_non-null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/changed_non-null_to_null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/changed_null_to_empty.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/changed_null_to_non-null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/removed.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/removed_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/removed_end_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/removed_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/removed_front_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/removed_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/removed_middle_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/same_element_updated.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/same_element_updated_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/shuffled.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/shuffled_added_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/shuffled_added_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/shuffled_added_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/shuffled_removed_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/shuffled_removed_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/shuffled_removed_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/shuffled_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/shuffled_with_duplicates.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/shuffled_with_duplicates_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/two_added.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/two_added_and_two_removed.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/two_removed.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/unchanged_empty.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/unchanged_non-empty.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/attribute_no_force_new/unchanged_null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/added.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/added_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/added_end_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/added_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/added_front_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/added_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/added_middle_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/changed_empty_to_null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/changed_non-null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/changed_non-null_to_null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/changed_null_to_empty.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/changed_null_to_non-null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/removed.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/removed_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/removed_end_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/removed_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/removed_front_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/removed_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/removed_middle_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/same_element_updated.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/same_element_updated_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/shuffled.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/shuffled_added_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/shuffled_added_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/shuffled_added_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/shuffled_removed_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/shuffled_removed_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/shuffled_removed_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/shuffled_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/shuffled_with_duplicates.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/shuffled_with_duplicates_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/two_added.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/two_added_and_two_removed.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/two_added_and_two_removed_shuffled,_no_overlaps.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/two_added_and_two_removed_shuffled,_one_overlaps.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/two_added_and_two_removed_shuffled,_with_duplicates.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/two_removed.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/unchanged_empty.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/unchanged_non-empty.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest/block_force_new => TestDetailedDiffSet/block_nested_force_new}/unchanged_null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/added.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/added_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/added_end_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/added_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/added_front_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/added_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/added_middle_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/changed_empty_to_null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/changed_non-null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/changed_non-null_to_null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/changed_null_to_empty.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/changed_null_to_non-null.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/removed.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/removed_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/removed_end_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/removed_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/removed_front_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/removed_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/removed_middle_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/same_element_updated.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/same_element_updated_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/shuffled.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/shuffled_added_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/shuffled_added_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/shuffled_added_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/shuffled_removed_end.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/shuffled_removed_front.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/shuffled_removed_middle.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/shuffled_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/shuffled_with_duplicates.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/shuffled_with_duplicates_unordered.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/two_added.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/two_added_and_two_removed.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/two_removed.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/unchanged_empty.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/unchanged_non-empty.golden (100%) rename pkg/tests/testdata/{TestDetailedDiffSetCrossTest => TestDetailedDiffSet}/block_no_force_new/unchanged_null.golden (100%) create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_end.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_end_unordered.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_front.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_front_unordered.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_middle.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_middle_unordered.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_empty_to_null.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_non-null.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_non-null_to_null.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_null_to_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_null_to_non-null.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_end.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_end_unordered.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_front.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_front_unordered.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_middle.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_middle_unordered.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/same_element_updated.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/same_element_updated_unordered.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_added_end.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_added_front.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_added_middle.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_removed_end.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_removed_front.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_removed_middle.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_unordered.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_with_duplicates.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_with_duplicates_unordered.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_removed.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/unchanged_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/unchanged_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/unchanged_null.golden diff --git a/pkg/tests/detailed_diff_set_test.go b/pkg/tests/detailed_diff_set_test.go index 53103a71c..874d78208 100644 --- a/pkg/tests/detailed_diff_set_test.go +++ b/pkg/tests/detailed_diff_set_test.go @@ -777,7 +777,7 @@ Plan: 0 to add, 1 to change, 0 to destroy. }) } -func TestDetailedDiffSetCrossTest(t *testing.T) { +func TestDetailedDiffSet(t *testing.T) { t.Parallel() attributeSchema := schema.Resource{ @@ -823,6 +823,24 @@ func TestDetailedDiffSetCrossTest(t *testing.T) { } blockSchemaForceNew := schema.Resource{ + Schema: map[string]*schema.Schema{ + "test": { + Type: schema.TypeSet, + Optional: true, + ForceNew: true, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "nested": { + Type: schema.TypeString, + Optional: true, + }, + }, + }, + }, + }, + } + + blockSchemaNestedForceNew := schema.Resource{ Schema: map[string]*schema.Schema{ "test": { Type: schema.TypeSet, @@ -872,15 +890,23 @@ func TestDetailedDiffSetCrossTest(t *testing.T) { return cty.ListVal(slice) } - schemaValueMakerPairs := []struct { + noForceNewSchemaValueMakerPairs := []struct { name string res schema.Resource valueMaker func(*[]string) cty.Value }{ {"attribute no force new", attributeSchema, attrList}, - {"attribute force new", attributeSchemaForceNew, attrList}, {"block no force new", blockSchema, nestedAttrList}, - {"block force new", blockSchemaForceNew, nestedAttrList}, + } + + forceNewSchemaValueMakerPairs := []struct { + name string + res schema.Resource + valueMaker func(*[]string) cty.Value + }{ + {"attribute force new", attributeSchemaForceNew, attrList}, + {"block top level force new", blockSchemaForceNew, nestedAttrList}, + {"block nested force new", blockSchemaNestedForceNew, nestedAttrList}, } scenarios := []struct { @@ -916,7 +942,6 @@ func TestDetailedDiffSetCrossTest(t *testing.T) { {"added end unordered", &[]string{"val2", "val3"}, &[]string{"val2", "val3", "val1"}}, {"same element updated", &[]string{"val1", "val2", "val3"}, &[]string{"val1", "val4", "val3"}}, - {"same element updated unordered", &[]string{"val2", "val3", "val1"}, &[]string{"val2", "val4", "val1"}}, {"shuffled", &[]string{"val1", "val2", "val3"}, &[]string{"val3", "val1", "val2"}}, {"shuffled unordered", &[]string{"val2", "val3", "val1"}, &[]string{"val3", "val1", "val2"}}, @@ -933,12 +958,24 @@ func TestDetailedDiffSetCrossTest(t *testing.T) { {"two added", &[]string{"val1", "val2"}, &[]string{"val1", "val2", "val3", "val4"}}, {"two removed", &[]string{"val1", "val2", "val3", "val4"}, &[]string{"val1", "val2"}}, + } + + // TODO[pulumi/pulumi-terraform-bridge#2726]: These tests fail to produce the correct replacement plan + // for the force new shcemas + extraScenarios := []struct { + name string + initialValue *[]string + changeValue *[]string + }{ + {"same element updated unordered", &[]string{"val2", "val3", "val1"}, &[]string{"val2", "val4", "val1"}}, {"two added and two removed", &[]string{"val1", "val2", "val3", "val4"}, &[]string{"val1", "val2", "val5", "val6"}}, {"two added and two removed shuffled, one overlaps", &[]string{"val1", "val2", "val3", "val4"}, &[]string{"val1", "val5", "val6", "val2"}}, {"two added and two removed shuffled, no overlaps", &[]string{"val1", "val2", "val3", "val4"}, &[]string{"val5", "val6", "val1", "val2"}}, {"two added and two removed shuffled, with duplicates", &[]string{"val1", "val2", "val3", "val4"}, &[]string{"val1", "val5", "val6", "val2", "val1", "val2"}}, } + allScenarios := append(scenarios, extraScenarios...) + type testOutput struct { initialValue *[]string changeValue *[]string @@ -947,24 +984,40 @@ func TestDetailedDiffSetCrossTest(t *testing.T) { detailedDiff map[string]any } - for _, schemaValueMakerPair := range schemaValueMakerPairs { + runTest := func(t *testing.T, schema schema.Resource, valueMaker func(*[]string) cty.Value, val1 *[]string, val2 *[]string) { + initialValue := valueMaker(val1) + changeValue := valueMaker(val2) + + diff := crosstests.Diff(t, &schema, map[string]cty.Value{"test": initialValue}, map[string]cty.Value{"test": changeValue}) + + autogold.ExpectFile(t, testOutput{ + initialValue: val1, + changeValue: val2, + tfOut: diff.TFOut, + pulumiOut: diff.PulumiOut, + detailedDiff: diff.PulumiDiff.DetailedDiff, + }) + } + + for _, schemaValueMakerPair := range forceNewSchemaValueMakerPairs { t.Run(schemaValueMakerPair.name, func(t *testing.T) { t.Parallel() for _, scenario := range scenarios { t.Run(scenario.name, func(t *testing.T) { t.Parallel() - initialValue := schemaValueMakerPair.valueMaker(scenario.initialValue) - changeValue := schemaValueMakerPair.valueMaker(scenario.changeValue) - - diff := crosstests.Diff(t, &schemaValueMakerPair.res, map[string]cty.Value{"test": initialValue}, map[string]cty.Value{"test": changeValue}) - - autogold.ExpectFile(t, testOutput{ - initialValue: scenario.initialValue, - changeValue: scenario.changeValue, - tfOut: diff.TFOut, - pulumiOut: diff.PulumiOut, - detailedDiff: diff.PulumiDiff.DetailedDiff, - }) + runTest(t, schemaValueMakerPair.res, schemaValueMakerPair.valueMaker, scenario.initialValue, scenario.changeValue) + }) + } + }) + } + + for _, schemaValueMakerPair := range noForceNewSchemaValueMakerPairs { + t.Run(schemaValueMakerPair.name, func(t *testing.T) { + t.Parallel() + for _, scenario := range allScenarios { + t.Run(scenario.name, func(t *testing.T) { + t.Parallel() + runTest(t, schemaValueMakerPair.res, schemaValueMakerPair.valueMaker, scenario.initialValue, scenario.changeValue) }) } }) diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added_end_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added_end_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added_end_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added_end_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added_front_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added_front_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added_front_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added_front_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added_middle_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added_middle_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/added_middle_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/added_middle_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/changed_empty_to_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/changed_empty_to_null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/changed_empty_to_null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/changed_empty_to_null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/changed_non-null.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/changed_non-null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/changed_non-null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/changed_non-null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/changed_non-null_to_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/changed_non-null_to_null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/changed_non-null_to_null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/changed_non-null_to_null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/changed_null_to_empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/changed_null_to_empty.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/changed_null_to_empty.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/changed_null_to_empty.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/changed_null_to_non-null.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/changed_null_to_non-null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/changed_null_to_non-null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/changed_null_to_non-null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed_end_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed_end_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed_end_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed_end_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed_front_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed_front_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed_front_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed_front_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed_middle_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed_middle_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/removed_middle_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/removed_middle_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/same_element_updated.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/same_element_updated.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/same_element_updated.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/same_element_updated.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/same_element_updated_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/same_element_updated_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/same_element_updated_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/same_element_updated_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_added_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_added_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_added_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_added_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_added_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_added_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_added_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_added_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_added_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_added_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_added_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_added_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_removed_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_removed_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_removed_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_removed_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_removed_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_removed_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_removed_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_removed_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_removed_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_removed_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_removed_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_removed_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_with_duplicates.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_with_duplicates.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_with_duplicates.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_with_duplicates.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_with_duplicates_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_with_duplicates_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/shuffled_with_duplicates_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/shuffled_with_duplicates_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/two_added.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/two_added.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/two_added.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/two_added.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/two_added_and_two_removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/two_added_and_two_removed.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/two_added_and_two_removed.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/two_added_and_two_removed.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/two_removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/two_removed.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/two_removed.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/two_removed.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/unchanged_empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/unchanged_empty.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/unchanged_empty.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/unchanged_empty.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/unchanged_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/unchanged_non-empty.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/unchanged_non-empty.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/unchanged_non-empty.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/unchanged_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/unchanged_null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_force_new/unchanged_null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_force_new/unchanged_null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added_end_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added_end_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added_end_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added_end_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added_front_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added_front_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added_front_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added_front_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added_middle_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added_middle_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/added_middle_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/added_middle_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/changed_empty_to_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/changed_empty_to_null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/changed_empty_to_null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/changed_empty_to_null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/changed_non-null.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/changed_non-null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/changed_non-null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/changed_non-null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/changed_non-null_to_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/changed_non-null_to_null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/changed_non-null_to_null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/changed_non-null_to_null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/changed_null_to_empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/changed_null_to_empty.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/changed_null_to_empty.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/changed_null_to_empty.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/changed_null_to_non-null.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/changed_null_to_non-null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/changed_null_to_non-null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/changed_null_to_non-null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed_end_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed_end_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed_end_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed_end_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed_front_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed_front_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed_front_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed_front_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed_middle_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed_middle_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/removed_middle_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/removed_middle_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/same_element_updated.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/same_element_updated.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/same_element_updated.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/same_element_updated.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/same_element_updated_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/same_element_updated_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/same_element_updated_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/same_element_updated_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_added_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_added_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_added_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_added_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_added_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_added_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_added_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_added_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_added_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_added_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_added_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_added_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_removed_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_removed_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_removed_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_removed_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_removed_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_removed_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_removed_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_removed_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_removed_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_removed_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_removed_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_removed_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_with_duplicates.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_with_duplicates.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_with_duplicates.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_with_duplicates.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_with_duplicates_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_with_duplicates_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/shuffled_with_duplicates_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/shuffled_with_duplicates_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/two_added.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/two_added.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/two_added.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/two_added.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/two_added_and_two_removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/two_added_and_two_removed.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/two_added_and_two_removed.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/two_added_and_two_removed.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/two_removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/two_removed.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/two_removed.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/two_removed.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/unchanged_empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/unchanged_empty.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/unchanged_empty.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/unchanged_empty.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/unchanged_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/unchanged_non-empty.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/unchanged_non-empty.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/unchanged_non-empty.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/unchanged_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/unchanged_null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/attribute_no_force_new/unchanged_null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/attribute_no_force_new/unchanged_null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added_end_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added_end_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added_end_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added_end_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added_front_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added_front_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added_front_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added_front_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added_middle_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added_middle_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/added_middle_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/added_middle_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/changed_empty_to_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/changed_empty_to_null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/changed_empty_to_null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/changed_empty_to_null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/changed_non-null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/changed_non-null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/changed_non-null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/changed_non-null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/changed_non-null_to_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/changed_non-null_to_null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/changed_non-null_to_null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/changed_non-null_to_null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/changed_null_to_empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/changed_null_to_empty.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/changed_null_to_empty.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/changed_null_to_empty.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/changed_null_to_non-null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/changed_null_to_non-null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/changed_null_to_non-null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/changed_null_to_non-null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed_end_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed_end_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed_end_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed_end_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed_front_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed_front_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed_front_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed_front_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed_middle_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed_middle_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/removed_middle_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/removed_middle_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/same_element_updated.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/same_element_updated.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/same_element_updated.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/same_element_updated.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/same_element_updated_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/same_element_updated_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/same_element_updated_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/same_element_updated_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_added_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_added_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_added_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_added_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_added_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_added_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_added_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_added_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_added_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_added_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_added_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_added_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_removed_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_removed_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_removed_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_removed_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_removed_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_removed_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_removed_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_removed_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_removed_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_removed_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_removed_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_removed_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_with_duplicates.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_with_duplicates.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_with_duplicates.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_with_duplicates.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_with_duplicates_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_with_duplicates_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/shuffled_with_duplicates_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/shuffled_with_duplicates_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/two_added.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/two_added.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/two_added.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/two_added.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/two_added_and_two_removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/two_added_and_two_removed.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/two_added_and_two_removed.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/two_added_and_two_removed.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/two_removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/two_removed.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/two_removed.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/two_removed.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/unchanged_empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/unchanged_empty.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/unchanged_empty.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/unchanged_empty.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/unchanged_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/unchanged_non-empty.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/unchanged_non-empty.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/unchanged_non-empty.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/unchanged_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/unchanged_null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_force_new/unchanged_null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_nested_force_new/unchanged_null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added_end_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added_end_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added_end_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added_end_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added_front_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added_front_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added_front_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added_front_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added_middle_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added_middle_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/added_middle_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/added_middle_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/changed_empty_to_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/changed_empty_to_null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/changed_empty_to_null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/changed_empty_to_null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/changed_non-null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/changed_non-null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/changed_non-null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/changed_non-null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/changed_non-null_to_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/changed_non-null_to_null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/changed_non-null_to_null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/changed_non-null_to_null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/changed_null_to_empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/changed_null_to_empty.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/changed_null_to_empty.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/changed_null_to_empty.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/changed_null_to_non-null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/changed_null_to_non-null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/changed_null_to_non-null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/changed_null_to_non-null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed_end_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed_end_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed_end_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed_end_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed_front_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed_front_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed_front_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed_front_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed_middle_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed_middle_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/removed_middle_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/removed_middle_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/same_element_updated.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/same_element_updated.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/same_element_updated.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/same_element_updated.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/same_element_updated_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/same_element_updated_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/same_element_updated_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/same_element_updated_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_added_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_added_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_added_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_added_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_added_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_added_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_added_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_added_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_added_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_added_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_added_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_added_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_removed_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_removed_end.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_removed_end.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_removed_end.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_removed_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_removed_front.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_removed_front.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_removed_front.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_removed_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_removed_middle.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_removed_middle.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_removed_middle.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_with_duplicates.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_with_duplicates.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_with_duplicates.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_with_duplicates.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_with_duplicates_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_with_duplicates_unordered.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/shuffled_with_duplicates_unordered.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/shuffled_with_duplicates_unordered.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/two_added.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/two_added.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/two_added.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/two_added.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/two_added_and_two_removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/two_added_and_two_removed.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/two_added_and_two_removed.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/two_added_and_two_removed.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/two_removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/two_removed.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/two_removed.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/two_removed.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/unchanged_empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/unchanged_empty.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/unchanged_empty.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/unchanged_empty.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/unchanged_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/unchanged_non-empty.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/unchanged_non-empty.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/unchanged_non-empty.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/unchanged_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/unchanged_null.golden similarity index 100% rename from pkg/tests/testdata/TestDetailedDiffSetCrossTest/block_no_force_new/unchanged_null.golden rename to pkg/tests/testdata/TestDetailedDiffSet/block_no_force_new/unchanged_null.golden diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added.golden new file mode 100644 index 000000000..10d82d99a --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added.golden @@ -0,0 +1,39 @@ +tests.testOutput{ + initialValue: &[]string{}, + changeValue: &[]string{"value"}, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + + test { # forces replacement + + nested = "value" + } + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + + tests: [ + + [0]: { + + nested : "value" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests": map[string]interface{}{"kind": "ADD_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_end.golden new file mode 100644 index 000000000..3e7a81565 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_end.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + }, + changeValue: &[]string{ + "val1", + "val2", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + + test { # forces replacement + + nested = "val3" + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [2]: { + + nested : "val3" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[2]": map[string]interface{}{"kind": "ADD_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_end_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_end_unordered.golden new file mode 100644 index 000000000..4f1b9a49b --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_end_unordered.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val2", + "val3", + }, + changeValue: &[]string{ + "val2", + "val3", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + + test { # forces replacement + + nested = "val1" + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [2]: { + + nested : "val1" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[2]": map[string]interface{}{"kind": "ADD_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_front.golden new file mode 100644 index 000000000..66286b3f2 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_front.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val2", + "val3", + }, + changeValue: &[]string{ + "val1", + "val2", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + + test { # forces replacement + + nested = "val1" + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [0]: { + + nested : "val1" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[0]": map[string]interface{}{"kind": "ADD_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_front_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_front_unordered.golden new file mode 100644 index 000000000..fc91a82a2 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_front_unordered.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val3", + "val1", + }, + changeValue: &[]string{ + "val2", + "val3", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + + test { # forces replacement + + nested = "val2" + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [0]: { + + nested : "val2" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[0]": map[string]interface{}{"kind": "ADD_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_middle.golden new file mode 100644 index 000000000..6500bba0b --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_middle.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val3", + }, + changeValue: &[]string{ + "val1", + "val2", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + + test { # forces replacement + + nested = "val2" + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [1]: { + + nested : "val2" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[1]": map[string]interface{}{"kind": "ADD_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_middle_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_middle_unordered.golden new file mode 100644 index 000000000..7f3d1108f --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/added_middle_unordered.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val2", + "val1", + }, + changeValue: &[]string{ + "val2", + "val3", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + + test { # forces replacement + + nested = "val3" + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [1]: { + + nested : "val3" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[1]": map[string]interface{}{"kind": "ADD_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_empty_to_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_empty_to_null.golden new file mode 100644 index 000000000..7bae19762 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_empty_to_null.golden @@ -0,0 +1,15 @@ +tests.testOutput{ + initialValue: &[]string{}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_non-null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_non-null.golden new file mode 100644 index 000000000..0d933f20b --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_non-null.golden @@ -0,0 +1,44 @@ +tests.testOutput{ + initialValue: &[]string{ + "value", + }, + changeValue: &[]string{"value1"}, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + - test { + - nested = "value" -> null + } + + test { + + nested = "value1" + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + ~ [0]: { + ~ nested: "value" => "value1" + } + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[0].nested": map[string]interface{}{"kind": "UPDATE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_non-null_to_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_non-null_to_null.golden new file mode 100644 index 000000000..8ea0f409e --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_non-null_to_null.golden @@ -0,0 +1,40 @@ +tests.testOutput{ + initialValue: &[]string{ + "value", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + - test { # forces replacement + - nested = "value" -> null + } + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + - tests: [ + - [0]: { + - nested: "value" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests": map[string]interface{}{"kind": "DELETE_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_null_to_empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_null_to_empty.golden new file mode 100644 index 000000000..9589fbdf0 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_null_to_empty.golden @@ -0,0 +1,15 @@ +tests.testOutput{ + changeValue: &[]string{}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_null_to_non-null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_null_to_non-null.golden new file mode 100644 index 000000000..c6125570d --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/changed_null_to_non-null.golden @@ -0,0 +1,40 @@ +tests.testOutput{ + changeValue: &[]string{ + "value", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + + test { # forces replacement + + nested = "value" + } + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + + tests: [ + + [0]: { + + nested : "value" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests": map[string]interface{}{"kind": "ADD_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed.golden new file mode 100644 index 000000000..bce9459bf --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed.golden @@ -0,0 +1,41 @@ +tests.testOutput{ + initialValue: &[]string{ + "value", + }, + changeValue: &[]string{}, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + - test { # forces replacement + - nested = "value" -> null + } + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + - tests: [ + - [0]: { + - nested: "value" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests": map[string]interface{}{"kind": "DELETE_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_end.golden new file mode 100644 index 000000000..b9b27b7bb --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_end.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val1", + "val2", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + - test { # forces replacement + - nested = "val3" -> null + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + - [2]: { + - nested: "val3" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[2]": map[string]interface{}{"kind": "DELETE_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_end_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_end_unordered.golden new file mode 100644 index 000000000..3d59d39ff --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_end_unordered.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val2", + "val3", + "val1", + }, + changeValue: &[]string{ + "val2", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + - test { # forces replacement + - nested = "val1" -> null + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + - [0]: { + - nested: "val1" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[0]": map[string]interface{}{"kind": "DELETE_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_front.golden new file mode 100644 index 000000000..b166de593 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_front.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val2", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + - test { # forces replacement + - nested = "val1" -> null + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + - [0]: { + - nested: "val1" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[0]": map[string]interface{}{"kind": "DELETE_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_front_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_front_unordered.golden new file mode 100644 index 000000000..b32adae12 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_front_unordered.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val2", + "val3", + "val1", + }, + changeValue: &[]string{ + "val3", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + - test { # forces replacement + - nested = "val2" -> null + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + - [1]: { + - nested: "val2" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[1]": map[string]interface{}{"kind": "DELETE_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_middle.golden new file mode 100644 index 000000000..43d40629c --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_middle.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val1", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + - test { # forces replacement + - nested = "val2" -> null + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + - [1]: { + - nested: "val2" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[1]": map[string]interface{}{"kind": "DELETE_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_middle_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_middle_unordered.golden new file mode 100644 index 000000000..53f28a103 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/removed_middle_unordered.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val3", + "val1", + "val2", + }, + changeValue: &[]string{ + "val3", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + - test { # forces replacement + - nested = "val2" -> null + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + - [1]: { + - nested: "val2" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[1]": map[string]interface{}{"kind": "DELETE_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/same_element_updated.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/same_element_updated.golden new file mode 100644 index 000000000..6e47be3c9 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/same_element_updated.golden @@ -0,0 +1,52 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val1", + "val4", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + - test { + - nested = "val2" -> null + } + + test { + + nested = "val4" + } + + # (2 unchanged blocks hidden) + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + ~ [1]: { + ~ nested: "val2" => "val4" + } + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[1].nested": map[string]interface{}{"kind": "UPDATE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/same_element_updated_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/same_element_updated_unordered.golden new file mode 100644 index 000000000..24a6cb727 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/same_element_updated_unordered.golden @@ -0,0 +1,58 @@ +tests.testOutput{ + initialValue: &[]string{ + "val2", + "val3", + "val1", + }, + changeValue: &[]string{ + "val2", + "val4", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + - test { + - nested = "val3" -> null + } + + test { + + nested = "val4" + } + + # (2 unchanged blocks hidden) + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [1]: { + + nested : "val4" + } + - [2]: { + - nested: "val3" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "tests[1]": map[string]interface{}{"kind": "ADD_REPLACE"}, + "tests[2]": map[string]interface{}{"kind": "DELETE_REPLACE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled.golden new file mode 100644 index 000000000..b22444299 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled.golden @@ -0,0 +1,24 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val3", + "val1", + "val2", + }, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_added_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_added_end.golden new file mode 100644 index 000000000..37ea56333 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_added_end.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + }, + changeValue: &[]string{ + "val2", + "val1", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + + test { # forces replacement + + nested = "val3" + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [2]: { + + nested : "val3" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[2]": map[string]interface{}{"kind": "ADD_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_added_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_added_front.golden new file mode 100644 index 000000000..0f837543f --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_added_front.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val2", + "val3", + }, + changeValue: &[]string{ + "val1", + "val3", + "val2", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + + test { # forces replacement + + nested = "val1" + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [0]: { + + nested : "val1" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[0]": map[string]interface{}{"kind": "ADD_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_added_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_added_middle.golden new file mode 100644 index 000000000..1eafb16ce --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_added_middle.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val3", + }, + changeValue: &[]string{ + "val3", + "val2", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + + test { # forces replacement + + nested = "val2" + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [1]: { + + nested : "val2" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[1]": map[string]interface{}{"kind": "ADD_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_removed_end.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_removed_end.golden new file mode 100644 index 000000000..b7c1fcc2e --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_removed_end.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val2", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + - test { # forces replacement + - nested = "val3" -> null + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + - [2]: { + - nested: "val3" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[2]": map[string]interface{}{"kind": "DELETE_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_removed_front.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_removed_front.golden new file mode 100644 index 000000000..7f391ace8 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_removed_front.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val3", + "val2", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + - test { # forces replacement + - nested = "val1" -> null + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + - [0]: { + - nested: "val1" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[0]": map[string]interface{}{"kind": "DELETE_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_removed_middle.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_removed_middle.golden new file mode 100644 index 000000000..a172a3cf2 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_removed_middle.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val3", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + - test { # forces replacement + - nested = "val2" -> null + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + - [1]: { + - nested: "val2" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{"tests[1]": map[string]interface{}{"kind": "DELETE_REPLACE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_unordered.golden new file mode 100644 index 000000000..d98c38179 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_unordered.golden @@ -0,0 +1,24 @@ +tests.testOutput{ + initialValue: &[]string{ + "val2", + "val3", + "val1", + }, + changeValue: &[]string{ + "val3", + "val1", + "val2", + }, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_with_duplicates.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_with_duplicates.golden new file mode 100644 index 000000000..09d74fe9c --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_with_duplicates.golden @@ -0,0 +1,25 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val3", + "val1", + "val2", + "val3", + }, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_with_duplicates_unordered.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_with_duplicates_unordered.golden new file mode 100644 index 000000000..1011e92c5 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/shuffled_with_duplicates_unordered.golden @@ -0,0 +1,25 @@ +tests.testOutput{ + initialValue: &[]string{ + "val2", + "val3", + "val1", + }, + changeValue: &[]string{ + "val3", + "val1", + "val2", + "val3", + }, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added.golden new file mode 100644 index 000000000..245630f95 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added.golden @@ -0,0 +1,58 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + }, + changeValue: &[]string{ + "val1", + "val2", + "val3", + "val4", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + + test { # forces replacement + + nested = "val3" + } + + test { # forces replacement + + nested = "val4" + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [2]: { + + nested : "val3" + } + + [3]: { + + nested : "val4" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "tests[2]": map[string]interface{}{"kind": "ADD_REPLACE"}, + "tests[3]": map[string]interface{}{"kind": "ADD_REPLACE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed.golden new file mode 100644 index 000000000..7fb5ca1ca --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed.golden @@ -0,0 +1,66 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + "val4", + }, + changeValue: &[]string{ + "val1", + "val2", + "val5", + "val6", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + - test { + - nested = "val3" -> null + } + - test { + - nested = "val4" -> null + } + + test { + + nested = "val5" + } + + test { + + nested = "val6" + } + + # (2 unchanged blocks hidden) + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + ~ [2]: { + ~ nested: "val3" => "val5" + } + ~ [3]: { + ~ nested: "val4" => "val6" + } + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "tests[2].nested": map[string]interface{}{"kind": "UPDATE"}, + "tests[3].nested": map[string]interface{}{"kind": "UPDATE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden new file mode 100644 index 000000000..f14f7acf2 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed_shuffled,_no_overlaps.golden @@ -0,0 +1,74 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + "val4", + }, + changeValue: &[]string{ + "val5", + "val6", + "val1", + "val2", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + - test { + - nested = "val3" -> null + } + - test { + - nested = "val4" -> null + } + + test { + + nested = "val5" + } + + test { + + nested = "val6" + } + + # (2 unchanged blocks hidden) + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [0]: { + + nested : "val5" + } + + [1]: { + + nested : "val6" + } + - [2]: { + - nested: "val3" + } + - [3]: { + - nested: "val4" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "tests[0]": map[string]interface{}{"kind": "ADD_REPLACE"}, + "tests[1]": map[string]interface{}{"kind": "ADD_REPLACE"}, + "tests[2]": map[string]interface{}{"kind": "DELETE_REPLACE"}, + "tests[3]": map[string]interface{}{"kind": "DELETE_REPLACE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden new file mode 100644 index 000000000..01e7fd027 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed_shuffled,_one_overlaps.golden @@ -0,0 +1,70 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + "val4", + }, + changeValue: &[]string{ + "val1", + "val5", + "val6", + "val2", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + - test { + - nested = "val3" -> null + } + - test { + - nested = "val4" -> null + } + + test { + + nested = "val5" + } + + test { + + nested = "val6" + } + + # (2 unchanged blocks hidden) + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [1]: { + + nested : "val5" + } + ~ [2]: { + ~ nested: "val3" => "val6" + } + - [3]: { + - nested: "val4" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "tests[1]": map[string]interface{}{"kind": "ADD_REPLACE"}, + "tests[2].nested": map[string]interface{}{"kind": "UPDATE"}, + "tests[3]": map[string]interface{}{"kind": "DELETE_REPLACE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden new file mode 100644 index 000000000..214c2f28b --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_added_and_two_removed_shuffled,_with_duplicates.golden @@ -0,0 +1,72 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + "val4", + }, + changeValue: &[]string{ + "val1", + "val5", + "val6", + "val2", + "val1", + "val2", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + - test { + - nested = "val3" -> null + } + - test { + - nested = "val4" -> null + } + + test { + + nested = "val5" + } + + test { + + nested = "val6" + } + + # (2 unchanged blocks hidden) + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + + [1]: { + + nested : "val5" + } + ~ [2]: { + ~ nested: "val3" => "val6" + } + - [3]: { + - nested: "val4" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "tests[1]": map[string]interface{}{"kind": "ADD_REPLACE"}, + "tests[2].nested": map[string]interface{}{"kind": "UPDATE"}, + "tests[3]": map[string]interface{}{"kind": "DELETE_REPLACE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_removed.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_removed.golden new file mode 100644 index 000000000..751e8ad77 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/two_removed.golden @@ -0,0 +1,58 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + "val4", + }, + changeValue: &[]string{ + "val1", + "val2", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: ++/- create replacement and then destroy + +Terraform will perform the following actions: + + # crossprovider_test_res.example must be replaced ++/- resource "crossprovider_test_res" "example" { + ~ id = "newid" -> (known after apply) + + - test { # forces replacement + - nested = "val3" -> null + } + - test { # forces replacement + - nested = "val4" -> null + } + + # (2 unchanged blocks hidden) + } + +Plan: 1 to add, 0 to change, 1 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + +-crossprovider:index/testRes:TestRes: (replace) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ tests: [ + - [2]: { + - nested: "val3" + } + - [3]: { + - nested: "val4" + } + ] +Resources: + +-1 to replace + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "tests[2]": map[string]interface{}{"kind": "DELETE_REPLACE"}, + "tests[3]": map[string]interface{}{"kind": "DELETE_REPLACE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/unchanged_empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/unchanged_empty.golden new file mode 100644 index 000000000..7d07227be --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/unchanged_empty.golden @@ -0,0 +1,16 @@ +tests.testOutput{ + initialValue: &[]string{}, + changeValue: &[]string{}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/unchanged_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/unchanged_non-empty.golden new file mode 100644 index 000000000..a87f03643 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/unchanged_non-empty.golden @@ -0,0 +1,18 @@ +tests.testOutput{ + initialValue: &[]string{ + "value", + }, + changeValue: &[]string{"value"}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/unchanged_null.golden b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/unchanged_null.golden new file mode 100644 index 000000000..a53fac34d --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffSet/block_top_level_force_new/unchanged_null.golden @@ -0,0 +1,11 @@ +tests.testOutput{tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`} From 047f4050813edd1578423d6ca6087a9bc9c42e69 Mon Sep 17 00:00:00 2001 From: Venelin Date: Wed, 11 Dec 2024 17:42:17 +0000 Subject: [PATCH 4/4] Refactor SDKv2 non-set detailed diff tests --- pkg/tests/detailed_diff_list_test.go | 221 ++ pkg/tests/detailed_diff_test.go | 2187 +---------------- pkg/tests/detailed_diff_unknown_test.go | 1058 ++++++++ .../list_attribute/added_empty.golden | 15 + .../list_attribute/added_non-empty.golden | 37 + .../list_attribute/changed.golden | 38 + .../list_element_added_back.golden | 45 + .../list_element_added_front.golden | 51 + .../list_element_added_middle.golden | 49 + .../list_element_removed_end.golden | 52 + .../list_element_removed_front.golden | 50 + .../list_element_removed_middle.golden | 52 + .../list_attribute/removed_empty.golden | 15 + .../list_attribute/removed_non-empty.golden | 37 + .../list_attribute/unchanged_empty.golden | 11 + .../list_attribute/unchanged_non-empty.golden | 18 + .../list_block/added_empty.golden | 15 + .../list_block/added_non-empty.golden | 40 + .../list_block/changed.golden | 41 + .../list_block/list_element_added_back.golden | 48 + .../list_element_added_front.golden | 62 + .../list_element_added_middle.golden | 57 + .../list_element_removed_end.golden | 62 + .../list_element_removed_front.golden | 57 + .../list_element_removed_middle.golden | 62 + .../list_block/removed_empty.golden | 15 + .../list_block/removed_non-empty.golden | 40 + .../list_block/unchanged_empty.golden | 11 + .../list_block/unchanged_non-empty.golden | 18 + .../added_empty.golden | 15 + .../added_non-empty.golden | 35 + .../max_items_one_attribute/changed.golden | 36 + .../removed_empty.golden | 15 + .../removed_non-empty.golden | 35 + .../unchanged_empty.golden | 11 + .../unchanged_non-empty.golden | 18 + .../max_items_one_block/added_empty.golden | 15 + .../added_non-empty.golden | 38 + .../max_items_one_block/changed.golden | 39 + .../max_items_one_block/removed_empty.golden | 15 + .../removed_non-empty.golden | 38 + .../unchanged_empty.golden | 11 + .../unchanged_non-empty.golden | 18 + .../testdata/TestDetailedDiffMap/added.golden | 36 + .../TestDetailedDiffMap/key_changed.golden | 43 + .../TestDetailedDiffMap/removed.golden | 38 + .../unchanged_empty.golden | 16 + .../unchanged_non-empty.golden | 18 + .../TestDetailedDiffMap/value_changed.golden | 38 + .../TestDetailedDiffString/added.golden | 30 + .../TestDetailedDiffString/changed.golden | 31 + .../TestDetailedDiffString/removed.golden | 30 + .../unchanged_empty.golden | 11 + .../unchanged_non-empty.golden | 15 + 54 files changed, 3019 insertions(+), 2090 deletions(-) create mode 100644 pkg/tests/detailed_diff_list_test.go create mode 100644 pkg/tests/detailed_diff_unknown_test.go create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/added_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/added_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/changed.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_added_back.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_added_front.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_added_middle.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_removed_end.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_removed_front.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_removed_middle.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/removed_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/removed_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/unchanged_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_attribute/unchanged_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/added_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/added_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/changed.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_added_back.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_added_front.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_added_middle.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_removed_end.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_removed_front.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_removed_middle.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/removed_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/removed_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/unchanged_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/list_block/unchanged_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/added_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/added_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/changed.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/removed_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/removed_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/unchanged_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/unchanged_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/added_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/added_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/changed.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/removed_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/removed_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/unchanged_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/unchanged_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffMap/added.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffMap/key_changed.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffMap/removed.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffMap/unchanged_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffMap/unchanged_non-empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffMap/value_changed.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffString/added.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffString/changed.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffString/removed.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffString/unchanged_empty.golden create mode 100644 pkg/tests/testdata/TestDetailedDiffString/unchanged_non-empty.golden diff --git a/pkg/tests/detailed_diff_list_test.go b/pkg/tests/detailed_diff_list_test.go new file mode 100644 index 000000000..493cc6258 --- /dev/null +++ b/pkg/tests/detailed_diff_list_test.go @@ -0,0 +1,221 @@ +package tests + +import ( + "testing" + + "github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" + "github.com/hexops/autogold/v2" + "github.com/zclconf/go-cty/cty" + + crosstests "github.com/pulumi/pulumi-terraform-bridge/v3/pkg/internal/tests/cross-tests" +) + +func TestDetailedDiffList(t *testing.T) { + t.Parallel() + + listAttrSchema := schema.Resource{ + Schema: map[string]*schema.Schema{ + "list_attr": { + Type: schema.TypeList, + Optional: true, + Elem: &schema.Schema{Type: schema.TypeString}, + }, + }, + } + + maxItemsOneAttrSchema := schema.Resource{ + Schema: map[string]*schema.Schema{ + "list_attr": { + Type: schema.TypeList, + Optional: true, + MaxItems: 1, + Elem: &schema.Schema{Type: schema.TypeString}, + }, + }, + } + + listBlockSchema := schema.Resource{ + Schema: map[string]*schema.Schema{ + "list_block": { + Type: schema.TypeList, + Optional: true, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "prop": { + Type: schema.TypeString, + Optional: true, + }, + }, + }, + }, + }, + } + + maxItemsOneBlockSchema := schema.Resource{ + Schema: map[string]*schema.Schema{ + "list_block": { + Type: schema.TypeList, + Optional: true, + MaxItems: 1, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "nested_prop": { + Type: schema.TypeString, + Optional: true, + }, + }, + }, + }, + }, + } + + attrList := func(arr *[]string) map[string]cty.Value { + if arr == nil { + return map[string]cty.Value{} + } + + if len(*arr) == 0 { + return map[string]cty.Value{ + "list_attr": cty.ListValEmpty(cty.String), + } + } + + slice := make([]cty.Value, len(*arr)) + for i, v := range *arr { + slice[i] = cty.StringVal(v) + } + return map[string]cty.Value{ + "list_attr": cty.ListVal(slice), + } + } + + blockList := func(arr *[]string) map[string]cty.Value { + if arr == nil { + return map[string]cty.Value{} + } + + if len(*arr) == 0 { + return map[string]cty.Value{ + "list_block": cty.ListValEmpty(cty.DynamicPseudoType), + } + } + + slice := make([]cty.Value, len(*arr)) + for i, v := range *arr { + slice[i] = cty.ObjectVal(map[string]cty.Value{"prop": cty.StringVal(v)}) + } + return map[string]cty.Value{ + "list_block": cty.ListVal(slice), + } + } + + nestedBlockList := func(arr *[]string) map[string]cty.Value { + if arr == nil { + return map[string]cty.Value{} + } + + if len(*arr) == 0 { + return map[string]cty.Value{ + "list_block": cty.ListValEmpty(cty.DynamicPseudoType), + } + } + + slice := make([]cty.Value, len(*arr)) + for i, v := range *arr { + slice[i] = cty.ObjectVal(map[string]cty.Value{"nested_prop": cty.StringVal(v)}) + } + return map[string]cty.Value{ + "list_block": cty.ListVal(slice), + } + } + + listPairs := []struct { + name string + schema schema.Resource + valueMaker func(*[]string) map[string]cty.Value + }{ + {"list attribute", listAttrSchema, attrList}, + {"list block", listBlockSchema, blockList}, + } + + maxItemsOnePairs := []struct { + name string + schema schema.Resource + valueMaker func(*[]string) map[string]cty.Value + }{ + {"max items one attribute", maxItemsOneAttrSchema, attrList}, + {"max items one block", maxItemsOneBlockSchema, nestedBlockList}, + } + + oneElementScenarios := []struct { + name string + initialValue *[]string + changeValue *[]string + }{ + {"unchanged empty", nil, nil}, + {"unchanged non-empty", ref([]string{"val1"}), ref([]string{"val1"})}, + {"added non-empty", nil, ref([]string{"val1"})}, + {"added empty", nil, ref([]string{})}, + {"removed non-empty", ref([]string{"val1"}), nil}, + {"removed empty", ref([]string{}), nil}, + {"changed", ref([]string{"val1"}), ref([]string{"val2"})}, + } + + multiElementScenarios := []struct { + name string + initialValue *[]string + changeValue *[]string + }{ + {"list element added front", ref([]string{"val2", "val3"}), ref([]string{"val1", "val2", "val3"})}, + {"list element added back", ref([]string{"val1", "val2"}), ref([]string{"val1", "val2", "val3"})}, + {"list element added middle", ref([]string{"val1", "val3"}), ref([]string{"val1", "val2", "val3"})}, + {"list element removed front", ref([]string{"val1", "val2", "val3"}), ref([]string{"val3", "val2"})}, + {"list element removed middle", ref([]string{"val1", "val2", "val3"}), ref([]string{"val3", "val1"})}, + {"list element removed end", ref([]string{"val1", "val2", "val3"}), ref([]string{"val2", "val1"})}, + } + + scenarios := append(oneElementScenarios, multiElementScenarios...) + + type testOutput struct { + initialValue *[]string + changeValue *[]string + tfOut string + pulumiOut string + detailedDiff map[string]any + } + + runTest := func(t *testing.T, schema schema.Resource, valueMaker func(*[]string) map[string]cty.Value, initialValue *[]string, changeValue *[]string) { + diff := crosstests.Diff(t, &schema, valueMaker(initialValue), valueMaker(changeValue)) + autogold.ExpectFile(t, testOutput{ + initialValue: initialValue, + changeValue: changeValue, + tfOut: diff.TFOut, + pulumiOut: diff.PulumiOut, + detailedDiff: diff.PulumiDiff.DetailedDiff, + }) + } + + for _, schemaValueMakerPair := range listPairs { + t.Run(schemaValueMakerPair.name, func(t *testing.T) { + t.Parallel() + for _, scenario := range scenarios { + t.Run(scenario.name, func(t *testing.T) { + t.Parallel() + runTest(t, schemaValueMakerPair.schema, schemaValueMakerPair.valueMaker, scenario.initialValue, scenario.changeValue) + }) + } + }) + } + + for _, schemaValueMakerPair := range maxItemsOnePairs { + t.Run(schemaValueMakerPair.name, func(t *testing.T) { + t.Parallel() + for _, scenario := range oneElementScenarios { + t.Run(scenario.name, func(t *testing.T) { + t.Parallel() + runTest(t, schemaValueMakerPair.schema, schemaValueMakerPair.valueMaker, scenario.initialValue, scenario.changeValue) + }) + } + }) + } +} diff --git a/pkg/tests/detailed_diff_test.go b/pkg/tests/detailed_diff_test.go index 686a4006b..df5a500f4 100644 --- a/pkg/tests/detailed_diff_test.go +++ b/pkg/tests/detailed_diff_test.go @@ -1,2134 +1,141 @@ package tests import ( - "context" - "encoding/json" - "fmt" - "os" - "path/filepath" - "strings" "testing" - "github.com/hashicorp/terraform-plugin-sdk/v2/diag" "github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" "github.com/hexops/autogold/v2" - "github.com/pulumi/pulumi/sdk/v3/go/auto/optpreview" - "github.com/stretchr/testify/require" + "github.com/zclconf/go-cty/cty" - "github.com/pulumi/pulumi-terraform-bridge/v3/pkg/internal/tests/pulcheck" - "github.com/pulumi/pulumi-terraform-bridge/v3/pkg/tfbridge/info" + crosstests "github.com/pulumi/pulumi-terraform-bridge/v3/pkg/internal/tests/cross-tests" ) -func TestUnknownHandling(t *testing.T) { - t.Parallel() - resMap := map[string]*schema.Resource{ - "prov_test": { - Schema: map[string]*schema.Schema{ - "test": { - Type: schema.TypeString, - Optional: true, - }, - }, - }, - "prov_aux": { - Schema: map[string]*schema.Schema{ - "aux": { - Type: schema.TypeString, - Computed: true, - Optional: true, - }, - }, - CreateContext: func(_ context.Context, d *schema.ResourceData, _ interface{}) diag.Diagnostics { - d.SetId("aux") - err := d.Set("aux", "aux") - require.NoError(t, err) - return nil - }, - }, - } - tfp := &schema.Provider{ResourcesMap: resMap} - bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp) - program := ` -name: test -runtime: yaml -resources: - auxRes: - type: prov:index:Aux - mainRes: - type: prov:index:Test - properties: - test: ${auxRes.aux} -outputs: - testOut: ${mainRes.test} -` - pt := pulcheck.PulCheck(t, bridgedProvider, program) - res := pt.Preview(t, optpreview.Diff()) - // Test that the test property is unknown at preview time - require.Contains(t, res.StdOut, "test : output") - resUp := pt.Up(t) - // assert that the property gets resolved - require.Equal(t, "aux", resUp.Outputs["testOut"].Value) -} - -func trimDiff(t *testing.T, diff string) string { - urnLine := " [urn=urn:pulumi:test::test::pulumi:pulumi:Stack::test-test]" - resourcesSummaryLine := "Resources:\n" - require.Contains(t, diff, urnLine) - require.Contains(t, diff, resourcesSummaryLine) - - // trim the diff to only include the contents after the URN line and before the summary - urnIndex := strings.Index(diff, urnLine) - resourcesSummaryIndex := strings.Index(diff, resourcesSummaryLine) - return diff[urnIndex+len(urnLine) : resourcesSummaryIndex] +func ref[T any](v T) *T { + return &v } -func TestUnknownBlocks(t *testing.T) { +func TestDetailedDiffString(t *testing.T) { t.Parallel() - resMap := map[string]*schema.Resource{ - "prov_test": { - Schema: map[string]*schema.Schema{ - "test": { - Type: schema.TypeList, - Optional: true, - Elem: &schema.Resource{ - Schema: map[string]*schema.Schema{ - "test_prop": { - Type: schema.TypeString, - Optional: true, - }, - }, - }, - }, - }, - }, - "prov_nested_test": { - Schema: map[string]*schema.Schema{ - "test": { - Type: schema.TypeList, - Optional: true, - Elem: &schema.Resource{ - Schema: map[string]*schema.Schema{ - "nested_prop": { - Type: schema.TypeList, - Optional: true, - Elem: &schema.Resource{ - Schema: map[string]*schema.Schema{ - "test_prop": { - Type: schema.TypeList, - Optional: true, - Elem: &schema.Schema{ - Type: schema.TypeString, - }, - }, - }, - }, - }, - }, - }, - }, - }, - }, - "prov_aux": { - Schema: map[string]*schema.Schema{ - "aux": { - Type: schema.TypeList, - Computed: true, - Optional: true, - Elem: &schema.Resource{ - Schema: map[string]*schema.Schema{ - "test_prop": { - Type: schema.TypeString, - Computed: true, - Optional: true, - }, - }, - }, - }, - "nested_aux": { - Type: schema.TypeList, - Optional: true, - Computed: true, - Elem: &schema.Resource{ - Schema: map[string]*schema.Schema{ - "nested_prop": { - Type: schema.TypeList, - Optional: true, - Computed: true, - Elem: &schema.Resource{ - Schema: map[string]*schema.Schema{ - "test_prop": { - Type: schema.TypeList, - Optional: true, - Computed: true, - Elem: &schema.Schema{ - Type: schema.TypeString, - }, - }, - }, - }, - }, - }, - }, - }, - }, - CreateContext: func(_ context.Context, d *schema.ResourceData, _ interface{}) diag.Diagnostics { - d.SetId("aux") - if d.Get("aux") == nil { - err := d.Set("aux", []map[string]interface{}{{"test_prop": "aux"}}) - require.NoError(t, err) - } - if d.Get("nested_aux") == nil { - err := d.Set("nested_aux", []map[string]interface{}{ - { - "nested_prop": []map[string]interface{}{ - {"test_prop": []string{"aux"}}, - }, - }, - }) - require.NoError(t, err) - } - return nil + + res := schema.Resource{ + Schema: map[string]*schema.Schema{ + "string_prop": { + Type: schema.TypeString, + Optional: true, }, }, } - tfp := &schema.Provider{ResourcesMap: resMap} - bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp) - provTestKnownProgram := ` -name: test -runtime: yaml -resources: - mainRes: - type: prov:index:Test - properties: - tests: - - testProp: "known_val" -` - nestedProvTestKnownProgram := ` -name: test -runtime: yaml -resources: - mainRes: - type: prov:index:NestedTest - properties: - tests: - - nestedProps: - - testProps: - - "known_val" -` + valueOne := ref("val1") + valueTwo := ref("val2") + var noValue *string - for _, tc := range []struct { - name string - program string - initialKnownProgram string - expectedInitial autogold.Value - expectedUpdate autogold.Value + ctyVal := func(v *string) map[string]cty.Value { + if v == nil { + return map[string]cty.Value{} + } + return map[string]cty.Value{ + "string_prop": cty.StringVal(*v), + } + } + + scenarios := []struct { + name string + initialValue *string + changeValue *string }{ - { - "list of objects", - ` -name: test -runtime: yaml -resources: - auxRes: - type: prov:index:Aux - properties: - auxes: %s - nestedAuxes: %s - mainRes: - type: prov:index:Test - properties: - tests: ${auxRes.auxes} -`, - provTestKnownProgram, - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - + prov:index/test:Test: (create) - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - tests : output -`), - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - tests: [ - - [0]: { - - testProp: "known_val" - } - ] - + tests: output -`), - }, - { - "unknown object", - ` -name: test -runtime: yaml -resources: - auxRes: - type: prov:index:Aux - properties: - auxes: %s - nestedAuxes: %s - mainRes: - type: prov:index:Test - properties: - tests: - - ${auxRes.auxes[0]} -`, - provTestKnownProgram, + {"unchanged empty", noValue, noValue}, + {"unchanged non-empty", valueOne, valueOne}, + {"added", noValue, valueOne}, + {"removed", valueOne, noValue}, + {"changed", valueOne, valueTwo}, + } - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - + prov:index/test:Test: (create) - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - tests : [ - [0]: output - ] -`), - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ tests: [ - - [0]: { - - testProp: "known_val" - } - + [0]: output - ] -`), - }, - { - "unknown object with others", - ` -name: test -runtime: yaml -resources: - auxRes: - type: prov:index:Aux - properties: - auxes: %s - nestedAuxes: %s - mainRes: - type: prov:index:Test - properties: - tests: - - ${auxRes.auxes[0]} - - {"testProp": "val"} -`, - provTestKnownProgram, + type testOutput struct { + initialValue *string + changeValue *string + tfOut string + pulumiOut string + detailedDiff map[string]any + } - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - + prov:index/test:Test: (create) - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - tests : [ - [0]: output - [1]: { - testProp : "val" - } - ] -`), - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ tests: [ - - [0]: { - - testProp: "known_val" - } - + [0]: output - + [1]: { - + testProp : "val" - } - ] -`), - }, - { - "unknown nested", - ` -name: test -runtime: yaml -resources: - auxRes: - type: prov:index:Aux - properties: - auxes: %s - nestedAuxes: %s - mainRes: - type: prov:index:NestedTest - properties: - tests: ${auxRes.nestedAuxes} -`, - nestedProvTestKnownProgram, - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - + prov:index/nestedTest:NestedTest: (create) - [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] - tests : output -`), - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - ~ prov:index/nestedTest:NestedTest: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] - - tests: [ - - [0]: { - - nestedProps: [ - - [0]: { - - testProps: [ - - [0]: "known_val" - ] - } - ] - } - ] - + tests: output -`), - }, - { - "unknown nested level 1", - ` -name: test -runtime: yaml -resources: - auxRes: - type: prov:index:Aux - properties: - auxes: %s - nestedAuxes: %s - mainRes: - type: prov:index:NestedTest - properties: - tests: - - ${auxRes.nestedAuxes[0]} -`, - nestedProvTestKnownProgram, - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - + prov:index/nestedTest:NestedTest: (create) - [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] - tests : [ - [0]: output - ] -`), - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - ~ prov:index/nestedTest:NestedTest: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] - ~ tests: [ - - [0]: { - - nestedProps: [ - - [0]: { - - testProps: [ - - [0]: "known_val" - ] - } - ] - } - + [0]: output - ] -`), - }, - { - "unknown nested level 2", - ` -name: test -runtime: yaml -resources: - auxRes: - type: prov:index:Aux - properties: - auxes: %s - nestedAuxes: %s - mainRes: - type: prov:index:NestedTest - properties: - tests: - - nestedProps: ${auxRes.nestedAuxes[0].nestedProps} -`, - nestedProvTestKnownProgram, - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - + prov:index/nestedTest:NestedTest: (create) - [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] - tests : [ - [0]: { - nestedProps: output - } - ] -`), - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - ~ prov:index/nestedTest:NestedTest: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] - ~ tests: [ - ~ [0]: { - - nestedProps: [ - - [0]: { - - testProps: [ - - [0]: "known_val" - ] - } - ] - + nestedProps: output - } - ] -`), - }, - { - "unknown nested level 3", - ` -name: test -runtime: yaml -resources: - auxRes: - type: prov:index:Aux - properties: - auxes: %s - nestedAuxes: %s - mainRes: - type: prov:index:NestedTest - properties: - tests: - - nestedProps: - - ${auxRes.nestedAuxes[0].nestedProps[0]} -`, - nestedProvTestKnownProgram, - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - + prov:index/nestedTest:NestedTest: (create) - [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] - tests : [ - [0]: { - nestedProps: [ - [0]: output - ] - } - ] -`), - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - ~ prov:index/nestedTest:NestedTest: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] - ~ tests: [ - ~ [0]: { - ~ nestedProps: [ - - [0]: { - - testProps: [ - - [0]: "known_val" - ] - } - + [0]: output - ] - } - ] -`), - }, - { - "unknown nested level 4", - ` -name: test -runtime: yaml -resources: - auxRes: - type: prov:index:Aux - properties: - auxes: %s - nestedAuxes: %s - mainRes: - type: prov:index:NestedTest - properties: - tests: - - nestedProps: - - testProps: ${auxRes.nestedAuxes[0].nestedProps[0].testProps} -`, - nestedProvTestKnownProgram, - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - + prov:index/nestedTest:NestedTest: (create) - [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] - tests : [ - [0]: { - nestedProps: [ - [0]: { - testProps : output - } - ] - } - ] -`), - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - ~ prov:index/nestedTest:NestedTest: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] - ~ tests: [ - ~ [0]: { - ~ nestedProps: [ - ~ [0]: { - - testProps: [ - - [0]: "known_val" - ] - + testProps: output - } - ] - } - ] -`), - }, - { - "unknown nested level 5", - ` -name: test -runtime: yaml -resources: - auxRes: - type: prov:index:Aux - properties: - auxes: %s - nestedAuxes: %s - mainRes: - type: prov:index:NestedTest - properties: - tests: - - nestedProps: - - testProps: - - ${auxRes.nestedAuxes[0].nestedProps[0].testProps[0]} -`, - nestedProvTestKnownProgram, - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - + prov:index/nestedTest:NestedTest: (create) - [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] - tests : [ - [0]: { - nestedProps: [ - [0]: { - testProps : [ - [0]: output - ] - } - ] - } - ] -`), - autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - ~ prov:index/nestedTest:NestedTest: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] - ~ tests: [ - ~ [0]: { - ~ nestedProps: [ - ~ [0]: { - ~ testProps: [ - ~ [0]: "known_val" => output - ] - } - ] - } - ] -`), - }, - } { - t.Run(tc.name, func(t *testing.T) { + for _, scenario := range scenarios { + t.Run(scenario.name, func(t *testing.T) { t.Parallel() - computedProgram := fmt.Sprintf(tc.program, "null", "null") - - t.Run("initial preview", func(t *testing.T) { - pt := pulcheck.PulCheck(t, bridgedProvider, computedProgram) - res := pt.Preview(t, optpreview.Diff()) - t.Log(res.StdOut) - - tc.expectedInitial.Equal(t, trimDiff(t, res.StdOut)) - }) - - t.Run("update preview", func(t *testing.T) { - t.Parallel() - t.Skipf("Skipping this test as it this case is not handled by the TF plugin sdk") - // The TF plugin SDK does not handle removing an input for a computed value, even if the provider implements it. - // The plugin SDK always fills an empty Computed property with the value from the state. - // Diff in these cases always returns no diff and the old state value is used. - nonComputedProgram := fmt.Sprintf(tc.program, "[{testProp: \"val1\"}]", "[{nestedProps: [{testProps: [\"val1\"]}]}]") - pt := pulcheck.PulCheck(t, bridgedProvider, nonComputedProgram) - pt.Up(t) - - pulumiYamlPath := filepath.Join(pt.CurrentStack().Workspace().WorkDir(), "Pulumi.yaml") - - err := os.WriteFile(pulumiYamlPath, []byte(computedProgram), 0o600) - require.NoError(t, err) - - res := pt.Preview(t, optpreview.Diff()) - t.Log(res.StdOut) - tc.expectedUpdate.Equal(t, trimDiff(t, res.StdOut)) - }) - - t.Run("update preview with computed", func(t *testing.T) { - t.Parallel() - pt := pulcheck.PulCheck(t, bridgedProvider, tc.initialKnownProgram) - pt.Up(t) - - pulumiYamlPath := filepath.Join(pt.CurrentStack().Workspace().WorkDir(), "Pulumi.yaml") - - err := os.WriteFile(pulumiYamlPath, []byte(computedProgram), 0o600) - require.NoError(t, err) - - res := pt.Preview(t, optpreview.Diff()) - t.Log(res.StdOut) - tc.expectedUpdate.Equal(t, trimDiff(t, res.StdOut)) + diff := crosstests.Diff(t, &res, ctyVal(scenario.initialValue), ctyVal(scenario.changeValue)) + autogold.ExpectFile(t, testOutput{ + initialValue: scenario.initialValue, + changeValue: scenario.changeValue, + tfOut: diff.TFOut, + pulumiOut: diff.PulumiOut, + detailedDiff: diff.PulumiDiff.DetailedDiff, }) }) } } -func TestDetailedDiffPlainTypes(t *testing.T) { +func TestDetailedDiffMap(t *testing.T) { t.Parallel() - resMap := map[string]*schema.Resource{ - "prov_test": { - Schema: map[string]*schema.Schema{ - "string_prop": { - Type: schema.TypeString, - Optional: true, - }, - "list_prop": { - Type: schema.TypeList, - Optional: true, - Elem: &schema.Schema{Type: schema.TypeString}, - }, - "set_prop": { - Type: schema.TypeSet, - Optional: true, - Elem: &schema.Schema{Type: schema.TypeString}, - }, - "map_prop": { - Type: schema.TypeMap, - Optional: true, - Elem: &schema.Schema{Type: schema.TypeString}, - }, - "list_block": { - Type: schema.TypeList, - Optional: true, - Elem: &schema.Resource{ - Schema: map[string]*schema.Schema{ - "prop": { - Type: schema.TypeString, - Optional: true, - }, - }, - }, - }, - "set_block": { - Type: schema.TypeSet, - Optional: true, - Elem: &schema.Resource{ - Schema: map[string]*schema.Schema{ - "prop": { - Type: schema.TypeString, - Optional: true, - }, - }, - }, - }, - "max_items_one_block": { - Type: schema.TypeList, - Optional: true, - MaxItems: 1, - Elem: &schema.Resource{ - Schema: map[string]*schema.Schema{ - "prop": { - Type: schema.TypeString, - Optional: true, - }, - }, - }, - }, + + res := schema.Resource{ + Schema: map[string]*schema.Schema{ + "map_prop": { + Type: schema.TypeMap, + Optional: true, + Elem: &schema.Schema{Type: schema.TypeString}, }, }, } - tfp := &schema.Provider{ResourcesMap: resMap} - bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp, pulcheck.EnableAccurateBridgePreviews()) - - program := ` -name: test -runtime: yaml -resources: - mainRes: - type: prov:index:Test - properties: %s -` - - for _, tc := range []struct { - name string - props1 interface{} - props2 interface{} - expected autogold.Value - }{ - { - "string unchanged", - map[string]interface{}{"stringProp": "val"}, - map[string]interface{}{"stringProp": "val"}, - autogold.Expect("\n"), - }, - { - "string added", - map[string]interface{}{}, - map[string]interface{}{"stringProp": "val"}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - + stringProp: "val" -`), - }, - { - "string removed", - map[string]interface{}{"stringProp": "val1"}, - map[string]interface{}{}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - stringProp: "val1" -`), - }, - { - "string changed", - map[string]interface{}{"stringProp": "val1"}, - map[string]interface{}{"stringProp": "val2"}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ stringProp: "val1" => "val2" -`), - }, - { - "list unchanged", - map[string]interface{}{"listProps": []interface{}{"val"}}, - map[string]interface{}{"listProps": []interface{}{"val"}}, - autogold.Expect("\n"), - }, - { - "list added", - map[string]interface{}{}, - map[string]interface{}{"listProps": []interface{}{"val"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - + listProps: [ - + [0]: "val" - ] -`), - }, - // pulumi/pulumi-terraform-bridge#2233: This is the intended behavior - { - "list added empty", - map[string]interface{}{}, - map[string]interface{}{"listProps": []interface{}{}}, - autogold.Expect("\n"), - }, - { - "list removed", - map[string]interface{}{"listProps": []interface{}{"val"}}, - map[string]interface{}{}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - listProps: [ - - [0]: "val" - ] -`), - }, - // pulumi/pulumi-terraform-bridge#2233: This is the intended behavior - { - "list removed empty", - map[string]interface{}{"listProps": []interface{}{}}, - map[string]interface{}{}, - autogold.Expect("\n"), - }, - { - "list element added front", - map[string]interface{}{"listProps": []interface{}{"val2", "val3"}}, - map[string]interface{}{"listProps": []interface{}{"val1", "val2", "val3"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listProps: [ - ~ [0]: "val2" => "val1" - ~ [1]: "val3" => "val2" - + [2]: "val3" - ] -`), - }, - { - "list element added back", - map[string]interface{}{"listProps": []interface{}{"val1", "val2"}}, - map[string]interface{}{"listProps": []interface{}{"val1", "val2", "val3"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listProps: [ - + [2]: "val3" - ] -`), - }, - { - "list element added middle", - map[string]interface{}{"listProps": []interface{}{"val1", "val3"}}, - map[string]interface{}{"listProps": []interface{}{"val1", "val2", "val3"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listProps: [ - ~ [1]: "val3" => "val2" - + [2]: "val3" - ] -`), - }, - { - "list element removed front", - map[string]interface{}{"listProps": []interface{}{"val1", "val2", "val3"}}, - map[string]interface{}{"listProps": []interface{}{"val2", "val3"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listProps: [ - ~ [0]: "val1" => "val2" - ~ [1]: "val2" => "val3" - - [2]: "val3" - ] -`), - }, - { - "list element removed back", - map[string]interface{}{"listProps": []interface{}{"val1", "val2", "val3"}}, - map[string]interface{}{"listProps": []interface{}{"val1", "val2"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listProps: [ - - [2]: "val3" - ] -`), - }, - { - "list element removed middle", - map[string]interface{}{"listProps": []interface{}{"val1", "val2", "val3"}}, - map[string]interface{}{"listProps": []interface{}{"val1", "val3"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listProps: [ - ~ [1]: "val2" => "val3" - - [2]: "val3" - ] -`), - }, - { - "list element changed", - map[string]interface{}{"listProps": []interface{}{"val1"}}, - map[string]interface{}{"listProps": []interface{}{"val2"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listProps: [ - ~ [0]: "val1" => "val2" - ] -`), - }, - { - "set unchanged", - map[string]interface{}{"setProps": []interface{}{"val"}}, - map[string]interface{}{"setProps": []interface{}{"val"}}, - autogold.Expect("\n"), - }, - { - "set added", - map[string]interface{}{}, - map[string]interface{}{"setProps": []interface{}{"val"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - + setProps: [ - + [0]: "val" - ] -`), - }, - // pulumi/pulumi-terraform-bridge#2233: This is the intended behavior - { - "set added empty", - map[string]interface{}{}, - map[string]interface{}{"setProps": []interface{}{}}, - autogold.Expect("\n"), - }, - { - "set removed", - map[string]interface{}{"setProps": []interface{}{"val"}}, - map[string]interface{}{}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - setProps: [ - - [0]: "val" - ] -`), - }, - // pulumi/pulumi-terraform-bridge#2233: This is the intended behavior - { - "set removed empty", - map[string]interface{}{"setProps": []interface{}{}}, - map[string]interface{}{}, - autogold.Expect("\n"), - }, - { - "set element added front", - map[string]interface{}{"setProps": []interface{}{"val2", "val3"}}, - map[string]interface{}{"setProps": []interface{}{"val1", "val2", "val3"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setProps: [ - + [0]: "val1" - ] -`), - }, - { - "set element added back", - map[string]interface{}{"setProps": []interface{}{"val1", "val2"}}, - map[string]interface{}{"setProps": []interface{}{"val1", "val2", "val3"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setProps: [ - + [2]: "val3" - ] -`), - }, - { - "set element added middle", - map[string]interface{}{"setProps": []interface{}{"val1", "val3"}}, - map[string]interface{}{"setProps": []interface{}{"val1", "val2", "val3"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setProps: [ - + [1]: "val2" - ] -`), - }, - { - "set element removed front", - map[string]interface{}{"setProps": []interface{}{"val1", "val2", "val3"}}, - map[string]interface{}{"setProps": []interface{}{"val2", "val3"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setProps: [ - - [0]: "val1" - ] -`), - }, - { - "set element removed back", - map[string]interface{}{"setProps": []interface{}{"val1", "val2", "val3"}}, - map[string]interface{}{"setProps": []interface{}{"val1", "val2"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setProps: [ - - [2]: "val3" - ] -`), - }, - { - "set element removed middle", - map[string]interface{}{"setProps": []interface{}{"val1", "val2", "val3"}}, - map[string]interface{}{"setProps": []interface{}{"val1", "val3"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setProps: [ - - [1]: "val2" - ] -`), - }, - { - "set element changed", - map[string]interface{}{"setProps": []interface{}{"val1"}}, - map[string]interface{}{"setProps": []interface{}{"val2"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setProps: [ - ~ [0]: "val1" => "val2" - ] -`), - }, - { - "map unchanged", - map[string]interface{}{"mapProp": map[string]interface{}{"key": "val"}}, - map[string]interface{}{"mapProp": map[string]interface{}{"key": "val"}}, - autogold.Expect("\n"), - }, - { - "map added", - map[string]interface{}{}, - map[string]interface{}{"mapProp": map[string]interface{}{"key": "val"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - + mapProp: { - + key: "val" - } -`), - }, - // pulumi/pulumi-terraform-bridge#2233: This is the intended behavior - { - "map added empty", - map[string]interface{}{}, - map[string]interface{}{"mapProp": map[string]interface{}{}}, - autogold.Expect("\n"), - }, - { - "map removed", - map[string]interface{}{"mapProp": map[string]interface{}{"key": "val"}}, - map[string]interface{}{}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - mapProp: { - - key: "val" - } -`), - }, - // pulumi/pulumi-terraform-bridge#2233: This is the intended behavior - { - "map removed empty", - map[string]interface{}{"mapProp": map[string]interface{}{}}, - map[string]interface{}{}, - autogold.Expect("\n"), - }, - { - "map element added", - map[string]interface{}{"mapProp": map[string]interface{}{}}, - map[string]interface{}{"mapProp": map[string]interface{}{"key": "val"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - + mapProp: { - + key: "val" - } -`), - }, - { - "map element removed", - map[string]interface{}{"mapProp": map[string]interface{}{"key": "val"}}, - map[string]interface{}{"mapProp": map[string]interface{}{}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ mapProp: { - - key: "val" - } -`), - }, - { - "map value changed", - map[string]interface{}{"mapProp": map[string]interface{}{"key": "val1"}}, - map[string]interface{}{"mapProp": map[string]interface{}{"key": "val2"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ mapProp: { - ~ key: "val1" => "val2" - } -`), - }, - { - "map key changed", - map[string]interface{}{"mapProp": map[string]interface{}{"key1": "val"}}, - map[string]interface{}{"mapProp": map[string]interface{}{"key2": "val"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ mapProp: { - - key1: "val" - + key2: "val" - } -`), - }, - { - "list block unchanged", - map[string]interface{}{"listBlocks": []interface{}{map[string]interface{}{"prop": "val"}}}, - map[string]interface{}{"listBlocks": []interface{}{map[string]interface{}{"prop": "val"}}}, - autogold.Expect("\n"), - }, - { - "list block added", - map[string]interface{}{}, - map[string]interface{}{"listBlocks": []interface{}{map[string]interface{}{"prop": "val"}}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - + listBlocks: [ - + [0]: { - + prop : "val" - } - ] -`), - }, - // This is expected to be a no-op because blocks can not be nil in TF - { - "list block added empty", - map[string]interface{}{}, - map[string]interface{}{"listBlocks": []interface{}{}}, - autogold.Expect("\n"), - }, - { - "list block added empty object", - map[string]interface{}{}, - map[string]interface{}{"listBlocks": []interface{}{map[string]interface{}{}}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - + listBlocks: [ - + [0]: { - } - ] -`), - }, - { - "list block removed", - map[string]interface{}{"listBlocks": []interface{}{map[string]interface{}{"prop": "val"}}}, - map[string]interface{}{}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - listBlocks: [ - - [0]: { - - prop: "val" - } - ] -`), - }, - // This is expected to be a no-op because blocks can not be nil in TF - { - "list block removed empty", - map[string]interface{}{"listBlocks": []interface{}{}}, - map[string]interface{}{}, - autogold.Expect("\n"), - }, - { - "list block removed empty object", - map[string]interface{}{"listBlocks": []interface{}{map[string]interface{}{}}}, - map[string]interface{}{}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - listBlocks: [ - - [0]: { - - prop: - } - ] -`), - }, - { - "list block element added front", - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listBlocks: [ - ~ [0]: { - ~ prop: "val2" => "val1" - } - ~ [1]: { - ~ prop: "val3" => "val2" - } - + [2]: { - + prop : "val3" - } - ] -`), - }, - { - "list block element added back", - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - }}, - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listBlocks: [ - + [2]: { - + prop : "val3" - } - ] -`), - }, - { - "list block element added middle", - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val3"}, - }}, - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listBlocks: [ - ~ [1]: { - ~ prop: "val3" => "val2" - } - + [2]: { - + prop : "val3" - } - ] -`), - }, - { - "list block element removed front", - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listBlocks: [ - ~ [0]: { - ~ prop: "val1" => "val2" - } - ~ [1]: { - ~ prop: "val2" => "val3" - } - - [2]: { - - prop: "val3" - } - ] -`), - }, - { - "list block element removed back", - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listBlocks: [ - - [2]: { - - prop: "val3" - } - ] -`), - }, - { - "list block element removed middle", - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val3"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listBlocks: [ - ~ [1]: { - ~ prop: "val2" => "val3" - } - - [2]: { - - prop: "val3" - } - ] -`), - }, - { - "list block element changed", - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - }}, - map[string]interface{}{"listBlocks": []interface{}{ - map[string]interface{}{"prop": "val2"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ listBlocks: [ - ~ [0]: { - ~ prop: "val1" => "val2" - } - ] -`), - }, - { - "set block unchanged", - map[string]interface{}{"setBlocks": []interface{}{map[string]interface{}{"prop": "val"}}}, - map[string]interface{}{"setBlocks": []interface{}{map[string]interface{}{"prop": "val"}}}, - autogold.Expect("\n"), - }, - { - "set block added", - map[string]interface{}{}, - map[string]interface{}{"setBlocks": []interface{}{map[string]interface{}{"prop": "val"}}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - + setBlocks: [ - + [0]: { - + prop : "val" - } - ] -`), - }, - // This is expected to be a no-op because blocks can not be nil in TF - { - "set block added empty", - map[string]interface{}{}, - map[string]interface{}{"setBlocks": []interface{}{}}, - autogold.Expect("\n"), - }, - { - "set block added empty object", - map[string]interface{}{}, - map[string]interface{}{"setBlocks": []interface{}{map[string]interface{}{}}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - + setBlocks: [ - + [0]: { - } - ] -`), - }, - { - "set block removed", - map[string]interface{}{"setBlocks": []interface{}{map[string]interface{}{"prop": "val"}}}, - map[string]interface{}{}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - setBlocks: [ - - [0]: { - - prop: "val" - } - ] -`), - }, - // This is expected to be a no-op because blocks can not be nil in TF - { - "set block removed empty", - map[string]interface{}{"setBlocks": []interface{}{}}, - map[string]interface{}{}, - autogold.Expect("\n"), - }, - // TODO[pulumi/pulumi-terraform-bridge#2399] nested prop diff - { - "set block removed empty object", - map[string]interface{}{"setBlocks": []interface{}{map[string]interface{}{}}}, - map[string]interface{}{}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - setBlocks: [ - - [0]: { - - prop: "" - } - ] -`), - }, - { - "set block element added front", - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setBlocks: [ - + [0]: { - + prop : "val1" - } - ] -`), - }, - { - "set block element added back", - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - }}, - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setBlocks: [ - + [2]: { - + prop : "val3" - } - ] -`), - }, - { - "set block element added middle", - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val3"}, - }}, - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setBlocks: [ - + [1]: { - + prop : "val2" - } - ] -`), - }, - { - "set block element removed front", - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setBlocks: [ - - [0]: { - - prop: "val1" - } - ] -`), - }, - { - "set block element removed back", - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setBlocks: [ - - [2]: { - - prop: "val3" - } - ] -`), - }, - { - "set block element removed middle", - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val2"}, - map[string]interface{}{"prop": "val3"}, - }}, - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - map[string]interface{}{"prop": "val3"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setBlocks: [ - - [1]: { - - prop: "val2" - } - ] -`), - }, - { - "set block element changed", - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val1"}, - }}, - map[string]interface{}{"setBlocks": []interface{}{ - map[string]interface{}{"prop": "val2"}, - }}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ setBlocks: [ - ~ [0]: { - ~ prop: "val1" => "val2" - } - ] -`), - }, - { - "maxItemsOne block unchanged", - map[string]interface{}{"maxItemsOneBlock": map[string]interface{}{"prop": "val"}}, - map[string]interface{}{"maxItemsOneBlock": map[string]interface{}{"prop": "val"}}, - autogold.Expect("\n"), - }, - { - "maxItemsOne block added", - map[string]interface{}{}, - map[string]interface{}{"maxItemsOneBlock": map[string]interface{}{"prop": "val"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - + maxItemsOneBlock: { - + prop : "val" - } -`), - }, - { - "maxItemsOne block added empty", - map[string]interface{}{}, - map[string]interface{}{"maxItemsOneBlock": map[string]interface{}{}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - + maxItemsOneBlock: { - } -`), - }, - { - "maxItemsOne block removed", - map[string]interface{}{"maxItemsOneBlock": map[string]interface{}{"prop": "val"}}, - map[string]interface{}{}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - maxItemsOneBlock: { - - prop: "val" - } -`), - }, - // TODO[pulumi/pulumi-terraform-bridge#2399] nested prop diff - { - "maxItemsOne block removed empty", - map[string]interface{}{"maxItemsOneBlock": map[string]interface{}{}}, - map[string]interface{}{}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - maxItemsOneBlock: { - - prop: - } -`), - }, - { - "maxItemsOne block changed", - map[string]interface{}{"maxItemsOneBlock": map[string]interface{}{"prop": "val1"}}, - map[string]interface{}{"maxItemsOneBlock": map[string]interface{}{"prop": "val2"}}, - autogold.Expect(` - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ maxItemsOneBlock: { - ~ prop: "val1" => "val2" - } -`), - }, - } { - tc := tc - t.Run(tc.name, func(t *testing.T) { - t.Parallel() - props1, err := json.Marshal(tc.props1) - require.NoError(t, err) - program1 := fmt.Sprintf(program, string(props1)) - props2, err := json.Marshal(tc.props2) - require.NoError(t, err) - program2 := fmt.Sprintf(program, string(props2)) - pt := pulcheck.PulCheck(t, bridgedProvider, program1) - pt.Up(t) - - pulumiYamlPath := filepath.Join(pt.CurrentStack().Workspace().WorkDir(), "Pulumi.yaml") - - err = os.WriteFile(pulumiYamlPath, []byte(program2), 0o600) - require.NoError(t, err) - res := pt.Preview(t, optpreview.Diff()) - t.Log(res.StdOut) - tc.expected.Equal(t, trimDiff(t, res.StdOut)) - }) - } -} + ctyVal := func(v map[string]string) map[string]cty.Value { + ctyMap := make(map[string]cty.Value) -func TestUnknownCollectionForceNewDetailedDiff(t *testing.T) { - t.Parallel() - collectionForceNewResource := func(typ schema.ValueType) *schema.Resource { - return &schema.Resource{ - Schema: map[string]*schema.Schema{ - "test": { - Type: typ, - Optional: true, - ForceNew: true, - Elem: &schema.Resource{ - Schema: map[string]*schema.Schema{ - "prop": { - Type: schema.TypeString, - Optional: true, - }, - }, - }, - }, - }, + if len(v) == 0 { + return map[string]cty.Value{ + "map_prop": cty.MapValEmpty(cty.String), + } } - } - propertyForceNewResource := func(typ schema.ValueType) *schema.Resource { - return &schema.Resource{ - Schema: map[string]*schema.Schema{ - "test": { - Type: typ, - Optional: true, - Elem: &schema.Resource{ - Schema: map[string]*schema.Schema{ - "prop": { - Type: schema.TypeString, - Optional: true, - ForceNew: true, - }, - }, - }, - }, - }, + for k, v := range v { + ctyMap[k] = cty.StringVal(v) } - } - - auxResource := func(typ schema.ValueType) *schema.Resource { - return &schema.Resource{ - Schema: map[string]*schema.Schema{ - "aux": { - Type: typ, - Computed: true, - Elem: &schema.Resource{ - Schema: map[string]*schema.Schema{ - "prop": { - Type: schema.TypeString, - Computed: true, - }, - }, - }, - }, - }, - CreateContext: func(_ context.Context, d *schema.ResourceData, _ interface{}) diag.Diagnostics { - d.SetId("aux") - err := d.Set("aux", []map[string]interface{}{{"prop": "aux"}}) - require.NoError(t, err) - return nil - }, + return map[string]cty.Value{ + "map_prop": cty.MapVal(ctyMap), } } - initialProgram := ` - name: test - runtime: yaml - resources: - mainRes: - type: prov:index:Test - properties: - tests: [{prop: 'value'}] -` - - program := ` - name: test - runtime: yaml - resources: - auxRes: - type: prov:index:Aux - mainRes: - type: prov:index:Test - properties: - tests: %s -` - - runTest := func(t *testing.T, program2 string, bridgedProvider info.Provider, expectedOutput autogold.Value) { - pt := pulcheck.PulCheck(t, bridgedProvider, initialProgram) - pt.Up(t) - pt.WritePulumiYaml(t, program2) - - res := pt.Preview(t, optpreview.Diff()) - - expectedOutput.Equal(t, trimDiff(t, res.StdOut)) + scenarios := []struct { + name string + initialValue map[string]string + changeValue map[string]string + }{ + {"unchanged empty", map[string]string{}, map[string]string{}}, + {"unchanged non-empty", map[string]string{"key": "val"}, map[string]string{"key": "val"}}, + {"added", map[string]string{}, map[string]string{"key": "val"}}, + {"removed", map[string]string{"key": "val"}, map[string]string{}}, + {"value changed", map[string]string{"key": "val"}, map[string]string{"key": "val2"}}, + {"key changed", map[string]string{"key": "val"}, map[string]string{"key2": "val"}}, } - t.Run("list force new", func(t *testing.T) { - resMap := map[string]*schema.Resource{ - "prov_test": collectionForceNewResource(schema.TypeList), - "prov_aux": auxResource(schema.TypeList), - } - - tfp := &schema.Provider{ResourcesMap: resMap} - bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp, pulcheck.EnableAccurateBridgePreviews()) - runTest := func(t *testing.T, program2 string, expectedOutput autogold.Value) { - runTest(t, program2, bridgedProvider, expectedOutput) - } - - t.Run("unknown plain property", func(t *testing.T) { - program2 := fmt.Sprintf(program, "[{prop: \"${auxRes.auxes[0].prop}\"}]") - runTest(t, program2, autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ tests: [ - ~ [0]: { - ~ prop: "value" => output - } - ] -`)) - }) - - t.Run("unknown object", func(t *testing.T) { - program2 := fmt.Sprintf(program, "[\"${auxRes.auxes[0]}\"]") - runTest(t, program2, autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - +-prov:index/test:Test: (replace) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ tests: [ - - [0]: { - - prop: "value" - } - + [0]: output - ] -`)) - }) - - t.Run("unknown collection", func(t *testing.T) { - program2 := fmt.Sprintf(program, "\"${auxRes.auxes}\"") - runTest(t, program2, autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - +-prov:index/test:Test: (replace) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - tests: [ - - [0]: { - - prop: "value" - } - ] - + tests: output -`)) - }) - }) - - t.Run("list property force new", func(t *testing.T) { - resMap := map[string]*schema.Resource{ - "prov_test": propertyForceNewResource(schema.TypeList), - "prov_aux": auxResource(schema.TypeList), - } - - tfp := &schema.Provider{ResourcesMap: resMap} - bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp, pulcheck.EnableAccurateBridgePreviews()) - runTest := func(t *testing.T, program2 string, expectedOutput autogold.Value) { - runTest(t, program2, bridgedProvider, expectedOutput) - } - - t.Run("unknown plain property", func(t *testing.T) { - program2 := fmt.Sprintf(program, "[{prop: \"${auxRes.auxes[0].prop}\"}]") - runTest(t, program2, autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - +-prov:index/test:Test: (replace) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ tests: [ - ~ [0]: { - ~ prop: "value" => output - } - ] -`)) - }) - - t.Run("unknown object", func(t *testing.T) { - program2 := fmt.Sprintf(program, "[\"${auxRes.auxes[0]}\"]") - runTest(t, program2, autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - +-prov:index/test:Test: (replace) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ tests: [ - - [0]: { - - prop: "value" - } - + [0]: output - ] -`)) - }) - - t.Run("unknown collection", func(t *testing.T) { - program2 := fmt.Sprintf(program, "\"${auxRes.auxes}\"") - runTest(t, program2, autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - +-prov:index/test:Test: (replace) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - tests: [ - - [0]: { - - prop: "value" - } - ] - + tests: output -`)) - }) - }) - - t.Run("set force new", func(t *testing.T) { - resMap := map[string]*schema.Resource{ - "prov_test": collectionForceNewResource(schema.TypeSet), - "prov_aux": auxResource(schema.TypeSet), - } - - tfp := &schema.Provider{ResourcesMap: resMap} - bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp, pulcheck.EnableAccurateBridgePreviews()) - runTest := func(t *testing.T, program2 string, expectedOutput autogold.Value) { - runTest(t, program2, bridgedProvider, expectedOutput) - } - - t.Run("unknown plain property", func(t *testing.T) { - program2 := fmt.Sprintf(program, "[{prop: \"${auxRes.auxes[0].prop}\"}]") - runTest(t, program2, autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - ~ prov:index/test:Test: (update) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ tests: [ - ~ [0]: { - ~ prop: "value" => output - } - ] -`)) - }) - - t.Run("unknown object", func(t *testing.T) { - program2 := fmt.Sprintf(program, "[\"${auxRes.auxes[0]}\"]") - runTest(t, program2, autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - +-prov:index/test:Test: (replace) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ tests: [ - - [0]: { - - prop: "value" - } - + [0]: output - ] -`)) - }) - - t.Run("unknown collection", func(t *testing.T) { - program2 := fmt.Sprintf(program, "\"${auxRes.auxes}\"") - runTest(t, program2, autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - +-prov:index/test:Test: (replace) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - tests: [ - - [0]: { - - prop: "value" - } - ] - + tests: output -`)) - }) - }) - - t.Run("set property force new", func(t *testing.T) { - resMap := map[string]*schema.Resource{ - "prov_test": propertyForceNewResource(schema.TypeSet), - "prov_aux": auxResource(schema.TypeSet), - } - - tfp := &schema.Provider{ResourcesMap: resMap} - bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp, pulcheck.EnableAccurateBridgePreviews()) - runTest := func(t *testing.T, program2 string, expectedOutput autogold.Value) { - runTest(t, program2, bridgedProvider, expectedOutput) - } - - t.Run("unknown plain property", func(t *testing.T) { - program2 := fmt.Sprintf(program, "[{prop: \"${auxRes.auxes[0].prop}\"}]") - runTest(t, program2, autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - +-prov:index/test:Test: (replace) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ tests: [ - ~ [0]: { - ~ prop: "value" => output - } - ] -`)) - }) - - t.Run("unknown object", func(t *testing.T) { - program2 := fmt.Sprintf(program, "[\"${auxRes.auxes[0]}\"]") - runTest(t, program2, autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - +-prov:index/test:Test: (replace) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - ~ tests: [ - - [0]: { - - prop: "value" - } - + [0]: output - ] -`)) - }) + type testOutput struct { + initialValue map[string]string + changeValue map[string]string + tfOut string + pulumiOut string + detailedDiff map[string]any + } - t.Run("unknown collection", func(t *testing.T) { - program2 := fmt.Sprintf(program, "\"${auxRes.auxes}\"") - runTest(t, program2, autogold.Expect(` - + prov:index/aux:Aux: (create) - [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] - +-prov:index/test:Test: (replace) - [id=newid] - [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] - - tests: [ - - [0]: { - - prop: "value" - } - ] - + tests: output -`)) + for _, scenario := range scenarios { + t.Run(scenario.name, func(t *testing.T) { + t.Parallel() + diff := crosstests.Diff(t, &res, ctyVal(scenario.initialValue), ctyVal(scenario.changeValue)) + autogold.ExpectFile(t, testOutput{ + initialValue: scenario.initialValue, + changeValue: scenario.changeValue, + tfOut: diff.TFOut, + pulumiOut: diff.PulumiOut, + detailedDiff: diff.PulumiDiff.DetailedDiff, + }) }) - }) + } } diff --git a/pkg/tests/detailed_diff_unknown_test.go b/pkg/tests/detailed_diff_unknown_test.go new file mode 100644 index 000000000..b0faa9d9f --- /dev/null +++ b/pkg/tests/detailed_diff_unknown_test.go @@ -0,0 +1,1058 @@ +package tests + +import ( + "context" + "fmt" + "os" + "path/filepath" + "strings" + "testing" + + "github.com/hashicorp/terraform-plugin-sdk/v2/diag" + "github.com/hashicorp/terraform-plugin-sdk/v2/helper/schema" + "github.com/hexops/autogold/v2" + "github.com/pulumi/pulumi/sdk/v3/go/auto/optpreview" + "github.com/stretchr/testify/require" + + "github.com/pulumi/pulumi-terraform-bridge/v3/pkg/internal/tests/pulcheck" + "github.com/pulumi/pulumi-terraform-bridge/v3/pkg/tfbridge/info" +) + +func TestUnknownHandling(t *testing.T) { + t.Parallel() + resMap := map[string]*schema.Resource{ + "prov_test": { + Schema: map[string]*schema.Schema{ + "test": { + Type: schema.TypeString, + Optional: true, + }, + }, + }, + "prov_aux": { + Schema: map[string]*schema.Schema{ + "aux": { + Type: schema.TypeString, + Computed: true, + Optional: true, + }, + }, + CreateContext: func(_ context.Context, d *schema.ResourceData, _ interface{}) diag.Diagnostics { + d.SetId("aux") + err := d.Set("aux", "aux") + require.NoError(t, err) + return nil + }, + }, + } + tfp := &schema.Provider{ResourcesMap: resMap} + bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp) + program := ` +name: test +runtime: yaml +resources: + auxRes: + type: prov:index:Aux + mainRes: + type: prov:index:Test + properties: + test: ${auxRes.aux} +outputs: + testOut: ${mainRes.test} +` + pt := pulcheck.PulCheck(t, bridgedProvider, program) + res := pt.Preview(t, optpreview.Diff()) + // Test that the test property is unknown at preview time + require.Contains(t, res.StdOut, "test : output") + resUp := pt.Up(t) + // assert that the property gets resolved + require.Equal(t, "aux", resUp.Outputs["testOut"].Value) +} + +func trimDiff(t *testing.T, diff string) string { + urnLine := " [urn=urn:pulumi:test::test::pulumi:pulumi:Stack::test-test]" + resourcesSummaryLine := "Resources:\n" + require.Contains(t, diff, urnLine) + require.Contains(t, diff, resourcesSummaryLine) + + // trim the diff to only include the contents after the URN line and before the summary + urnIndex := strings.Index(diff, urnLine) + resourcesSummaryIndex := strings.Index(diff, resourcesSummaryLine) + return diff[urnIndex+len(urnLine) : resourcesSummaryIndex] +} + +func TestUnknownBlocks(t *testing.T) { + t.Parallel() + resMap := map[string]*schema.Resource{ + "prov_test": { + Schema: map[string]*schema.Schema{ + "test": { + Type: schema.TypeList, + Optional: true, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "test_prop": { + Type: schema.TypeString, + Optional: true, + }, + }, + }, + }, + }, + }, + "prov_nested_test": { + Schema: map[string]*schema.Schema{ + "test": { + Type: schema.TypeList, + Optional: true, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "nested_prop": { + Type: schema.TypeList, + Optional: true, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "test_prop": { + Type: schema.TypeList, + Optional: true, + Elem: &schema.Schema{ + Type: schema.TypeString, + }, + }, + }, + }, + }, + }, + }, + }, + }, + }, + "prov_aux": { + Schema: map[string]*schema.Schema{ + "aux": { + Type: schema.TypeList, + Computed: true, + Optional: true, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "test_prop": { + Type: schema.TypeString, + Computed: true, + Optional: true, + }, + }, + }, + }, + "nested_aux": { + Type: schema.TypeList, + Optional: true, + Computed: true, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "nested_prop": { + Type: schema.TypeList, + Optional: true, + Computed: true, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "test_prop": { + Type: schema.TypeList, + Optional: true, + Computed: true, + Elem: &schema.Schema{ + Type: schema.TypeString, + }, + }, + }, + }, + }, + }, + }, + }, + }, + CreateContext: func(_ context.Context, d *schema.ResourceData, _ interface{}) diag.Diagnostics { + d.SetId("aux") + if d.Get("aux") == nil { + err := d.Set("aux", []map[string]interface{}{{"test_prop": "aux"}}) + require.NoError(t, err) + } + if d.Get("nested_aux") == nil { + err := d.Set("nested_aux", []map[string]interface{}{ + { + "nested_prop": []map[string]interface{}{ + {"test_prop": []string{"aux"}}, + }, + }, + }) + require.NoError(t, err) + } + return nil + }, + }, + } + tfp := &schema.Provider{ResourcesMap: resMap} + bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp) + + provTestKnownProgram := ` +name: test +runtime: yaml +resources: + mainRes: + type: prov:index:Test + properties: + tests: + - testProp: "known_val" +` + nestedProvTestKnownProgram := ` +name: test +runtime: yaml +resources: + mainRes: + type: prov:index:NestedTest + properties: + tests: + - nestedProps: + - testProps: + - "known_val" +` + + for _, tc := range []struct { + name string + program string + initialKnownProgram string + expectedInitial autogold.Value + expectedUpdate autogold.Value + }{ + { + "list of objects", + ` +name: test +runtime: yaml +resources: + auxRes: + type: prov:index:Aux + properties: + auxes: %s + nestedAuxes: %s + mainRes: + type: prov:index:Test + properties: + tests: ${auxRes.auxes} +`, + provTestKnownProgram, + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + + prov:index/test:Test: (create) + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + tests : output +`), + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + ~ prov:index/test:Test: (update) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + - tests: [ + - [0]: { + - testProp: "known_val" + } + ] + + tests: output +`), + }, + { + "unknown object", + ` +name: test +runtime: yaml +resources: + auxRes: + type: prov:index:Aux + properties: + auxes: %s + nestedAuxes: %s + mainRes: + type: prov:index:Test + properties: + tests: + - ${auxRes.auxes[0]} +`, + provTestKnownProgram, + + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + + prov:index/test:Test: (create) + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + tests : [ + [0]: output + ] +`), + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + ~ prov:index/test:Test: (update) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + ~ tests: [ + - [0]: { + - testProp: "known_val" + } + + [0]: output + ] +`), + }, + { + "unknown object with others", + ` +name: test +runtime: yaml +resources: + auxRes: + type: prov:index:Aux + properties: + auxes: %s + nestedAuxes: %s + mainRes: + type: prov:index:Test + properties: + tests: + - ${auxRes.auxes[0]} + - {"testProp": "val"} +`, + provTestKnownProgram, + + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + + prov:index/test:Test: (create) + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + tests : [ + [0]: output + [1]: { + testProp : "val" + } + ] +`), + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + ~ prov:index/test:Test: (update) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + ~ tests: [ + - [0]: { + - testProp: "known_val" + } + + [0]: output + + [1]: { + + testProp : "val" + } + ] +`), + }, + { + "unknown nested", + ` +name: test +runtime: yaml +resources: + auxRes: + type: prov:index:Aux + properties: + auxes: %s + nestedAuxes: %s + mainRes: + type: prov:index:NestedTest + properties: + tests: ${auxRes.nestedAuxes} +`, + nestedProvTestKnownProgram, + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + + prov:index/nestedTest:NestedTest: (create) + [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] + tests : output +`), + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + ~ prov:index/nestedTest:NestedTest: (update) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] + - tests: [ + - [0]: { + - nestedProps: [ + - [0]: { + - testProps: [ + - [0]: "known_val" + ] + } + ] + } + ] + + tests: output +`), + }, + { + "unknown nested level 1", + ` +name: test +runtime: yaml +resources: + auxRes: + type: prov:index:Aux + properties: + auxes: %s + nestedAuxes: %s + mainRes: + type: prov:index:NestedTest + properties: + tests: + - ${auxRes.nestedAuxes[0]} +`, + nestedProvTestKnownProgram, + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + + prov:index/nestedTest:NestedTest: (create) + [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] + tests : [ + [0]: output + ] +`), + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + ~ prov:index/nestedTest:NestedTest: (update) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] + ~ tests: [ + - [0]: { + - nestedProps: [ + - [0]: { + - testProps: [ + - [0]: "known_val" + ] + } + ] + } + + [0]: output + ] +`), + }, + { + "unknown nested level 2", + ` +name: test +runtime: yaml +resources: + auxRes: + type: prov:index:Aux + properties: + auxes: %s + nestedAuxes: %s + mainRes: + type: prov:index:NestedTest + properties: + tests: + - nestedProps: ${auxRes.nestedAuxes[0].nestedProps} +`, + nestedProvTestKnownProgram, + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + + prov:index/nestedTest:NestedTest: (create) + [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] + tests : [ + [0]: { + nestedProps: output + } + ] +`), + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + ~ prov:index/nestedTest:NestedTest: (update) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] + ~ tests: [ + ~ [0]: { + - nestedProps: [ + - [0]: { + - testProps: [ + - [0]: "known_val" + ] + } + ] + + nestedProps: output + } + ] +`), + }, + { + "unknown nested level 3", + ` +name: test +runtime: yaml +resources: + auxRes: + type: prov:index:Aux + properties: + auxes: %s + nestedAuxes: %s + mainRes: + type: prov:index:NestedTest + properties: + tests: + - nestedProps: + - ${auxRes.nestedAuxes[0].nestedProps[0]} +`, + nestedProvTestKnownProgram, + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + + prov:index/nestedTest:NestedTest: (create) + [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] + tests : [ + [0]: { + nestedProps: [ + [0]: output + ] + } + ] +`), + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + ~ prov:index/nestedTest:NestedTest: (update) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] + ~ tests: [ + ~ [0]: { + ~ nestedProps: [ + - [0]: { + - testProps: [ + - [0]: "known_val" + ] + } + + [0]: output + ] + } + ] +`), + }, + { + "unknown nested level 4", + ` +name: test +runtime: yaml +resources: + auxRes: + type: prov:index:Aux + properties: + auxes: %s + nestedAuxes: %s + mainRes: + type: prov:index:NestedTest + properties: + tests: + - nestedProps: + - testProps: ${auxRes.nestedAuxes[0].nestedProps[0].testProps} +`, + nestedProvTestKnownProgram, + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + + prov:index/nestedTest:NestedTest: (create) + [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] + tests : [ + [0]: { + nestedProps: [ + [0]: { + testProps : output + } + ] + } + ] +`), + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + ~ prov:index/nestedTest:NestedTest: (update) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] + ~ tests: [ + ~ [0]: { + ~ nestedProps: [ + ~ [0]: { + - testProps: [ + - [0]: "known_val" + ] + + testProps: output + } + ] + } + ] +`), + }, + { + "unknown nested level 5", + ` +name: test +runtime: yaml +resources: + auxRes: + type: prov:index:Aux + properties: + auxes: %s + nestedAuxes: %s + mainRes: + type: prov:index:NestedTest + properties: + tests: + - nestedProps: + - testProps: + - ${auxRes.nestedAuxes[0].nestedProps[0].testProps[0]} +`, + nestedProvTestKnownProgram, + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + + prov:index/nestedTest:NestedTest: (create) + [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] + tests : [ + [0]: { + nestedProps: [ + [0]: { + testProps : [ + [0]: output + ] + } + ] + } + ] +`), + autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + ~ prov:index/nestedTest:NestedTest: (update) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/nestedTest:NestedTest::mainRes] + ~ tests: [ + ~ [0]: { + ~ nestedProps: [ + ~ [0]: { + ~ testProps: [ + ~ [0]: "known_val" => output + ] + } + ] + } + ] +`), + }, + } { + t.Run(tc.name, func(t *testing.T) { + t.Parallel() + computedProgram := fmt.Sprintf(tc.program, "null", "null") + + t.Run("initial preview", func(t *testing.T) { + pt := pulcheck.PulCheck(t, bridgedProvider, computedProgram) + res := pt.Preview(t, optpreview.Diff()) + t.Log(res.StdOut) + + tc.expectedInitial.Equal(t, trimDiff(t, res.StdOut)) + }) + + t.Run("update preview", func(t *testing.T) { + t.Parallel() + t.Skipf("Skipping this test as it this case is not handled by the TF plugin sdk") + // The TF plugin SDK does not handle removing an input for a computed value, even if the provider implements it. + // The plugin SDK always fills an empty Computed property with the value from the state. + // Diff in these cases always returns no diff and the old state value is used. + nonComputedProgram := fmt.Sprintf(tc.program, "[{testProp: \"val1\"}]", "[{nestedProps: [{testProps: [\"val1\"]}]}]") + pt := pulcheck.PulCheck(t, bridgedProvider, nonComputedProgram) + pt.Up(t) + + pulumiYamlPath := filepath.Join(pt.CurrentStack().Workspace().WorkDir(), "Pulumi.yaml") + + err := os.WriteFile(pulumiYamlPath, []byte(computedProgram), 0o600) + require.NoError(t, err) + + res := pt.Preview(t, optpreview.Diff()) + t.Log(res.StdOut) + tc.expectedUpdate.Equal(t, trimDiff(t, res.StdOut)) + }) + + t.Run("update preview with computed", func(t *testing.T) { + t.Parallel() + pt := pulcheck.PulCheck(t, bridgedProvider, tc.initialKnownProgram) + pt.Up(t) + + pulumiYamlPath := filepath.Join(pt.CurrentStack().Workspace().WorkDir(), "Pulumi.yaml") + + err := os.WriteFile(pulumiYamlPath, []byte(computedProgram), 0o600) + require.NoError(t, err) + + res := pt.Preview(t, optpreview.Diff()) + t.Log(res.StdOut) + tc.expectedUpdate.Equal(t, trimDiff(t, res.StdOut)) + }) + }) + } +} + +func TestUnknownCollectionForceNewDetailedDiff(t *testing.T) { + t.Parallel() + collectionForceNewResource := func(typ schema.ValueType) *schema.Resource { + return &schema.Resource{ + Schema: map[string]*schema.Schema{ + "test": { + Type: typ, + Optional: true, + ForceNew: true, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "prop": { + Type: schema.TypeString, + Optional: true, + }, + }, + }, + }, + }, + } + } + + propertyForceNewResource := func(typ schema.ValueType) *schema.Resource { + return &schema.Resource{ + Schema: map[string]*schema.Schema{ + "test": { + Type: typ, + Optional: true, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "prop": { + Type: schema.TypeString, + Optional: true, + ForceNew: true, + }, + }, + }, + }, + }, + } + } + + auxResource := func(typ schema.ValueType) *schema.Resource { + return &schema.Resource{ + Schema: map[string]*schema.Schema{ + "aux": { + Type: typ, + Computed: true, + Elem: &schema.Resource{ + Schema: map[string]*schema.Schema{ + "prop": { + Type: schema.TypeString, + Computed: true, + }, + }, + }, + }, + }, + CreateContext: func(_ context.Context, d *schema.ResourceData, _ interface{}) diag.Diagnostics { + d.SetId("aux") + err := d.Set("aux", []map[string]interface{}{{"prop": "aux"}}) + require.NoError(t, err) + return nil + }, + } + } + + initialProgram := ` + name: test + runtime: yaml + resources: + mainRes: + type: prov:index:Test + properties: + tests: [{prop: 'value'}] +` + + program := ` + name: test + runtime: yaml + resources: + auxRes: + type: prov:index:Aux + mainRes: + type: prov:index:Test + properties: + tests: %s +` + + runTest := func(t *testing.T, program2 string, bridgedProvider info.Provider, expectedOutput autogold.Value) { + pt := pulcheck.PulCheck(t, bridgedProvider, initialProgram) + pt.Up(t) + pt.WritePulumiYaml(t, program2) + + res := pt.Preview(t, optpreview.Diff()) + + expectedOutput.Equal(t, trimDiff(t, res.StdOut)) + } + + t.Run("list force new", func(t *testing.T) { + resMap := map[string]*schema.Resource{ + "prov_test": collectionForceNewResource(schema.TypeList), + "prov_aux": auxResource(schema.TypeList), + } + + tfp := &schema.Provider{ResourcesMap: resMap} + bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp, pulcheck.EnableAccurateBridgePreviews()) + runTest := func(t *testing.T, program2 string, expectedOutput autogold.Value) { + runTest(t, program2, bridgedProvider, expectedOutput) + } + + t.Run("unknown plain property", func(t *testing.T) { + program2 := fmt.Sprintf(program, "[{prop: \"${auxRes.auxes[0].prop}\"}]") + runTest(t, program2, autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + ~ prov:index/test:Test: (update) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + ~ tests: [ + ~ [0]: { + ~ prop: "value" => output + } + ] +`)) + }) + + t.Run("unknown object", func(t *testing.T) { + program2 := fmt.Sprintf(program, "[\"${auxRes.auxes[0]}\"]") + runTest(t, program2, autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + +-prov:index/test:Test: (replace) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + ~ tests: [ + - [0]: { + - prop: "value" + } + + [0]: output + ] +`)) + }) + + t.Run("unknown collection", func(t *testing.T) { + program2 := fmt.Sprintf(program, "\"${auxRes.auxes}\"") + runTest(t, program2, autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + +-prov:index/test:Test: (replace) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + - tests: [ + - [0]: { + - prop: "value" + } + ] + + tests: output +`)) + }) + }) + + t.Run("list property force new", func(t *testing.T) { + resMap := map[string]*schema.Resource{ + "prov_test": propertyForceNewResource(schema.TypeList), + "prov_aux": auxResource(schema.TypeList), + } + + tfp := &schema.Provider{ResourcesMap: resMap} + bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp, pulcheck.EnableAccurateBridgePreviews()) + runTest := func(t *testing.T, program2 string, expectedOutput autogold.Value) { + runTest(t, program2, bridgedProvider, expectedOutput) + } + + t.Run("unknown plain property", func(t *testing.T) { + program2 := fmt.Sprintf(program, "[{prop: \"${auxRes.auxes[0].prop}\"}]") + runTest(t, program2, autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + +-prov:index/test:Test: (replace) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + ~ tests: [ + ~ [0]: { + ~ prop: "value" => output + } + ] +`)) + }) + + t.Run("unknown object", func(t *testing.T) { + program2 := fmt.Sprintf(program, "[\"${auxRes.auxes[0]}\"]") + runTest(t, program2, autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + +-prov:index/test:Test: (replace) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + ~ tests: [ + - [0]: { + - prop: "value" + } + + [0]: output + ] +`)) + }) + + t.Run("unknown collection", func(t *testing.T) { + program2 := fmt.Sprintf(program, "\"${auxRes.auxes}\"") + runTest(t, program2, autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + +-prov:index/test:Test: (replace) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + - tests: [ + - [0]: { + - prop: "value" + } + ] + + tests: output +`)) + }) + }) + + t.Run("set force new", func(t *testing.T) { + resMap := map[string]*schema.Resource{ + "prov_test": collectionForceNewResource(schema.TypeSet), + "prov_aux": auxResource(schema.TypeSet), + } + + tfp := &schema.Provider{ResourcesMap: resMap} + bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp, pulcheck.EnableAccurateBridgePreviews()) + runTest := func(t *testing.T, program2 string, expectedOutput autogold.Value) { + runTest(t, program2, bridgedProvider, expectedOutput) + } + + t.Run("unknown plain property", func(t *testing.T) { + program2 := fmt.Sprintf(program, "[{prop: \"${auxRes.auxes[0].prop}\"}]") + runTest(t, program2, autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + ~ prov:index/test:Test: (update) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + ~ tests: [ + ~ [0]: { + ~ prop: "value" => output + } + ] +`)) + }) + + t.Run("unknown object", func(t *testing.T) { + program2 := fmt.Sprintf(program, "[\"${auxRes.auxes[0]}\"]") + runTest(t, program2, autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + +-prov:index/test:Test: (replace) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + ~ tests: [ + - [0]: { + - prop: "value" + } + + [0]: output + ] +`)) + }) + + t.Run("unknown collection", func(t *testing.T) { + program2 := fmt.Sprintf(program, "\"${auxRes.auxes}\"") + runTest(t, program2, autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + +-prov:index/test:Test: (replace) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + - tests: [ + - [0]: { + - prop: "value" + } + ] + + tests: output +`)) + }) + }) + + t.Run("set property force new", func(t *testing.T) { + resMap := map[string]*schema.Resource{ + "prov_test": propertyForceNewResource(schema.TypeSet), + "prov_aux": auxResource(schema.TypeSet), + } + + tfp := &schema.Provider{ResourcesMap: resMap} + bridgedProvider := pulcheck.BridgedProvider(t, "prov", tfp, pulcheck.EnableAccurateBridgePreviews()) + runTest := func(t *testing.T, program2 string, expectedOutput autogold.Value) { + runTest(t, program2, bridgedProvider, expectedOutput) + } + + t.Run("unknown plain property", func(t *testing.T) { + program2 := fmt.Sprintf(program, "[{prop: \"${auxRes.auxes[0].prop}\"}]") + runTest(t, program2, autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + +-prov:index/test:Test: (replace) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + ~ tests: [ + ~ [0]: { + ~ prop: "value" => output + } + ] +`)) + }) + + t.Run("unknown object", func(t *testing.T) { + program2 := fmt.Sprintf(program, "[\"${auxRes.auxes[0]}\"]") + runTest(t, program2, autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + +-prov:index/test:Test: (replace) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + ~ tests: [ + - [0]: { + - prop: "value" + } + + [0]: output + ] +`)) + }) + + t.Run("unknown collection", func(t *testing.T) { + program2 := fmt.Sprintf(program, "\"${auxRes.auxes}\"") + runTest(t, program2, autogold.Expect(` + + prov:index/aux:Aux: (create) + [urn=urn:pulumi:test::test::prov:index/aux:Aux::auxRes] + +-prov:index/test:Test: (replace) + [id=newid] + [urn=urn:pulumi:test::test::prov:index/test:Test::mainRes] + - tests: [ + - [0]: { + - prop: "value" + } + ] + + tests: output +`)) + }) + }) +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/added_empty.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/added_empty.golden new file mode 100644 index 000000000..9589fbdf0 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/added_empty.golden @@ -0,0 +1,15 @@ +tests.testOutput{ + changeValue: &[]string{}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/added_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/added_non-empty.golden new file mode 100644 index 000000000..f574065dc --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/added_non-empty.golden @@ -0,0 +1,37 @@ +tests.testOutput{ + changeValue: &[]string{ + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + list_attr = [ + + "val1", + ] + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + + listAttrs: [ + + [0]: "val1" + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listAttrs": map[string]interface{}{}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/changed.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/changed.golden new file mode 100644 index 000000000..5337d13ba --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/changed.golden @@ -0,0 +1,38 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + }, + changeValue: &[]string{"val2"}, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ list_attr = [ + ~ "val1" -> "val2", + ] + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listAttrs: [ + ~ [0]: "val1" => "val2" + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listAttrs[0]": map[string]interface{}{"kind": "UPDATE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_added_back.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_added_back.golden new file mode 100644 index 000000000..6661382d3 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_added_back.golden @@ -0,0 +1,45 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + }, + changeValue: &[]string{ + "val1", + "val2", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ list_attr = [ + # (1 unchanged element hidden) + "val2", + + "val3", + ] + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listAttrs: [ + + [2]: "val3" + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listAttrs[2]": map[string]interface{}{}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_added_front.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_added_front.golden new file mode 100644 index 000000000..9d3584432 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_added_front.golden @@ -0,0 +1,51 @@ +tests.testOutput{ + initialValue: &[]string{ + "val2", + "val3", + }, + changeValue: &[]string{ + "val1", + "val2", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ list_attr = [ + + "val1", + "val2", + # (1 unchanged element hidden) + ] + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listAttrs: [ + ~ [0]: "val2" => "val1" + ~ [1]: "val3" => "val2" + + [2]: "val3" + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "listAttrs[0]": map[string]interface{}{"kind": "UPDATE"}, + "listAttrs[1]": map[string]interface{}{"kind": "UPDATE"}, + "listAttrs[2]": map[string]interface{}{}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_added_middle.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_added_middle.golden new file mode 100644 index 000000000..a4cdb618a --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_added_middle.golden @@ -0,0 +1,49 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val3", + }, + changeValue: &[]string{ + "val1", + "val2", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ list_attr = [ + "val1", + + "val2", + "val3", + ] + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listAttrs: [ + ~ [1]: "val3" => "val2" + + [2]: "val3" + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "listAttrs[1]": map[string]interface{}{"kind": "UPDATE"}, + "listAttrs[2]": map[string]interface{}{}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_removed_end.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_removed_end.golden new file mode 100644 index 000000000..0fad4d531 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_removed_end.golden @@ -0,0 +1,52 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val2", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ list_attr = [ + - "val1", + "val2", + - "val3", + + "val1", + ] + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listAttrs: [ + ~ [0]: "val1" => "val2" + ~ [1]: "val2" => "val1" + - [2]: "val3" + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "listAttrs[0]": map[string]interface{}{"kind": "UPDATE"}, + "listAttrs[1]": map[string]interface{}{"kind": "UPDATE"}, + "listAttrs[2]": map[string]interface{}{"kind": "DELETE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_removed_front.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_removed_front.golden new file mode 100644 index 000000000..425e86f7b --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_removed_front.golden @@ -0,0 +1,50 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val3", + "val2", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ list_attr = [ + - "val1", + - "val2", + "val3", + + "val2", + ] + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listAttrs: [ + ~ [0]: "val1" => "val3" + - [2]: "val3" + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "listAttrs[0]": map[string]interface{}{"kind": "UPDATE"}, + "listAttrs[2]": map[string]interface{}{"kind": "DELETE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_removed_middle.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_removed_middle.golden new file mode 100644 index 000000000..d8373408b --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/list_element_removed_middle.golden @@ -0,0 +1,52 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val3", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ list_attr = [ + - "val1", + - "val2", + "val3", + + "val1", + ] + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listAttrs: [ + ~ [0]: "val1" => "val3" + ~ [1]: "val2" => "val1" + - [2]: "val3" + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "listAttrs[0]": map[string]interface{}{"kind": "UPDATE"}, + "listAttrs[1]": map[string]interface{}{"kind": "UPDATE"}, + "listAttrs[2]": map[string]interface{}{"kind": "DELETE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/removed_empty.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/removed_empty.golden new file mode 100644 index 000000000..7bae19762 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/removed_empty.golden @@ -0,0 +1,15 @@ +tests.testOutput{ + initialValue: &[]string{}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/removed_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/removed_non-empty.golden new file mode 100644 index 000000000..91db7793b --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/removed_non-empty.golden @@ -0,0 +1,37 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ list_attr = [ + - "val1", + ] + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + - listAttrs: [ + - [0]: "val1" + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listAttrs": map[string]interface{}{"kind": "DELETE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/unchanged_empty.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/unchanged_empty.golden new file mode 100644 index 000000000..a53fac34d --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/unchanged_empty.golden @@ -0,0 +1,11 @@ +tests.testOutput{tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_attribute/unchanged_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/unchanged_non-empty.golden new file mode 100644 index 000000000..e250a9f9c --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_attribute/unchanged_non-empty.golden @@ -0,0 +1,18 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + }, + changeValue: &[]string{"val1"}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/added_empty.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/added_empty.golden new file mode 100644 index 000000000..9589fbdf0 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/added_empty.golden @@ -0,0 +1,15 @@ +tests.testOutput{ + changeValue: &[]string{}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/added_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/added_non-empty.golden new file mode 100644 index 000000000..303139674 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/added_non-empty.golden @@ -0,0 +1,40 @@ +tests.testOutput{ + changeValue: &[]string{ + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + + list_block { + + prop = "val1" + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + + listBlocks: [ + + [0]: { + + prop : "val1" + } + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listBlocks": map[string]interface{}{}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/changed.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/changed.golden new file mode 100644 index 000000000..b2bedc1d1 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/changed.golden @@ -0,0 +1,41 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + }, + changeValue: &[]string{"val2"}, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + ~ list_block { + ~ prop = "val1" -> "val2" + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listBlocks: [ + ~ [0]: { + ~ prop: "val1" => "val2" + } + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listBlocks[0].prop": map[string]interface{}{"kind": "UPDATE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_added_back.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_added_back.golden new file mode 100644 index 000000000..3982f29d0 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_added_back.golden @@ -0,0 +1,48 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + }, + changeValue: &[]string{ + "val1", + "val2", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + + list_block { + + prop = "val3" + } + + # (2 unchanged blocks hidden) + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listBlocks: [ + + [2]: { + + prop : "val3" + } + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listBlocks[2]": map[string]interface{}{}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_added_front.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_added_front.golden new file mode 100644 index 000000000..ce38ccea8 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_added_front.golden @@ -0,0 +1,62 @@ +tests.testOutput{ + initialValue: &[]string{ + "val2", + "val3", + }, + changeValue: &[]string{ + "val1", + "val2", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + ~ list_block { + ~ prop = "val2" -> "val1" + } + ~ list_block { + ~ prop = "val3" -> "val2" + } + + list_block { + + prop = "val3" + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listBlocks: [ + ~ [0]: { + ~ prop: "val2" => "val1" + } + ~ [1]: { + ~ prop: "val3" => "val2" + } + + [2]: { + + prop : "val3" + } + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "listBlocks[0].prop": map[string]interface{}{"kind": "UPDATE"}, + "listBlocks[1].prop": map[string]interface{}{"kind": "UPDATE"}, + "listBlocks[2]": map[string]interface{}{}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_added_middle.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_added_middle.golden new file mode 100644 index 000000000..708a968c6 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_added_middle.golden @@ -0,0 +1,57 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val3", + }, + changeValue: &[]string{ + "val1", + "val2", + "val3", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + ~ list_block { + ~ prop = "val3" -> "val2" + } + + list_block { + + prop = "val3" + } + + # (1 unchanged block hidden) + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listBlocks: [ + ~ [1]: { + ~ prop: "val3" => "val2" + } + + [2]: { + + prop : "val3" + } + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "listBlocks[1].prop": map[string]interface{}{"kind": "UPDATE"}, + "listBlocks[2]": map[string]interface{}{}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_removed_end.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_removed_end.golden new file mode 100644 index 000000000..7f26041ed --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_removed_end.golden @@ -0,0 +1,62 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val2", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + ~ list_block { + ~ prop = "val1" -> "val2" + } + ~ list_block { + ~ prop = "val2" -> "val1" + } + - list_block { + - prop = "val3" -> null + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listBlocks: [ + ~ [0]: { + ~ prop: "val1" => "val2" + } + ~ [1]: { + ~ prop: "val2" => "val1" + } + - [2]: { + - prop: "val3" + } + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "listBlocks[0].prop": map[string]interface{}{"kind": "UPDATE"}, + "listBlocks[1].prop": map[string]interface{}{"kind": "UPDATE"}, + "listBlocks[2]": map[string]interface{}{"kind": "DELETE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_removed_front.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_removed_front.golden new file mode 100644 index 000000000..f695583e5 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_removed_front.golden @@ -0,0 +1,57 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val3", + "val2", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + ~ list_block { + ~ prop = "val1" -> "val3" + } + - list_block { + - prop = "val3" -> null + } + + # (1 unchanged block hidden) + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listBlocks: [ + ~ [0]: { + ~ prop: "val1" => "val3" + } + - [2]: { + - prop: "val3" + } + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "listBlocks[0].prop": map[string]interface{}{"kind": "UPDATE"}, + "listBlocks[2]": map[string]interface{}{"kind": "DELETE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_removed_middle.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_removed_middle.golden new file mode 100644 index 000000000..fad5f4aef --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/list_element_removed_middle.golden @@ -0,0 +1,62 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + "val2", + "val3", + }, + changeValue: &[]string{ + "val3", + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + ~ list_block { + ~ prop = "val1" -> "val3" + } + ~ list_block { + ~ prop = "val2" -> "val1" + } + - list_block { + - prop = "val3" -> null + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listBlocks: [ + ~ [0]: { + ~ prop: "val1" => "val3" + } + ~ [1]: { + ~ prop: "val2" => "val1" + } + - [2]: { + - prop: "val3" + } + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "listBlocks[0].prop": map[string]interface{}{"kind": "UPDATE"}, + "listBlocks[1].prop": map[string]interface{}{"kind": "UPDATE"}, + "listBlocks[2]": map[string]interface{}{"kind": "DELETE"}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/removed_empty.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/removed_empty.golden new file mode 100644 index 000000000..7bae19762 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/removed_empty.golden @@ -0,0 +1,15 @@ +tests.testOutput{ + initialValue: &[]string{}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/removed_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/removed_non-empty.golden new file mode 100644 index 000000000..59f844e2b --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/removed_non-empty.golden @@ -0,0 +1,40 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + - list_block { + - prop = "val1" -> null + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + - listBlocks: [ + - [0]: { + - prop: "val1" + } + ] +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listBlocks": map[string]interface{}{"kind": "DELETE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/unchanged_empty.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/unchanged_empty.golden new file mode 100644 index 000000000..a53fac34d --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/unchanged_empty.golden @@ -0,0 +1,11 @@ +tests.testOutput{tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`} diff --git a/pkg/tests/testdata/TestDetailedDiffList/list_block/unchanged_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffList/list_block/unchanged_non-empty.golden new file mode 100644 index 000000000..e250a9f9c --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/list_block/unchanged_non-empty.golden @@ -0,0 +1,18 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + }, + changeValue: &[]string{"val1"}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/added_empty.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/added_empty.golden new file mode 100644 index 000000000..9589fbdf0 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/added_empty.golden @@ -0,0 +1,15 @@ +tests.testOutput{ + changeValue: &[]string{}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/added_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/added_non-empty.golden new file mode 100644 index 000000000..89350bb88 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/added_non-empty.golden @@ -0,0 +1,35 @@ +tests.testOutput{ + changeValue: &[]string{ + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + list_attr = [ + + "val1", + ] + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + + listAttr: "val1" +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listAttr": map[string]interface{}{}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/changed.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/changed.golden new file mode 100644 index 000000000..2e7dd97f3 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/changed.golden @@ -0,0 +1,36 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + }, + changeValue: &[]string{"val2"}, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ list_attr = [ + ~ "val1" -> "val2", + ] + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listAttr: "val1" => "val2" +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listAttr": map[string]interface{}{"kind": "UPDATE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/removed_empty.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/removed_empty.golden new file mode 100644 index 000000000..7bae19762 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/removed_empty.golden @@ -0,0 +1,15 @@ +tests.testOutput{ + initialValue: &[]string{}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/removed_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/removed_non-empty.golden new file mode 100644 index 000000000..26e9a05e7 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/removed_non-empty.golden @@ -0,0 +1,35 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ list_attr = [ + - "val1", + ] + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + - listAttr: "val1" +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listAttr": map[string]interface{}{"kind": "DELETE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/unchanged_empty.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/unchanged_empty.golden new file mode 100644 index 000000000..a53fac34d --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/unchanged_empty.golden @@ -0,0 +1,11 @@ +tests.testOutput{tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/unchanged_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/unchanged_non-empty.golden new file mode 100644 index 000000000..e250a9f9c --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_attribute/unchanged_non-empty.golden @@ -0,0 +1,18 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + }, + changeValue: &[]string{"val1"}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/added_empty.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/added_empty.golden new file mode 100644 index 000000000..9589fbdf0 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/added_empty.golden @@ -0,0 +1,15 @@ +tests.testOutput{ + changeValue: &[]string{}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/added_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/added_non-empty.golden new file mode 100644 index 000000000..c81e0e1f0 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/added_non-empty.golden @@ -0,0 +1,38 @@ +tests.testOutput{ + changeValue: &[]string{ + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + + list_block { + + nested_prop = "val1" + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + + listBlock: { + + nestedProp: "val1" + } +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listBlock": map[string]interface{}{}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/changed.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/changed.golden new file mode 100644 index 000000000..c0d558ea3 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/changed.golden @@ -0,0 +1,39 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + }, + changeValue: &[]string{"val2"}, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + ~ list_block { + ~ nested_prop = "val1" -> "val2" + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ listBlock: { + ~ nestedProp: "val1" => "val2" + } +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listBlock.nestedProp": map[string]interface{}{"kind": "UPDATE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/removed_empty.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/removed_empty.golden new file mode 100644 index 000000000..7bae19762 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/removed_empty.golden @@ -0,0 +1,15 @@ +tests.testOutput{ + initialValue: &[]string{}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/removed_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/removed_non-empty.golden new file mode 100644 index 000000000..fbc5e67f8 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/removed_non-empty.golden @@ -0,0 +1,38 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + }, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + - list_block { + - nested_prop = "val1" -> null + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + - listBlock: { + - nestedProp: "val1" + } +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"listBlock": map[string]interface{}{"kind": "DELETE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/unchanged_empty.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/unchanged_empty.golden new file mode 100644 index 000000000..a53fac34d --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/unchanged_empty.golden @@ -0,0 +1,11 @@ +tests.testOutput{tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`} diff --git a/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/unchanged_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/unchanged_non-empty.golden new file mode 100644 index 000000000..e250a9f9c --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffList/max_items_one_block/unchanged_non-empty.golden @@ -0,0 +1,18 @@ +tests.testOutput{ + initialValue: &[]string{ + "val1", + }, + changeValue: &[]string{"val1"}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffMap/added.golden b/pkg/tests/testdata/TestDetailedDiffMap/added.golden new file mode 100644 index 000000000..cc55fbc40 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffMap/added.golden @@ -0,0 +1,36 @@ +tests.testOutput{ + initialValue: map[string]string{}, + changeValue: map[string]string{"key": "val"}, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + map_prop = { + + "key" = "val" + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + + mapProp: { + + key: "val" + } +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"mapProp": map[string]interface{}{}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffMap/key_changed.golden b/pkg/tests/testdata/TestDetailedDiffMap/key_changed.golden new file mode 100644 index 000000000..2193f1ef6 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffMap/key_changed.golden @@ -0,0 +1,43 @@ +tests.testOutput{ + initialValue: map[string]string{ + "key": "val", + }, + changeValue: map[string]string{"key2": "val"}, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ map_prop = { + - "key" = "val" -> null + + "key2" = "val" + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ mapProp: { + - key : "val" + + key2: "val" + } +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{ + "mapProp.key": map[string]interface{}{"kind": "DELETE"}, + "mapProp.key2": map[string]interface{}{}, + }, +} diff --git a/pkg/tests/testdata/TestDetailedDiffMap/removed.golden b/pkg/tests/testdata/TestDetailedDiffMap/removed.golden new file mode 100644 index 000000000..34fbfc62c --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffMap/removed.golden @@ -0,0 +1,38 @@ +tests.testOutput{ + initialValue: map[string]string{ + "key": "val", + }, + changeValue: map[string]string{}, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ map_prop = { + - "key" = "val" -> null + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ mapProp: { + - key: "val" + } +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"mapProp.key": map[string]interface{}{"kind": "DELETE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffMap/unchanged_empty.golden b/pkg/tests/testdata/TestDetailedDiffMap/unchanged_empty.golden new file mode 100644 index 000000000..342ab2068 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffMap/unchanged_empty.golden @@ -0,0 +1,16 @@ +tests.testOutput{ + initialValue: map[string]string{}, + changeValue: map[string]string{}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffMap/unchanged_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffMap/unchanged_non-empty.golden new file mode 100644 index 000000000..95f4be8e3 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffMap/unchanged_non-empty.golden @@ -0,0 +1,18 @@ +tests.testOutput{ + initialValue: map[string]string{ + "key": "val", + }, + changeValue: map[string]string{"key": "val"}, + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +} diff --git a/pkg/tests/testdata/TestDetailedDiffMap/value_changed.golden b/pkg/tests/testdata/TestDetailedDiffMap/value_changed.golden new file mode 100644 index 000000000..3edd45c2c --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffMap/value_changed.golden @@ -0,0 +1,38 @@ +tests.testOutput{ + initialValue: map[string]string{ + "key": "val", + }, + changeValue: map[string]string{"key": "val2"}, + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ map_prop = { + ~ "key" = "val" -> "val2" + } + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ mapProp: { + ~ key: "val" => "val2" + } +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"mapProp.key": map[string]interface{}{"kind": "UPDATE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffString/added.golden b/pkg/tests/testdata/TestDetailedDiffString/added.golden new file mode 100644 index 000000000..e1dba4ce3 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffString/added.golden @@ -0,0 +1,30 @@ +tests.testOutput{ + changeValue: valast.Ptr("val1"), tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + + string_prop = "val1" + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + + stringProp: "val1" +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"stringProp": map[string]interface{}{}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffString/changed.golden b/pkg/tests/testdata/TestDetailedDiffString/changed.golden new file mode 100644 index 000000000..d32fc8650 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffString/changed.golden @@ -0,0 +1,31 @@ +tests.testOutput{ + initialValue: valast.Ptr("val1"), changeValue: valast.Ptr("val2"), + tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + ~ string_prop = "val1" -> "val2" + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + ~ stringProp: "val1" => "val2" +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"stringProp": map[string]interface{}{"kind": "UPDATE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffString/removed.golden b/pkg/tests/testdata/TestDetailedDiffString/removed.golden new file mode 100644 index 000000000..fca1ebb81 --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffString/removed.golden @@ -0,0 +1,30 @@ +tests.testOutput{ + initialValue: valast.Ptr("val1"), tfOut: ` +Terraform used the selected providers to generate the following execution +plan. Resource actions are indicated with the following symbols: + ~ update in-place + +Terraform will perform the following actions: + + # crossprovider_test_res.example will be updated in-place + ~ resource "crossprovider_test_res" "example" { + id = "newid" + - string_prop = "val1" -> null + } + +Plan: 0 to add, 1 to change, 0 to destroy. + +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] + ~ crossprovider:index/testRes:TestRes: (update) + [id=newid] + [urn=urn:pulumi:test::project::crossprovider:index/testRes:TestRes::example] + - stringProp: "val1" +Resources: + ~ 1 to update + 1 unchanged +`, + detailedDiff: map[string]interface{}{"stringProp": map[string]interface{}{"kind": "DELETE"}}, +} diff --git a/pkg/tests/testdata/TestDetailedDiffString/unchanged_empty.golden b/pkg/tests/testdata/TestDetailedDiffString/unchanged_empty.golden new file mode 100644 index 000000000..a53fac34d --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffString/unchanged_empty.golden @@ -0,0 +1,11 @@ +tests.testOutput{tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`} diff --git a/pkg/tests/testdata/TestDetailedDiffString/unchanged_non-empty.golden b/pkg/tests/testdata/TestDetailedDiffString/unchanged_non-empty.golden new file mode 100644 index 000000000..25581866f --- /dev/null +++ b/pkg/tests/testdata/TestDetailedDiffString/unchanged_non-empty.golden @@ -0,0 +1,15 @@ +tests.testOutput{ + initialValue: valast.Ptr("val1"), changeValue: valast.Ptr("val1"), + tfOut: ` +No changes. Your infrastructure matches the configuration. + +Terraform has compared your real infrastructure against your configuration +and found no differences, so no changes are needed. +`, + pulumiOut: `Previewing update (test): + pulumi:pulumi:Stack: (same) + [urn=urn:pulumi:test::project::pulumi:pulumi:Stack::project-test] +Resources: + 2 unchanged +`, +}