Skip to content

Commit 9e6de46

Browse files
committed
feat: allow WIP pacts to be verified without causing the process to return an non zero exit code
1 parent 6519819 commit 9e6de46

22 files changed

+335
-54
lines changed

lib/pact/cli.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ class CLI < Thor
88
desc 'verify', "Verify a pact"
99
method_option :pact_helper, aliases: "-h", desc: "Pact helper file", :required => true
1010
method_option :pact_uri, aliases: "-p", desc: "Pact URI"
11+
method_option :wip, type: :boolean, default: false, desc: "If WIP, process will always exit with exit code 0", hide: true
1112
method_option :pact_broker_username, aliases: "-u", desc: "Pact broker user name"
1213
method_option :pact_broker_password, aliases: "-w", desc: "Pact broker password"
1314
method_option :backtrace, aliases: "-b", desc: "Show full backtrace", :default => false, :type => :boolean
@@ -33,6 +34,5 @@ def docs
3334
require 'pact/doc/generator'
3435
Pact::Doc::Generate.call(options[:pact_dir], options[:doc_dir], [Pact::Doc::Markdown::Generator])
3536
end
36-
3737
end
3838
end

lib/pact/cli/run_pact_verification.rb

+2-1
Original file line numberDiff line numberDiff line change
@@ -71,7 +71,8 @@ def pact_spec_options
7171
full_backtrace: options[:backtrace],
7272
criteria: SpecCriteria.call(options),
7373
format: options[:format],
74-
out: options[:out]
74+
out: options[:out],
75+
wip: options[:wip]
7576
}
7677
end
7778
end

lib/pact/hal/entity.rb

+2-2
Original file line numberDiff line numberDiff line change
@@ -48,8 +48,8 @@ def response
4848
@response
4949
end
5050

51-
def fetch(key)
52-
@links[key]
51+
def fetch(key, fallback_key = nil)
52+
@links[key] || (fallback_key && @links[fallback_key])
5353
end
5454

5555
def method_missing(method_name, *args, &block)

lib/pact/pact_broker/fetch_pacts.rb

+2-1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@ class FetchPacts
1111
LATEST_PROVIDER_TAG_RELATION = 'pb:latest-provider-pacts-with-tag'.freeze
1212
LATEST_PROVIDER_RELATION = 'pb:latest-provider-pacts'.freeze
1313
PACTS = 'pacts'.freeze
14+
PB_PACTS = 'pb:pacts'.freeze
1415
HREF = 'href'.freeze
1516

1617
def initialize(provider, tags, broker_base_url, http_client_options)
@@ -79,7 +80,7 @@ def get_latest_pacts_for_provider
7980
end
8081

8182
def get_pact_urls(link_by_provider)
82-
link_by_provider.fetch(PACTS).collect do |pact|
83+
link_by_provider.fetch(PB_PACTS, PACTS).collect do |pact|
8384
Pact::Provider::PactURI.new(pact[HREF], http_client_options)
8485
end
8586
end
+62
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,62 @@
1+
require 'pact/hal/entity'
2+
require 'pact/hal/http_client'
3+
require 'pact/provider/pact_uri'
4+
5+
module Pact
6+
module PactBroker
7+
class FetchWipPacts
8+
attr_reader :provider, :tags, :broker_base_url, :http_client_options, :http_client, :index_entity
9+
10+
WIP_PROVIDER_RELATION = 'pb:wip-provider-pacts'.freeze
11+
PACTS = 'pacts'.freeze
12+
PB_PACTS = 'pb:pacts'.freeze
13+
HREF = 'href'.freeze
14+
15+
def initialize(provider, broker_base_url, http_client_options)
16+
@provider = provider
17+
@http_client_options = http_client_options
18+
@broker_base_url = broker_base_url
19+
@http_client = Pact::Hal::HttpClient.new(http_client_options)
20+
end
21+
22+
def self.call(provider, broker_base_url, http_client_options)
23+
new(provider, broker_base_url, http_client_options).call
24+
end
25+
26+
def call
27+
log_message
28+
if get_index.success?
29+
get_wip_pacts_for_provider
30+
else
31+
raise Pact::Error.new("Error retrieving #{broker_base_url} status=#{index_entity.response.code} #{index_entity.response.raw_body}")
32+
end
33+
end
34+
35+
private
36+
37+
def get_index
38+
@index_entity = Pact::Hal::Link.new({ "href" => broker_base_url }, http_client).get
39+
end
40+
41+
def get_wip_pacts_for_provider
42+
link = index_entity._link(WIP_PROVIDER_RELATION)
43+
if link
44+
get_pact_urls(link.expand(provider: provider).get)
45+
else
46+
[]
47+
end
48+
end
49+
50+
def get_pact_urls(link_by_provider)
51+
link_by_provider.fetch(PB_PACTS, PACTS).collect do |pact|
52+
Pact::Provider::PactURI.new(pact[HREF], http_client_options)
53+
end
54+
end
55+
56+
def log_message
57+
message = "INFO: Fetching WIP pacts for #{provider} from #{broker_base_url}"
58+
Pact.configuration.output_stream.puts message
59+
end
60+
end
61+
end
62+
end

lib/pact/provider/pact_spec_runner.rb

+6-2
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@ def run
3636
ensure
3737
::RSpec.reset
3838
Pact.clear_provider_world
39+
Pact.clear_consumer_world
3940
end
4041
end
4142

@@ -90,7 +91,7 @@ def run_specs
9091
::RSpec::Core::CommandLine.new(NoConfigurationOptions.new)
9192
.run(::RSpec.configuration.output_stream, ::RSpec.configuration.error_stream)
9293
end
93-
exit_code
94+
options[:wip] ? 0 : exit_code
9495
end
9596

9697
def rspec_runner_options
@@ -118,7 +119,8 @@ def pact_jsons
118119
def initialize_specs
119120
pact_sources.each do | pact_source |
120121
options = {
121-
criteria: @options[:criteria]
122+
criteria: @options[:criteria],
123+
wip: @options[:wip]
122124
}
123125
honour_pactfile pact_source.uri, ordered_pact_json(pact_source.pact_json), options
124126
end
@@ -144,6 +146,8 @@ def configure_output
144146
end
145147

146148
::RSpec.configuration.full_backtrace = @options[:full_backtrace]
149+
150+
::RSpec.configuration.failure_color = :yellow if @options[:wip]
147151
end
148152

149153
def ordered_pact_json(pact_json)

lib/pact/provider/rspec.rb

+5-3
Original file line numberDiff line numberDiff line change
@@ -21,10 +21,11 @@ module ClassMethods
2121
include ::RSpec::Core::DSL
2222

2323
def honour_pactfile pact_uri, pact_json, options
24-
Pact.configuration.output_stream.puts "INFO: Reading pact at #{pact_uri}"
24+
pact_description = options[:wip] ? "WIP pact" : "pact"
25+
Pact.configuration.output_stream.puts "INFO: Reading #{pact_description} at #{pact_uri}"
2526
Pact.configuration.output_stream.puts "INFO: Filtering interactions by: #{options[:criteria]}" if options[:criteria] && options[:criteria].any?
2627
consumer_contract = Pact::ConsumerContract.from_json(pact_json)
27-
::RSpec.describe "Verifying a pact between #{consumer_contract.consumer.name} and #{consumer_contract.provider.name}", pactfile_uri: pact_uri do
28+
::RSpec.describe "Verifying a #{pact_description} between #{consumer_contract.consumer.name} and #{consumer_contract.provider.name}", pactfile_uri: pact_uri do
2829
honour_consumer_contract consumer_contract, options.merge(pact_json: pact_json, pact_uri: pact_uri)
2930
end
3031
end
@@ -72,7 +73,8 @@ def describe_interaction interaction, options
7273
pact: :verify,
7374
pact_interaction: interaction,
7475
pact_interaction_example_description: interaction_description_for_rerun_command(interaction),
75-
pact_uri: options[:pact_uri]
76+
pact_uri: options[:pact_uri],
77+
pact_wip: options[:wip]
7678
}
7779

7880
describe description_for(interaction), metadata do

lib/pact/provider/rspec/formatter_rspec_3.rb

+18-2
Original file line numberDiff line numberDiff line change
@@ -42,9 +42,21 @@ def failed_interactions_count(summary)
4242
summary.failed_examples.collect{ |e| e.metadata[:pact_interaction_example_description] }.uniq.size
4343
end
4444

45+
def wip?(summary)
46+
summary.failed_examples.any?{ |e| e.metadata[:pact_wip] }
47+
end
48+
49+
def failure_title summary
50+
if wip?(summary)
51+
"#{failed_interactions_count(summary)} pending"
52+
else
53+
::RSpec::Core::Formatters::Helpers.pluralize(failed_interactions_count(summary), "failure")
54+
end
55+
end
56+
4557
def totals_line summary
4658
line = ::RSpec::Core::Formatters::Helpers.pluralize(interactions_count(summary), "interaction")
47-
line << ", " << ::RSpec::Core::Formatters::Helpers.pluralize(failed_interactions_count(summary), "failure")
59+
line << ", " << failure_title(summary)
4860
line
4961
end
5062

@@ -57,7 +69,11 @@ def color_for_summary summary
5769
end
5870

5971
def print_rerun_commands summary
60-
output.puts("\nFailed interactions:\n\n")
72+
if wip?(summary)
73+
output.puts("\nPending interactions: (Failures listed here are expected and do not affect your suite's status)\n\n")
74+
else
75+
output.puts("\nFailed interactions:\n\n")
76+
end
6177
interaction_rerun_commands(summary).each do | message |
6278
output.puts(message)
6379
end

lib/pact/tasks/task_helper.rb

+4-3
Original file line numberDiff line numberDiff line change
@@ -10,16 +10,16 @@ module TaskHelper
1010

1111
extend self
1212

13-
def execute_pact_verify pact_uri = nil, pact_helper = nil, rspec_opts = nil
14-
execute_cmd verify_command(pact_helper || Pact::Provider::PactHelperLocater.pact_helper_path, pact_uri, rspec_opts)
13+
def execute_pact_verify pact_uri = nil, pact_helper = nil, rspec_opts = nil, verification_opts = {}
14+
execute_cmd verify_command(pact_helper || Pact::Provider::PactHelperLocater.pact_helper_path, pact_uri, rspec_opts, verification_opts)
1515
end
1616

1717
def handle_verification_failure
1818
exit_status = yield
1919
abort if exit_status != 0
2020
end
2121

22-
def verify_command pact_helper, pact_uri, rspec_opts
22+
def verify_command pact_helper, pact_uri, rspec_opts, verification_opts
2323
command_parts = []
2424
# Clear SPEC_OPTS, otherwise we can get extra formatters, creating duplicate output eg. CI Reporting.
2525
# Allow deliberate configuration using rspec_opts in VerificationTask.
@@ -28,6 +28,7 @@ def verify_command pact_helper, pact_uri, rspec_opts
2828
command_parts << "-S pact verify"
2929
command_parts << "--pact-helper" << Shellwords.escape(pact_helper.end_with?(".rb") ? pact_helper : pact_helper + ".rb")
3030
(command_parts << "--pact-uri" << pact_uri) if pact_uri
31+
command_parts << "--wip" if verification_opts[:wip]
3132
command_parts << "--pact-broker-username" << ENV['PACT_BROKER_USERNAME'] if ENV['PACT_BROKER_USERNAME']
3233
command_parts << "--pact-broker-password" << ENV['PACT_BROKER_PASSWORD'] if ENV['PACT_BROKER_PASSWORD']
3334
command_parts << "--backtrace" if ENV['BACKTRACE'] == 'true'

lib/pact/tasks/verification_task.rb

+3-1
Original file line numberDiff line numberDiff line change
@@ -30,9 +30,11 @@ class VerificationTask < ::Rake::TaskLib
3030

3131
attr_reader :pact_spec_configs
3232
attr_accessor :rspec_opts
33+
attr_accessor :wip
3334

3435
def initialize(name)
3536
@rspec_opts = nil
37+
@wip = false
3638
@pact_spec_configs = []
3739
@name = name
3840
yield self
@@ -74,7 +76,7 @@ def rake_task
7476
require 'pact/tasks/task_helper'
7577

7678
exit_statuses = pact_spec_configs.collect do | config |
77-
Pact::TaskHelper.execute_pact_verify config[:uri], config[:pact_helper], rspec_opts
79+
Pact::TaskHelper.execute_pact_verify config[:uri], config[:pact_helper], rspec_opts, { wip: wip }
7880
end
7981

8082
Pact::TaskHelper.handle_verification_failure do

spec/lib/pact/hal/entity_spec.rb

+17-4
Original file line numberDiff line numberDiff line change
@@ -75,18 +75,31 @@ module Hal
7575
end
7676

7777
describe 'fetch' do
78-
context 'when the key exist' do
78+
context 'when the key exists' do
7979
it 'returns fetched value' do
80-
expect(subject.fetch('pb:provider')).to be do
81-
{href: 'http://provider'}
82-
end
80+
expect(subject.fetch('pb:provider')).to eq("href" => 'http://provider')
8381
end
8482
end
83+
8584
context "when the key doesn't not exist" do
8685
it 'returns nil' do
8786
expect(subject.fetch('i-dont-exist')).to be nil
8887
end
8988
end
89+
90+
context "when a fallback key is provided" do
91+
context "when the fallback value exists" do
92+
it "returns the fallback value" do
93+
expect(subject.fetch('i-dont-exist', 'pb:provider')).to eq("href" => 'http://provider')
94+
end
95+
end
96+
97+
context "when the fallback value does not exist" do
98+
it "returns nil" do
99+
expect(subject.fetch('i-dont-exist', 'i-also-dont-exist')).to be nil
100+
end
101+
end
102+
end
90103
end
91104
end
92105
end

spec/lib/pact/pact_broker/fetch_pacts_spec.rb

+5-4
Original file line numberDiff line numberDiff line change
@@ -5,15 +5,16 @@ module PactBroker
55
describe FetchPacts do
66

77
describe "call" do
8+
before do
9+
allow(Pact.configuration).to receive(:output_stream).and_return(double('output stream').as_null_object)
10+
stub_request(:get, "http://broker.org/").to_return(status: 500, body: "foo", headers: {})
11+
end
12+
813
let(:provider) { "Foo"}
914
let(:tags) { ["master", "prod"] }
1015
let(:broker_base_url) { "http://broker.org" }
1116
let(:http_client_options) { {} }
1217

13-
before do
14-
stub_request(:get, "http://broker.org/").to_return(status: 500, body: "foo", headers: {})
15-
end
16-
1718
subject { FetchPacts.call(provider, tags, broker_base_url, http_client_options)}
1819

1920
let(:subject_with_rescue) do
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,54 @@
1+
require 'pact/pact_broker/fetch_wip_pacts'
2+
3+
module Pact
4+
module PactBroker
5+
describe FetchWipPacts do
6+
describe "call" do
7+
before do
8+
allow(Pact.configuration).to receive(:output_stream).and_return(double('output stream').as_null_object)
9+
end
10+
11+
let(:provider) { "Foo"}
12+
let(:broker_base_url) { "http://broker.org" }
13+
let(:http_client_options) { {} }
14+
subject { FetchWipPacts.call(provider, broker_base_url, http_client_options)}
15+
16+
context "when there is an error retrieving the index resource" do
17+
before do
18+
stub_request(:get, "http://broker.org/").to_return(status: 500, body: "foo", headers: {})
19+
end
20+
21+
let(:subject_with_rescue) do
22+
begin
23+
subject
24+
rescue Pact::Error
25+
# can't be bothered stubbing out everything to make the rest of the code execute nicely
26+
# when all we care about is the message
27+
end
28+
end
29+
30+
it "raises a Pact::Error" do
31+
expect { subject }.to raise_error Pact::Error, /500.*foo/
32+
end
33+
end
34+
35+
context "when the pb:wip-provider-pacts relation does not exist" do
36+
before do
37+
stub_request(:get, "http://broker.org/").to_return(status: 200, body: response_body, headers: response_headers)
38+
end
39+
40+
let(:response_headers) { { "Content-Type" => "application/hal+json" } }
41+
let(:response_body) do
42+
{
43+
_links: {}
44+
}.to_json
45+
end
46+
47+
it "returns an empty list" do
48+
expect(subject).to eq []
49+
end
50+
end
51+
end
52+
end
53+
end
54+
end

0 commit comments

Comments
 (0)