From 5cd6861816868d33db0909d57e416bc0935ef825 Mon Sep 17 00:00:00 2001 From: Ash Rohde Date: Thu, 19 Nov 2020 16:06:09 +1100 Subject: [PATCH 1/5] add the ability to not fail on empty change set --- lib/stackup/change_set.rb | 9 +++++++-- lib/stackup/main_command.rb | 4 ++++ 2 files changed, 11 insertions(+), 2 deletions(-) diff --git a/lib/stackup/change_set.rb b/lib/stackup/change_set.rb index bd9bcb7..7515676 100644 --- a/lib/stackup/change_set.rb +++ b/lib/stackup/change_set.rb @@ -56,6 +56,7 @@ def create(options = {}) options[:change_set_name] = name options[:change_set_type] = stack.exists? ? "UPDATE" : "CREATE" force = options.delete(:force) + fail_change_set = options.delete(:fail_change_set) options[:template_body] = MultiJson.dump(options.delete(:template)) if options[:template] # optionally override template_body with the original template to preserve formatting (& comments in YAML) template_orig = options.delete(:template_orig) @@ -73,8 +74,12 @@ def create(options = {}) when /COMPLETE/ return current.status when "FAILED" - logger.error(current.status_reason) - raise StackUpdateError, "change-set creation failed" if status == "FAILED" + if fail_change_set + return current.status_reason if current.execution_status == "UNAVAILABLE" + else + logger.error(current.status_reason) + raise StackUpdateError, "change-set creation failed" if status == "FAILED" + end end sleep(wait_poll_interval) end diff --git a/lib/stackup/main_command.rb b/lib/stackup/main_command.rb index 35c9a1b..5140503 100644 --- a/lib/stackup/main_command.rb +++ b/lib/stackup/main_command.rb @@ -280,6 +280,9 @@ def pad(s, width) option ["--force"], :flag, "replace existing change-set of the same name" + option ["--no-fail-on-empty-change-set"], :flag, "dont fail on empty change-set", + :attribute_name => :fail_change_set + include HasParameters option "--tags", "FILE", "stack tags file", @@ -314,6 +317,7 @@ def execute options[:role_arn] = service_role_arn if service_role_arn options[:use_previous_template] = use_previous_template? options[:force] = force? + options[:fail_change_set] = fail_change_set? options[:capabilities] = capability_list options[:preserve] = preserve_template_formatting? report_change do From c09bf08f9cc6f33e310d417f830cff78223d49d6 Mon Sep 17 00:00:00 2001 From: Ash Rohde Date: Mon, 23 Nov 2020 10:59:48 +1100 Subject: [PATCH 2/5] alter if statement to check current.status_reason --- lib/stackup/change_set.rb | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/stackup/change_set.rb b/lib/stackup/change_set.rb index 7515676..5dcfab9 100644 --- a/lib/stackup/change_set.rb +++ b/lib/stackup/change_set.rb @@ -75,7 +75,7 @@ def create(options = {}) return current.status when "FAILED" if fail_change_set - return current.status_reason if current.execution_status == "UNAVAILABLE" + return current.status_reason if current.status_reason == "The submitted information didn't contain changes. Submit different information to create a change set." else logger.error(current.status_reason) raise StackUpdateError, "change-set creation failed" if status == "FAILED" From 8717bebc4ad1e12d2c23448c333c4ea8b6d4a2ae Mon Sep 17 00:00:00 2001 From: Angus McInnes Date: Wed, 25 Nov 2020 10:51:31 +1100 Subject: [PATCH 3/5] Clean up --no-fail-on-empty-change-set - Rename the variable to "allow_empty_change_set" to clarify its meaning - Add to README - Add tests - Rearrange "if" statement so it doesn't loop forever on unexpected failure message - Bump version and update changelog to prepare for release --- CHANGES.md | 4 ++++ README.md | 2 ++ lib/stackup/change_set.rb | 6 ++--- lib/stackup/main_command.rb | 6 ++--- lib/stackup/version.rb | 2 +- spec/stackup/main_command_spec.rb | 40 +++++++++++++++++++++++++++---- spec/stackup/stack_spec.rb | 32 +++++++++++++++++++++++++ 7 files changed, 80 insertions(+), 12 deletions(-) diff --git a/CHANGES.md b/CHANGES.md index 37d47f3..b9d7c15 100644 --- a/CHANGES.md +++ b/CHANGES.md @@ -1,5 +1,9 @@ # CHANGES +## 1.7.0 (2020-11-25) + +* Feature: --no-fail-on-empty-change-set + ## 1.6.0 (2020-11-19) * Feature: Support --service-role-arn on "change-set create" diff --git a/README.md b/README.md index 9805f3c..4cf7621 100644 --- a/README.md +++ b/README.md @@ -218,6 +218,8 @@ The change-set name defaults to "pending", but can be overridden using `--name`. The `change-set create` subcommand, like the `up` command, supports `--service-role-arn` to specify a service role. +It is impossible to create a change set with no changes. By default, stackup will only return successfully if a change set was actually created, and will otherwise fail. If the `--no-fail-on-empty-change-set` option is provided, stackup will return successfully if a change set was created _or_ if no change set was created because no changes were needed. + ## Programmatic usage Get a handle to a `Stack` object as follows: diff --git a/lib/stackup/change_set.rb b/lib/stackup/change_set.rb index 5dcfab9..62d6395 100644 --- a/lib/stackup/change_set.rb +++ b/lib/stackup/change_set.rb @@ -56,7 +56,7 @@ def create(options = {}) options[:change_set_name] = name options[:change_set_type] = stack.exists? ? "UPDATE" : "CREATE" force = options.delete(:force) - fail_change_set = options.delete(:fail_change_set) + allow_empty_change_set = options.delete(:allow_empty_change_set) options[:template_body] = MultiJson.dump(options.delete(:template)) if options[:template] # optionally override template_body with the original template to preserve formatting (& comments in YAML) template_orig = options.delete(:template_orig) @@ -74,8 +74,8 @@ def create(options = {}) when /COMPLETE/ return current.status when "FAILED" - if fail_change_set - return current.status_reason if current.status_reason == "The submitted information didn't contain changes. Submit different information to create a change set." + if allow_empty_change_set and current.status_reason == "The submitted information didn't contain changes. Submit different information to create a change set." + return current.status_reason else logger.error(current.status_reason) raise StackUpdateError, "change-set creation failed" if status == "FAILED" diff --git a/lib/stackup/main_command.rb b/lib/stackup/main_command.rb index 5140503..56c97ed 100644 --- a/lib/stackup/main_command.rb +++ b/lib/stackup/main_command.rb @@ -280,8 +280,8 @@ def pad(s, width) option ["--force"], :flag, "replace existing change-set of the same name" - option ["--no-fail-on-empty-change-set"], :flag, "dont fail on empty change-set", - :attribute_name => :fail_change_set + option ["--no-fail-on-empty-change-set"], :flag, "don't fail on empty change-set", + :attribute_name => :allow_empty_change_set include HasParameters @@ -317,7 +317,7 @@ def execute options[:role_arn] = service_role_arn if service_role_arn options[:use_previous_template] = use_previous_template? options[:force] = force? - options[:fail_change_set] = fail_change_set? + options[:allow_empty_change_set] = allow_empty_change_set? options[:capabilities] = capability_list options[:preserve] = preserve_template_formatting? report_change do diff --git a/lib/stackup/version.rb b/lib/stackup/version.rb index 8f66a8b..3ed1507 100644 --- a/lib/stackup/version.rb +++ b/lib/stackup/version.rb @@ -1,5 +1,5 @@ module Stackup - VERSION = "1.6.0".freeze + VERSION = "1.7.0".freeze end diff --git a/spec/stackup/main_command_spec.rb b/spec/stackup/main_command_spec.rb index 1dc6c08..c5df7d8 100644 --- a/spec/stackup/main_command_spec.rb +++ b/spec/stackup/main_command_spec.rb @@ -2,22 +2,52 @@ describe Stackup::MainCommand do context "change-set create --service-role-arn" - it "invokes stack.change_set.create with role arn passed through" do + before(:example) do mock_stackup = double() mock_stack = double() - mock_change_set = double() + @mock_change_set = double() allow_any_instance_of(Stackup::MainCommand).to receive(:Stackup).and_return(mock_stackup) allow(mock_stackup).to receive(:stack).and_return(mock_stack) - allow(mock_stack).to receive(:change_set).and_return(mock_change_set) + allow(mock_stack).to receive(:change_set).and_return(@mock_change_set) + end + it "invokes stack.change_set.create with role arn passed through" do expected_args = { role_arn: "arn:aws:iam::000000000000:role/example" } - expect(mock_change_set).to receive(:create).with(hash_including(expected_args)) + expect(@mock_change_set).to receive(:create).with(hash_including(expected_args)) Stackup::MainCommand.run("stackup", [ "STACK-NAME", "change-set", "create", "--template", "examples/template.yml", "--service-role-arn", "arn:aws:iam::000000000000:role/example"]) end -end + + context "change-set create" do + it "invokes stack.change_set.create with allow_empty_change_set nil" do + expected_args = { + allow_empty_change_set: nil + } + expect(@mock_change_set).to receive(:create).with(hash_including(expected_args)) + + Stackup::MainCommand.run("stackup", [ + "STACK-NAME", "change-set", "create", + "--template", "examples/template.yml"]) + end + end + + context "change-set create --no-fail-on-empty-change-set" do + it "invokes stack.change_set.create with allow_empty_change_set true" do + expected_args = { + allow_empty_change_set: true + } + expect(@mock_change_set).to receive(:create).with(hash_including(expected_args)) + + Stackup::MainCommand.run("stackup", [ + "STACK-NAME", "change-set", "create", + "--template", "examples/template.yml", + "--no-fail-on-empty-change-set"]) + end + end + + end diff --git a/spec/stackup/stack_spec.rb b/spec/stackup/stack_spec.rb index aef5258..8f491bc 100644 --- a/spec/stackup/stack_spec.rb +++ b/spec/stackup/stack_spec.rb @@ -429,6 +429,38 @@ def create_change_set end + context "when allow_empty_change_set is nil and there are no changes" do + it "raises an exception" do + cf_client.stub_responses(:describe_change_set, [{ + status: "FAILED", + status_reason: "The submitted information didn't contain changes. Submit different information to create a change set." + }]) + expect { create_change_set }.to raise_error(Stackup::StackUpdateError) + end + end + + context "when allow_empty_change_set is true and there are no changes" do + it "does not raise an exception" do + cf_client.stub_responses(:describe_change_set, [{ + status: "FAILED", + status_reason: "The submitted information didn't contain changes. Submit different information to create a change set." + }]) + options[:allow_empty_change_set] = true + expect { create_change_set }.not_to raise_error + end + end + + context "when allow_empty_change_set is true and there is some other failure" do + it "raises an exception" do + cf_client.stub_responses(:describe_change_set, [{ + status: "FAILED", + status_reason: "some other failure message" + }]) + options[:allow_empty_change_set] = true + expect { create_change_set }.to raise_error(Stackup::StackUpdateError) + end + end + end describe "#change_set#execute" do From 5ac0574bec553d9d4bbb9e479ca3a833c8f08bc9 Mon Sep 17 00:00:00 2001 From: Angus McInnes Date: Wed, 25 Nov 2020 13:56:52 +1100 Subject: [PATCH 4/5] Use the more idiomatic `let` in main_command_spec instead of instance variable --- spec/stackup/main_command_spec.rb | 11 ++++++----- 1 file changed, 6 insertions(+), 5 deletions(-) diff --git a/spec/stackup/main_command_spec.rb b/spec/stackup/main_command_spec.rb index c5df7d8..0179c97 100644 --- a/spec/stackup/main_command_spec.rb +++ b/spec/stackup/main_command_spec.rb @@ -2,20 +2,21 @@ describe Stackup::MainCommand do context "change-set create --service-role-arn" + let(:mock_change_set) { double() } + before(:example) do mock_stackup = double() mock_stack = double() - @mock_change_set = double() allow_any_instance_of(Stackup::MainCommand).to receive(:Stackup).and_return(mock_stackup) allow(mock_stackup).to receive(:stack).and_return(mock_stack) - allow(mock_stack).to receive(:change_set).and_return(@mock_change_set) + allow(mock_stack).to receive(:change_set).and_return(mock_change_set) end it "invokes stack.change_set.create with role arn passed through" do expected_args = { role_arn: "arn:aws:iam::000000000000:role/example" } - expect(@mock_change_set).to receive(:create).with(hash_including(expected_args)) + expect(mock_change_set).to receive(:create).with(hash_including(expected_args)) Stackup::MainCommand.run("stackup", [ "STACK-NAME", "change-set", "create", @@ -28,7 +29,7 @@ expected_args = { allow_empty_change_set: nil } - expect(@mock_change_set).to receive(:create).with(hash_including(expected_args)) + expect(mock_change_set).to receive(:create).with(hash_including(expected_args)) Stackup::MainCommand.run("stackup", [ "STACK-NAME", "change-set", "create", @@ -41,7 +42,7 @@ expected_args = { allow_empty_change_set: true } - expect(@mock_change_set).to receive(:create).with(hash_including(expected_args)) + expect(mock_change_set).to receive(:create).with(hash_including(expected_args)) Stackup::MainCommand.run("stackup", [ "STACK-NAME", "change-set", "create", From 0c64d9ee79fe372dcc86cb5a14373641ab2e76cb Mon Sep 17 00:00:00 2001 From: Angus McInnes Date: Wed, 25 Nov 2020 14:04:40 +1100 Subject: [PATCH 5/5] main_command_spec: fix structure and indentation --- spec/stackup/main_command_spec.rb | 70 ++++++++++++++++--------------- 1 file changed, 36 insertions(+), 34 deletions(-) diff --git a/spec/stackup/main_command_spec.rb b/spec/stackup/main_command_spec.rb index 0179c97..f05c118 100644 --- a/spec/stackup/main_command_spec.rb +++ b/spec/stackup/main_command_spec.rb @@ -1,17 +1,18 @@ require "stackup/main_command" describe Stackup::MainCommand do - context "change-set create --service-role-arn" - let(:mock_change_set) { double() } - - before(:example) do - mock_stackup = double() - mock_stack = double() - allow_any_instance_of(Stackup::MainCommand).to receive(:Stackup).and_return(mock_stackup) - allow(mock_stackup).to receive(:stack).and_return(mock_stack) - allow(mock_stack).to receive(:change_set).and_return(mock_change_set) - end + let(:mock_change_set) { double() } + + before(:example) do + mock_stackup = double() + mock_stack = double() + allow_any_instance_of(Stackup::MainCommand).to receive(:Stackup).and_return(mock_stackup) + allow(mock_stackup).to receive(:stack).and_return(mock_stack) + allow(mock_stack).to receive(:change_set).and_return(mock_change_set) + end + + context "change-set create --service-role-arn" do it "invokes stack.change_set.create with role arn passed through" do expected_args = { role_arn: "arn:aws:iam::000000000000:role/example" @@ -23,32 +24,33 @@ "--template", "examples/template.yml", "--service-role-arn", "arn:aws:iam::000000000000:role/example"]) end + end - context "change-set create" do - it "invokes stack.change_set.create with allow_empty_change_set nil" do - expected_args = { - allow_empty_change_set: nil - } - expect(mock_change_set).to receive(:create).with(hash_including(expected_args)) - - Stackup::MainCommand.run("stackup", [ - "STACK-NAME", "change-set", "create", - "--template", "examples/template.yml"]) - end - end + context "change-set create" do + it "invokes stack.change_set.create with allow_empty_change_set nil" do + expected_args = { + allow_empty_change_set: nil + } + expect(mock_change_set).to receive(:create).with(hash_including(expected_args)) - context "change-set create --no-fail-on-empty-change-set" do - it "invokes stack.change_set.create with allow_empty_change_set true" do - expected_args = { - allow_empty_change_set: true - } - expect(mock_change_set).to receive(:create).with(hash_including(expected_args)) - - Stackup::MainCommand.run("stackup", [ - "STACK-NAME", "change-set", "create", - "--template", "examples/template.yml", - "--no-fail-on-empty-change-set"]) - end + Stackup::MainCommand.run("stackup", [ + "STACK-NAME", "change-set", "create", + "--template", "examples/template.yml"]) end + end + + context "change-set create --no-fail-on-empty-change-set" do + it "invokes stack.change_set.create with allow_empty_change_set true" do + expected_args = { + allow_empty_change_set: true + } + expect(mock_change_set).to receive(:create).with(hash_including(expected_args)) + Stackup::MainCommand.run("stackup", [ + "STACK-NAME", "change-set", "create", + "--template", "examples/template.yml", + "--no-fail-on-empty-change-set"]) + end end + +end