Skip to content

Commit 0c56752

Browse files
committed
feat(publish test results): only publish verification results when all interactions have been run
1 parent e0dad27 commit 0c56752

File tree

9 files changed

+194
-37
lines changed

9 files changed

+194
-37
lines changed

lib/pact/provider/rspec.rb

+7-7
Original file line numberDiff line numberDiff line change
@@ -61,24 +61,24 @@ def describe_interaction_with_provider_state interaction, options
6161

6262
def describe_interaction interaction, options
6363

64-
# pact_uri, pact_provider_state and pact_description are used by
64+
# pact_uri and pact_interaction are used by
6565
# Pact::Provider::RSpec::PactBrokerFormatter
66+
67+
# pact_interaction_example_description is used by
68+
# Pact::Provider::RSpec::Formatter and Pact::Provider::RSpec::Formatter2
69+
70+
# pact: verify is used to allow RSpec before and after hooks.
6671
metadata = {
6772
pact: :verify,
6873
pact_interaction: interaction,
6974
pact_interaction_example_description: interaction_description_for_rerun_command(interaction),
70-
pact_json: options[:pact_json],
71-
pact_uri: options[:pact_uri],
72-
pact_provider_state: interaction.provider_state,
73-
pact_description: interaction.description
75+
pact_uri: options[:pact_uri]
7476
}
7577

7678
describe description_for(interaction), metadata do
7779

78-
7980
describe "with #{interaction.request.method_and_path}" do
8081

81-
8282
interaction_context = InteractionContext.new
8383

8484
before do | example |

lib/pact/provider/rspec/pact_broker_formatter.rb

+5-4
Original file line numberDiff line numberDiff line change
@@ -53,11 +53,12 @@ def close(_notification)
5353
def format_example(example)
5454
{
5555
exampleDescription: example.description,
56-
fullDescription: example.full_description,
56+
exampleFullDescription: example.full_description,
5757
status: example.execution_result.status.to_s,
58-
interactionProviderState: example.metadata[:pact_provider_state],
59-
interactionDescription: example.metadata[:pact_description],
60-
pact_uri: example.metadata[:pact_uri]
58+
interactionProviderState: example.metadata[:pact_interaction].provider_state,
59+
interactionDescription: example.metadata[:pact_interaction].description,
60+
pact_uri: example.metadata[:pact_uri],
61+
pact_interaction: example.metadata[:pact_interaction]
6162
}
6263
end
6364
end

lib/pact/provider/verification_results/create.rb

+19-6
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ def initialize pact_source, test_results_hash
1414
end
1515

1616
def call
17-
VerificationResult.new(!any_failures?, Pact.configuration.provider.application_version, test_results_hash_for_pact_uri)
17+
VerificationResult.new(publishable?, !any_failures?, Pact.configuration.provider.application_version, test_results_hash_for_pact_uri)
1818
end
1919

2020
private
@@ -27,19 +27,32 @@ def any_failures?
2727
count_failures_for_pact_uri > 0
2828
end
2929

30+
def publishable?
31+
executed_interactions_count == all_interactions_count && all_interactions_count > 0
32+
end
33+
3034
def examples_for_pact_uri
31-
@examples_for_pact_uri ||= test_results_hash[:examples]
32-
.select{ |e| e[:pact_uri] == pact_uri }
33-
.collect{ |e| clean_example(e) }
35+
@examples_for_pact_uri ||= test_results_hash[:examples].select{ |e| e[:pact_uri] == pact_uri }
3436
end
3537

3638
def count_failures_for_pact_uri
3739
examples_for_pact_uri.count{ |e| e[:status] != 'passed' }
3840
end
3941

42+
def executed_interactions_count
43+
examples_for_pact_uri
44+
.collect { |e| e[:pact_interaction].object_id }
45+
.uniq
46+
.count
47+
end
48+
49+
def all_interactions_count
50+
pact_source.pact_hash['interactions'].count
51+
end
52+
4053
def test_results_hash_for_pact_uri
4154
{
42-
examples: examples_for_pact_uri,
55+
examples: examples_for_pact_uri.collect{ |e| clean_example(e) },
4356
summary: {
4457
exampleCount: examples_for_pact_uri.size,
4558
failureCount: count_failures_for_pact_uri
@@ -48,7 +61,7 @@ def test_results_hash_for_pact_uri
4861
end
4962

5063
def clean_example(example)
51-
example.reject{ |k, v| k == :pact_uri }
64+
example.reject{ |k, v| k == :pact_uri || k == :pact_interaction }
5265
end
5366

5467
attr_reader :pact_source, :test_results_hash

lib/pact/provider/verification_results/publish.rb

+29-15
Original file line numberDiff line numberDiff line change
@@ -22,25 +22,29 @@ def initialize pact_source, verification_result
2222
end
2323

2424
def call
25-
if Pact.configuration.provider.publish_verification_results?
26-
if Pact.configuration.provider.tags.any?
27-
if tag_url('')
28-
tag_versions
29-
else
30-
Pact.configuration.error_stream.puts "WARN: Cannot tag provider version as there is no link named pb:tag-version in the pact JSON."
31-
end
32-
end
33-
34-
if publication_url
35-
publish
36-
else
37-
Pact.configuration.error_stream.puts "WARN: Cannot publish verification for #{consumer_name} as there is no link named pb:publish-verification-results in the pact JSON. If you are using a pact broker, please upgrade to version 2.0.0 or later."
38-
end
25+
if can_publish_verification_results?
26+
tag_versions_if_configured
27+
publish_verification_results
3928
end
4029
end
4130

4231
private
4332

33+
def can_publish_verification_results?
34+
return false unless Pact.configuration.provider.publish_verification_results?
35+
36+
if publication_url.nil?
37+
Pact.configuration.error_stream.puts "WARN: Cannot publish verification for #{consumer_name} as there is no link named pb:publish-verification-results in the pact JSON. If you are using a pact broker, please upgrade to version 2.0.0 or later."
38+
return false
39+
end
40+
41+
if !verification_result.publishable?
42+
Pact.configuration.error_stream.puts "WARN: Cannot publish verification for #{consumer_name} as not all interactions have been verified. Re-run the verification without the filter parameters or environment variables to publish the verification."
43+
return false
44+
end
45+
true
46+
end
47+
4448
def publication_url
4549
@publication_url ||= pact_source.pact_hash.fetch('_links', {}).fetch('pb:publish-verification-results', {})['href']
4650
end
@@ -50,6 +54,16 @@ def tag_url tag
5054
href ? href.gsub('{tag}', tag) : nil
5155
end
5256

57+
def tag_versions_if_configured
58+
if Pact.configuration.provider.tags.any?
59+
if tag_url('')
60+
tag_versions
61+
else
62+
Pact.configuration.error_stream.puts "WARN: Cannot tag provider version as there is no link named pb:tag-version in the pact JSON."
63+
end
64+
end
65+
end
66+
5367
def tag_versions
5468
Pact.configuration.provider.tags.each do | tag |
5569
uri = URI(tag_url(tag))
@@ -71,7 +85,7 @@ def tag_versions
7185
end
7286
end
7387

74-
def publish
88+
def publish_verification_results
7589
uri = URI(publication_url)
7690
request = build_request('Post', uri, verification_result.to_json, "Publishing verification result #{verification_result} to")
7791
response = nil

lib/pact/provider/verification_results/publish_all.rb

+2-3
Original file line numberDiff line numberDiff line change
@@ -15,10 +15,9 @@ def initialize pact_sources, test_results_hash
1515
@test_results_hash = test_results_hash
1616
end
1717

18-
# TODO do not publish unless all interactions have been run
1918
def call
20-
verification_results.collect do | pair |
21-
Publish.call(pair.first, pair.last)
19+
verification_results.collect do | (pact_source, verification_result) |
20+
Publish.call(pact_source, verification_result)
2221
end
2322
end
2423

lib/pact/provider/verification_results/verification_result.rb

+6-1
Original file line numberDiff line numberDiff line change
@@ -5,12 +5,17 @@ module Provider
55
module VerificationResults
66
class VerificationResult
77

8-
def initialize success, provider_application_version, test_results_hash
8+
def initialize publishable, success, provider_application_version, test_results_hash
9+
@publishable = publishable
910
@success = success
1011
@provider_application_version = provider_application_version
1112
@test_results_hash = test_results_hash
1213
end
1314

15+
def publishable?
16+
@publishable
17+
end
18+
1419
def provider_application_version_set?
1520
!!provider_application_version
1621
end

spec/integration/publish_verification_spec.rb

+1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@
2525

2626
let(:pact_hash) do
2727
{
28+
'interactions' => [{}],
2829
'_links' => {
2930
'pb:publish-verification-results' => {
3031
'href' => 'http://publish/'
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,113 @@
1+
require 'pact/provider/verification_results/create'
2+
3+
module Pact
4+
module Provider
5+
module VerificationResults
6+
describe Create do
7+
before do
8+
allow(Pact.configuration).to receive(:provider).and_return(provider_configuration)
9+
allow(VerificationResult).to receive(:new).and_return(verification_result)
10+
end
11+
12+
let(:verification_result) { double('VerificationResult') }
13+
let(:provider_configuration) do
14+
double('provider_configuration', application_version: '1.2.3')
15+
end
16+
let(:pact_source_1) do
17+
instance_double('Pact::Provider::PactSource', uri: pact_uri_1, pact_hash: pact_hash_1)
18+
end
19+
let(:pact_uri_1) { instance_double('Pact::Provider::PactURI', uri: URI('foo')) }
20+
let(:pact_uri_2) { instance_double('Pact::Provider::PactURI', uri: URI('bar')) }
21+
let(:example_1) do
22+
{
23+
pact_uri: pact_uri_1,
24+
pact_interaction: double('interaction'),
25+
status: 'passed'
26+
}
27+
end
28+
let(:example_2) do
29+
{
30+
pact_uri: pact_uri_2,
31+
pact_interaction: double('interaction'),
32+
status: 'passed'
33+
}
34+
end
35+
let(:test_results_hash) do
36+
{
37+
examples: [example_1, example_2]
38+
}
39+
end
40+
let(:pact_hash_1) do
41+
{
42+
'interactions' => [{}]
43+
}
44+
end
45+
46+
subject { Create.call(pact_source_1, test_results_hash) }
47+
48+
it "returns a verification result" do
49+
expect(subject).to eq verification_result
50+
end
51+
52+
it "creates a VerificationResult with the relevant test results" do
53+
expected_test_results_hash = {
54+
examples: [{ status: "passed" }],
55+
summary: { exampleCount: 1, failureCount: 0}
56+
}
57+
expect(VerificationResult).to receive(:new).with(anything, anything, anything, expected_test_results_hash)
58+
subject
59+
end
60+
61+
it "creates a VerificationResult with the provider application version" do
62+
expect(provider_configuration).to receive(:application_version)
63+
expect(VerificationResult).to receive(:new).with(anything, anything, '1.2.3', anything)
64+
subject
65+
end
66+
67+
context "when every interaction has been executed" do
68+
it "sets publishable to true" do
69+
expect(VerificationResult).to receive(:new).with(true, anything, anything, anything)
70+
subject
71+
end
72+
end
73+
74+
context "when not every interaction has been executed" do
75+
let(:pact_hash_1) do
76+
{
77+
'interactions' => [{}, {}]
78+
}
79+
end
80+
it "sets publishable to false" do
81+
expect(VerificationResult).to receive(:new).with(false, anything, anything, anything)
82+
subject
83+
end
84+
end
85+
86+
context "when all the examples passed" do
87+
it "sets the success to true" do
88+
expect(VerificationResult).to receive(:new).with(anything, true, anything, anything)
89+
subject
90+
end
91+
end
92+
93+
context "when not all the examples passed" do
94+
before do
95+
example_1[:status] = 'notpassed'
96+
end
97+
98+
it "sets the success to false" do
99+
expect(VerificationResult).to receive(:new).with(anything, false, anything, anything)
100+
subject
101+
end
102+
103+
it "sets the failureCount" do
104+
expect(VerificationResult).to receive(:new) do | _, _, _, test_results_hash|
105+
expect(test_results_hash[:summary][:failureCount]).to eq 1
106+
end
107+
subject
108+
end
109+
end
110+
end
111+
end
112+
end
113+
end

spec/lib/pact/provider/verification_results/publish_spec.rb

+12-1
Original file line numberDiff line numberDiff line change
@@ -35,11 +35,13 @@ module VerificationResults
3535
let(:app_version_set) { false }
3636
let(:verification_json) { '{"foo": "bar"}' }
3737
let(:publish_verification_results) { false }
38+
let(:publishable) { true }
3839
let(:tags) { [] }
3940
let(:verification) do
4041
instance_double("Pact::Verifications::Verification",
4142
to_json: verification_json,
42-
provider_application_version_set?: app_version_set
43+
provider_application_version_set?: app_version_set,
44+
publishable?: publishable
4345
)
4446
end
4547

@@ -74,6 +76,15 @@ module VerificationResults
7476
expect(WebMock).to have_requested(:post, publish_verification_url).with(body: verification_json, headers: {'Content-Type' => 'application/json'})
7577
end
7678

79+
context "when the verification result is not publishable" do
80+
let(:publishable) { false }
81+
82+
it "does not publish the verification" do
83+
subject
84+
expect(WebMock).to_not have_requested(:post, 'http://broker/verifications')
85+
end
86+
end
87+
7788
context "with tags" do
7889
let(:tags) { ['foo'] }
7990

0 commit comments

Comments
 (0)