Skip to content

Commit e0dad27

Browse files
committed
feat(publish test results): enable test results to be published to the pact broker in the verification results
1 parent 9c93c5c commit e0dad27

File tree

12 files changed

+248
-38
lines changed

12 files changed

+248
-38
lines changed

lib/pact/provider/pact_spec_runner.rb

+5-4
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@
88
require 'pact/provider/pact_source'
99
require 'pact/provider/help/write'
1010
require 'pact/provider/verification_results/publish_all'
11+
require 'pact/provider/rspec/pact_broker_formatter'
1112

1213
require_relative 'rspec'
1314

@@ -61,6 +62,8 @@ def configure_rspec
6162
config.output_stream = Pact.configuration.output_stream
6263
end
6364

65+
::RSpec.configuration.add_formatter Pact::Provider::RSpec::PactBrokerFormatter, StringIO.new
66+
6467
if options[:format]
6568
::RSpec.configuration.add_formatter options[:format]
6669
# Don't want to mess up the JSON parsing with messages to stdout, so send it to stderr
@@ -82,14 +85,12 @@ def configure_rspec
8285
Pact.configuration.provider.app
8386
end
8487

88+
# For the Pact::Provider::RSpec::PactBrokerFormatter
89+
Pact.provider_world.pact_sources = pact_sources
8590
jsons = pact_jsons
86-
sources = pact_sources
8791

8892
config.after(:suite) do | suite |
8993
Pact::Provider::Help::Write.call(jsons)
90-
Pact::RSpec.with_rspec_3 do
91-
Pact::Provider::VerificationResults::PublishAll.call(sources, ::RSpec.configuration.reporter.failed_examples)
92-
end
9394
end
9495

9596
end

lib/pact/provider/rspec.rb

+7-2
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,7 @@ def honour_pactfile pact_uri, pact_json, options
2525
Pact.configuration.output_stream.puts "INFO: Filtering interactions by: #{options[:criteria]}" if options[:criteria] && options[:criteria].any?
2626
consumer_contract = Pact::ConsumerContract.from_json(pact_json)
2727
::RSpec.describe "Verifying a pact between #{consumer_contract.consumer.name} and #{consumer_contract.provider.name}", pactfile_uri: pact_uri do
28-
honour_consumer_contract consumer_contract, options.merge(pact_json: pact_json)
28+
honour_consumer_contract consumer_contract, options.merge(pact_json: pact_json, pact_uri: pact_uri)
2929
end
3030
end
3131

@@ -61,11 +61,16 @@ 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
65+
# Pact::Provider::RSpec::PactBrokerFormatter
6466
metadata = {
6567
pact: :verify,
6668
pact_interaction: interaction,
6769
pact_interaction_example_description: interaction_description_for_rerun_command(interaction),
68-
pact_json: options[:pact_json]
70+
pact_json: options[:pact_json],
71+
pact_uri: options[:pact_uri],
72+
pact_provider_state: interaction.provider_state,
73+
pact_description: interaction.description
6974
}
7075

7176
describe description_for(interaction), metadata do
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
require 'rspec/core/formatters'
2+
require 'pact/provider/verification_results/publish_all'
3+
4+
module Pact
5+
module Provider
6+
module RSpec
7+
class PactBrokerFormatter < ::RSpec::Core::Formatters::BaseFormatter
8+
Pact::RSpec.with_rspec_3 do
9+
::RSpec::Core::Formatters.register self, :message, :dump_summary, :stop, :seed, :close
10+
end
11+
12+
attr_reader :output_hash
13+
14+
def initialize(output)
15+
super
16+
@output_hash = {
17+
:version => ::RSpec::Core::Version::STRING
18+
}
19+
end
20+
21+
def message(notification)
22+
(@output_hash[:messages] ||= []) << notification.message
23+
end
24+
25+
def dump_summary(summary)
26+
end
27+
28+
def stop(notification)
29+
@output_hash[:examples] = notification.examples.map do |example|
30+
format_example(example).tap do |hash|
31+
e = example.exception
32+
if e
33+
hash[:exception] = {
34+
class: e.class.name,
35+
message: e.message
36+
}
37+
end
38+
end
39+
end
40+
end
41+
42+
def seed(notification)
43+
return unless notification.seed_used?
44+
@output_hash[:seed] = notification.seed
45+
end
46+
47+
def close(_notification)
48+
Pact::Provider::VerificationResults::PublishAll.call(Pact.provider_world.pact_sources, output_hash)
49+
end
50+
51+
private
52+
53+
def format_example(example)
54+
{
55+
exampleDescription: example.description,
56+
fullDescription: example.full_description,
57+
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]
61+
}
62+
end
63+
end
64+
end
65+
end
66+
end

lib/pact/provider/verification_results/create.rb

+31-15
Original file line numberDiff line numberDiff line change
@@ -4,38 +4,54 @@ module Provider
44
module VerificationResults
55
class Create
66

7-
def self.call pact_json, failed_examples
8-
new(pact_json, failed_examples).call
7+
def self.call pact_source, test_results_hash
8+
new(pact_source, test_results_hash).call
99
end
1010

11-
def initialize pact_json, failed_examples
12-
@pact_json = pact_json
13-
@failed_examples = failed_examples
11+
def initialize pact_source, test_results_hash
12+
@pact_source = pact_source
13+
@test_results_hash = test_results_hash
1414
end
1515

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

2020
private
2121

22-
def pact_hash
23-
@pact_hash ||= json_load(pact_json)
22+
def pact_uri
23+
@pact_uri ||= pact_source.uri
2424
end
2525

26-
def json_load json
27-
JSON.load(json, nil, { max_nesting: 50 })
26+
def any_failures?
27+
count_failures_for_pact_uri > 0
2828
end
2929

30-
def count_failures_for_pact_json
31-
failed_examples.collect{ |e| e.metadata[:pact_json] == pact_json }.uniq.size
30+
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) }
3234
end
3335

34-
def any_failures?
35-
count_failures_for_pact_json > 0
36+
def count_failures_for_pact_uri
37+
examples_for_pact_uri.count{ |e| e[:status] != 'passed' }
38+
end
39+
40+
def test_results_hash_for_pact_uri
41+
{
42+
examples: examples_for_pact_uri,
43+
summary: {
44+
exampleCount: examples_for_pact_uri.size,
45+
failureCount: count_failures_for_pact_uri
46+
}
47+
}
48+
end
49+
50+
def clean_example(example)
51+
example.reject{ |k, v| k == :pact_uri }
3652
end
3753

38-
attr_reader :pact_json, :failed_examples
54+
attr_reader :pact_source, :test_results_hash
3955
end
4056
end
4157
end

lib/pact/provider/verification_results/publish.rb

+9-4
Original file line numberDiff line numberDiff line change
@@ -73,20 +73,25 @@ def tag_versions
7373

7474
def publish
7575
uri = URI(publication_url)
76-
request = build_request('Post', uri, verification_result.to_json, "Publishing verification result #{verification_result.to_json} to")
76+
request = build_request('Post', uri, verification_result.to_json, "Publishing verification result #{verification_result} to")
7777
response = nil
7878
begin
7979
options = {:use_ssl => uri.scheme == 'https'}
8080
response = Net::HTTP.start(uri.host, uri.port, options) do |http|
8181
http.request request
8282
end
8383
rescue StandardError => e
84-
error_message = "Failed to publish verification result due to: #{e.class} #{e.message}"
84+
error_message = "Failed to publish verification results due to: #{e.class} #{e.message}"
8585
raise PublicationError.new(error_message)
8686
end
8787

88-
unless response.code.start_with?("2")
89-
raise PublicationError.new("Error returned from verification result publication #{response.code} #{response.body}")
88+
89+
90+
if response.code.start_with?("2")
91+
new_resource_url = JSON.parse(response.body)['_links']['self']['href']
92+
$stdout.puts "INFO: Verification results published to #{new_resource_url}"
93+
else
94+
raise PublicationError.new("Error returned from verification results publication #{response.code} #{response.body}")
9095
end
9196
end
9297

lib/pact/provider/verification_results/publish_all.rb

+6-6
Original file line numberDiff line numberDiff line change
@@ -6,13 +6,13 @@ module Provider
66
module VerificationResults
77
class PublishAll
88

9-
def self.call pact_sources, rspec_summary
10-
new(pact_sources, rspec_summary).call
9+
def self.call pact_sources, test_results_hash
10+
new(pact_sources, test_results_hash).call
1111
end
1212

13-
def initialize pact_sources, rspec_summary
13+
def initialize pact_sources, test_results_hash
1414
@pact_sources = pact_sources
15-
@rspec_summary = rspec_summary
15+
@test_results_hash = test_results_hash
1616
end
1717

1818
# TODO do not publish unless all interactions have been run
@@ -26,11 +26,11 @@ def call
2626

2727
def verification_results
2828
pact_sources.collect do | pact_source |
29-
[pact_source, Create.call(pact_source.pact_json, rspec_summary)]
29+
[pact_source, Create.call(pact_source, test_results_hash)]
3030
end
3131
end
3232

33-
attr_reader :pact_sources, :rspec_summary
33+
attr_reader :pact_sources, :test_results_hash
3434
end
3535
end
3636
end

lib/pact/provider/verification_results/verification_result.rb

+11-3
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,14 @@
1+
require 'json'
2+
13
module Pact
24
module Provider
35
module VerificationResults
46
class VerificationResult
57

6-
def initialize success, provider_application_version
8+
def initialize success, provider_application_version, test_results_hash
79
@success = success
810
@provider_application_version = provider_application_version
11+
@test_results_hash = test_results_hash
912
end
1013

1114
def provider_application_version_set?
@@ -15,13 +18,18 @@ def provider_application_version_set?
1518
def to_json
1619
{
1720
success: success,
18-
providerApplicationVersion: provider_application_version
21+
providerApplicationVersion: provider_application_version,
22+
testResults: test_results_hash
1923
}.to_json
2024
end
2125

26+
def to_s
27+
"[success: #{success}, providerApplicationVersion: #{provider_application_version}]"
28+
end
29+
2230
private
2331

24-
attr_reader :success, :provider_application_version
32+
attr_reader :success, :provider_application_version, :test_results_hash
2533
end
2634
end
2735
end

lib/pact/provider/world.rb

+2-1
Original file line numberDiff line numberDiff line change
@@ -14,6 +14,8 @@ def self.clear_provider_world
1414
module Provider
1515
class World
1616

17+
attr_accessor :pact_sources
18+
1719
def provider_states
1820
@provider_states_proxy ||= Pact::Provider::State::ProviderStateProxy.new
1921
end
@@ -29,7 +31,6 @@ def pact_verifications
2931
def pact_urls
3032
pact_verifications.collect(&:uri)
3133
end
32-
3334
end
3435
end
3536
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
require 'pact/provider/verification_results/publish_all'
2+
require 'pact/provider/pact_uri'
3+
4+
describe "publishing verifications" do
5+
6+
before do
7+
allow(Pact.configuration).to receive(:provider).and_return(provider_configuration)
8+
allow($stdout).to receive(:puts)
9+
end
10+
11+
let(:provider_configuration) do
12+
double('provider_configuration',
13+
application_version: '1.2.3',
14+
publish_verification_results?: true,
15+
tags: [])
16+
end
17+
18+
let(:pact_sources) do
19+
[instance_double('Pact::Provider::PactSource', pact_hash: pact_hash, uri: pact_uri)]
20+
end
21+
22+
let(:pact_uri) do
23+
instance_double('Pact::Provider::PactURI', uri: 'pact.json', basic_auth?: false)
24+
end
25+
26+
let(:pact_hash) do
27+
{
28+
'_links' => {
29+
'pb:publish-verification-results' => {
30+
'href' => 'http://publish/'
31+
}
32+
}
33+
}
34+
end
35+
36+
let(:created_verification_body) do
37+
{
38+
'_links' => {
39+
'self' => {
40+
'href' => 'http://created'
41+
}
42+
}
43+
}.to_json
44+
end
45+
46+
let(:test_results_hash) do
47+
{
48+
examples: [
49+
{
50+
exampleDescription: '1',
51+
status: 'passed',
52+
pact_uri: pact_uri
53+
}
54+
]
55+
}
56+
end
57+
58+
subject { Pact::Provider::VerificationResults::PublishAll.call(pact_sources, test_results_hash) }
59+
60+
let!(:request) do
61+
stub_request(:post, 'http://publish').to_return(status: 200, body: created_verification_body)
62+
end
63+
64+
it "publishes the results" do
65+
subject
66+
expect(request).to have_been_made
67+
end
68+
end

0 commit comments

Comments
 (0)