Skip to content

Commit a586d80

Browse files
committed
feat: allow individual interactions to be re-run by setting PACT_BROKER_INTERACTION_ID
1 parent 32af081 commit a586d80

File tree

6 files changed

+90
-24
lines changed

6 files changed

+90
-24
lines changed

lib/pact/cli.rb

+2
Original file line numberDiff line numberDiff line change
@@ -18,6 +18,8 @@ class CLI < Thor
1818
default: Pact.configuration.interactions_replay_order
1919
method_option :description, aliases: "-d", desc: "Interaction description filter"
2020
method_option :provider_state, aliases: "-s", desc: "Provider state filter"
21+
method_option :interaction_index, type: :numeric, desc: "Index filter"
22+
method_option :pact_broker_interaction_id, desc: "Pact Broker interaction ID filter"
2123
method_option :format, aliases: "-f", banner: "FORMATTER", desc: "RSpec formatter. Defaults to custom Pact formatter. [j]son may also be used."
2224
method_option :out, aliases: "-o", banner: "FILE", desc: "Write output to a file instead of $stdout."
2325

lib/pact/cli/spec_criteria.rb

+3
Original file line numberDiff line numberDiff line change
@@ -6,8 +6,11 @@ def self.call options
66
criteria = {}
77

88
criteria[:description] = Regexp.new(options[:description]) if options[:description]
9+
criteria[:_id] = options[:pact_broker_interaction_id] if options[:pact_broker_interaction_id]
10+
criteria[:index] = options[:interaction_index] if options[:interaction_index]
911

1012
provider_state = options[:provider_state]
13+
1114
if provider_state
1215
if provider_state.length == 0
1316
criteria[:provider_state] = nil #Allow PACT_PROVIDER_STATE="" to mean no provider state

lib/pact/provider/rspec/formatter_rspec_3.rb

+20-5
Original file line numberDiff line numberDiff line change
@@ -93,17 +93,32 @@ def interaction_rerun_commands summary
9393

9494
def interaction_rerun_command_for example
9595
example_description = example.metadata[:pact_interaction_example_description]
96-
if ENV['PACT_INTERACTION_RERUN_COMMAND']
96+
97+
_id = example.metadata[:pact_interaction]._id
98+
index = example.metadata[:pact_interaction].index
99+
provider_state = example.metadata[:pact_interaction].provider_state
100+
description = example.metadata[:pact_interaction].description
101+
pactfile_uri = example.metadata[:pactfile_uri]
102+
103+
if _id && ENV['PACT_INTERACTION_RERUN_COMMAND_FOR_BROKER']
104+
cmd = String.new(ENV['PACT_INTERACTION_RERUN_COMMAND_FOR_BROKER'])
105+
cmd.gsub!("<PACT_URI>", example.metadata[:pactfile_uri].to_s)
106+
cmd.gsub!("<PACT_BROKER_INTERACTION_ID>", "#{_id}")
107+
colorizer.wrap("#{cmd} ", ::RSpec.configuration.failure_color) + colorizer.wrap("# #{example_description}", ::RSpec.configuration.detail_color)
108+
elsif ENV['PACT_INTERACTION_RERUN_COMMAND']
97109
cmd = String.new(ENV['PACT_INTERACTION_RERUN_COMMAND'])
98-
provider_state = example.metadata[:pact_interaction].provider_state
99-
description = example.metadata[:pact_interaction].description
100-
pactfile_uri = example.metadata[:pactfile_uri]
101110
cmd.gsub!("<PACT_URI>", pactfile_uri.to_s)
102111
cmd.gsub!("<PACT_DESCRIPTION>", description)
103112
cmd.gsub!("<PACT_PROVIDER_STATE>", "#{provider_state}")
113+
cmd.gsub!("<PACT_INTERACTION_INDEX>", "#{index}")
104114
colorizer.wrap("#{cmd} ", ::RSpec.configuration.failure_color) + colorizer.wrap("# #{example_description}", ::RSpec.configuration.detail_color)
105115
else
106-
colorizer.wrap("* #{example_description}", ::RSpec.configuration.failure_color)
116+
message = if _id
117+
"* #{example_description} (to re-run just this interaction, set environment variable PACT_BROKER_INTERACTION_ID=\"#{_id}\")"
118+
else
119+
"* #{example_description} (to re-run just this interaction, set environment variables PACT_DESCRIPTION=\"#{description}\" PACT_PROVIDER_STATE=\"#{provider_state}\")"
120+
end
121+
colorizer.wrap(message, ::RSpec.configuration.failure_color)
107122
end
108123
end
109124

lib/pact/tasks/task_helper.rb

+6-1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ module Pact
77
module TaskHelper
88

99
PACT_INTERACTION_RERUN_COMMAND = "bundle exec rake pact:verify:at[<PACT_URI>] PACT_DESCRIPTION=\"<PACT_DESCRIPTION>\" PACT_PROVIDER_STATE=\"<PACT_PROVIDER_STATE>\""
10+
PACT_INTERACTION_RERUN_COMMAND_FOR_BROKER = "bundle exec rake pact:verify:at[<PACT_URI>] PACT_BROKER_INTERACTION_ID=\"<PACT_BROKER_INTERACTION_ID>\""
1011

1112
extend self
1213

@@ -34,14 +35,18 @@ def verify_command pact_helper, pact_uri, rspec_opts, verification_opts
3435
command_parts << "--backtrace" if ENV['BACKTRACE'] == 'true'
3536
command_parts << "--description #{Shellwords.escape(ENV['PACT_DESCRIPTION'])}" if ENV['PACT_DESCRIPTION']
3637
command_parts << "--provider-state #{Shellwords.escape(ENV['PACT_PROVIDER_STATE'])}" if ENV['PACT_PROVIDER_STATE']
38+
command_parts << "--pact-broker-interaction-id #{Shellwords.escape(ENV['PACT_BROKER_INTERACTION_ID'])}" if ENV['PACT_BROKER_INTERACTION_ID']
39+
command_parts << "--interaction-index #{Shellwords.escape(ENV['PACT_INTERACTION_INDEX'])}" if ENV['PACT_INTERACTION_INDEX']
3740
command_parts.flatten.join(" ")
3841
end
3942

4043
def execute_cmd command
4144
Pact.configuration.output_stream.puts command
4245
temporarily_set_env_var 'PACT_EXECUTING_LANGUAGE', 'ruby' do
4346
temporarily_set_env_var 'PACT_INTERACTION_RERUN_COMMAND', PACT_INTERACTION_RERUN_COMMAND do
44-
exit_status = system(command) ? 0 : 1
47+
temporarily_set_env_var 'PACT_INTERACTION_RERUN_COMMAND_FOR_BROKER', PACT_INTERACTION_RERUN_COMMAND_FOR_BROKER do
48+
exit_status = system(command) ? 0 : 1
49+
end
4550
end
4651
end
4752
end

spec/lib/pact/cli/spec_criteria_spec.rb

+18-12
Original file line numberDiff line numberDiff line change
@@ -3,34 +3,41 @@
33
module Pact
44
module Cli
55
describe SpecCriteria do
6-
76
describe "#spec_criteria" do
87

98
let(:env_description) { "pact description set in ENV"}
109
let(:env_provider_state) { "provider state set in ENV"}
11-
let(:env_criteria){ {:description=>/#{env_description}/, :provider_state=>/#{env_provider_state}/} }
10+
let(:env_pact_broker_interaction_id) { "interaction id set in ENV" }
11+
let(:interaction_index) { 2 }
12+
let(:env_criteria) do
13+
{
14+
:description=>/#{env_description}/,
15+
:provider_state=>/#{env_provider_state}/,
16+
:_id => env_pact_broker_interaction_id,
17+
:index => interaction_index
18+
}
19+
end
1220

1321
let(:defaults) { {:description => default_description, :provider_state => default_provider_state} }
1422

1523
let(:subject) { Pact::App.new }
1624

1725
context "when options are defined" do
18-
before do
19-
20-
allow(ENV).to receive(:[])
21-
allow(ENV).to receive(:[]).with("PACT_DESCRIPTION").and_return(env_description)
22-
allow(ENV).to receive(:[]).with("PACT_PROVIDER_STATE").and_return(env_provider_state)
26+
let(:options) do
27+
{
28+
description: env_description,
29+
provider_state: env_provider_state,
30+
pact_broker_interaction_id: env_pact_broker_interaction_id,
31+
interaction_index: interaction_index
32+
}
2333
end
2434

25-
let(:options) { {description: env_description, provider_state: env_provider_state} }
26-
2735
it "returns the env vars as regexes" do
2836
expect(Pact::Cli::SpecCriteria.call(options)).to eq(env_criteria)
2937
end
3038
end
3139

3240
context "when ENV variables are not defined" do
33-
3441
let(:options) { {} }
3542

3643
it "returns an empty hash" do
@@ -39,8 +46,7 @@ module Cli
3946
end
4047

4148
context "when provider state is an empty string" do
42-
43-
let(:options) { {provider_state: ''} }
49+
let(:options) { { provider_state: '' } }
4450

4551
it "returns a nil provider state so that it matches a nil provider state on the interaction" do
4652
expect(Pact::Cli::SpecCriteria.call(options)[:provider_state]).to be_nil

spec/lib/pact/provider/rspec/formatter_rspec_3_spec.rb

+41-6
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,8 @@ module Provider
1010
module RSpec
1111
describe Formatter do
1212

13-
let(:interaction) { InteractionFactory.create 'provider_state' => 'a state', 'description' => 'a description'}
13+
let(:interaction) { InteractionFactory.create 'provider_state' => 'a state', 'description' => 'a description', '_id' => id, 'index' => 2 }
14+
let(:id) { nil }
1415
let(:pactfile_uri) { 'pact_file_uri' }
1516
let(:description) { 'an interaction' }
1617
let(:pact_json) { {some: 'pact json'}.to_json }
@@ -29,11 +30,13 @@ module RSpec
2930
let(:failed_examples) { [example, example] }
3031
let(:examples) { [example, example, example_2]}
3132
let(:output) { StringIO.new }
32-
let(:rerun_command) { "rake pact:verify:at[pact_file_uri] PACT_DESCRIPTION=\"a description\" PACT_PROVIDER_STATE=\"a state\" # an interaction" }
33+
let(:rerun_command) { 'PACT_DESCRIPTION="a description" PACT_PROVIDER_STATE="a state" # an interaction' }
34+
let(:broker_rerun_command) { "rake pact:verify:at[pact_file_uri] PACT_BROKER_INTERACTION_ID=\"1234\" # an interaction" }
3335
let(:missing_provider_states) { 'missing_provider_states'}
3436
let(:summary) { double("summary", failure_count: 1, failed_examples: failed_examples, examples: examples)}
3537
let(:pact_executing_language) { 'ruby' }
3638
let(:pact_interaction_rerun_command) { Pact::TaskHelper::PACT_INTERACTION_RERUN_COMMAND }
39+
let(:pact_interaction_rerun_command_for_broker) { Pact::TaskHelper::PACT_INTERACTION_RERUN_COMMAND_FOR_BROKER }
3740
let(:ignore_failures) { nil }
3841

3942
subject { Formatter.new output }
@@ -43,6 +46,7 @@ module RSpec
4346
before do
4447
allow(ENV).to receive(:[]).with('PACT_INTERACTION_RERUN_COMMAND').and_return(pact_interaction_rerun_command)
4548
allow(ENV).to receive(:[]).with('PACT_EXECUTING_LANGUAGE').and_return(pact_executing_language)
49+
allow(ENV).to receive(:[]).with('PACT_INTERACTION_RERUN_COMMAND_FOR_BROKER').and_return(pact_interaction_rerun_command_for_broker)
4650
allow(PrintMissingProviderStates).to receive(:call)
4751
allow(Pact::Provider::Help::PromptText).to receive(:call).and_return("some help")
4852
allow(subject).to receive(:failed_examples).and_return(failed_examples)
@@ -69,15 +73,46 @@ module RSpec
6973
end
7074
end
7175

72-
context "when PACT_INTERACTION_RERUN_COMMAND is not set" do
76+
context "when PACT_INTERACTION_RERUN_COMMAND_FOR_BROKER is set" do
77+
context "when the _id is populated" do
78+
let(:id) { "1234" }
79+
80+
it "prints a list of rerun commands" do
81+
expect(output_result).to include(broker_rerun_command)
82+
end
83+
84+
it "only prints unique commands" do
85+
expect(output_result.scan(broker_rerun_command).size).to eq 1
86+
end
87+
end
88+
89+
context "when the _id is not populated" do
90+
it "prints a list of rerun commands using the provider state and description" do
91+
expect(output_result).to include(rerun_command)
92+
end
93+
end
94+
end
95+
96+
context "when PACT_INTERACTION_RERUN_COMMAND and PACT_INTERACTION_RERUN_COMMAND_FOR_BROKER are not set" do
7397
let(:pact_interaction_rerun_command) { nil }
98+
let(:pact_interaction_rerun_command_for_broker) { nil }
99+
100+
context "when the _id is populated" do
101+
let(:id) { "1234" }
102+
103+
it "prints a list of failed interactions" do
104+
expect(output_result).to include('* an interaction (to re-run just this interaction, set environment variable PACT_BROKER_INTERACTION_ID="1234")')
105+
end
106+
end
74107

75-
it "prints a list of failed interactions" do
76-
expect(output_result).to include("* #{description}\n")
108+
context "when the _id is not populated" do
109+
it "prints a list of failed interactions" do
110+
expect(output_result).to include('* an interaction (to re-run just this interaction, set environment variables PACT_DESCRIPTION="a description" PACT_PROVIDER_STATE="a state")')
111+
end
77112
end
78113

79114
it "only prints unique commands" do
80-
expect(output_result.scan("* #{description}\n").size).to eq 1
115+
expect(output_result.scan("* #{description}").size).to eq 1
81116
end
82117
end
83118

0 commit comments

Comments
 (0)