Skip to content

Commit bff4611

Browse files
committed
feat(verifications): allow provider version tags to be specified
1 parent 4f8688d commit bff4611

File tree

9 files changed

+94
-17
lines changed

9 files changed

+94
-17
lines changed

lib/pact/provider/configuration/service_provider_config.rb

+6-1
Original file line numberDiff line numberDiff line change
@@ -5,8 +5,9 @@ class ServiceProviderConfig
55

66
attr_accessor :application_version
77

8-
def initialize application_version, publish_verification_results, &app_block
8+
def initialize application_version, tags, publish_verification_results, &app_block
99
@application_version = application_version
10+
@tags = [*tags]
1011
@publish_verification_results = publish_verification_results
1112
@app_block = app_block
1213
end
@@ -18,6 +19,10 @@ def app
1819
def publish_verification_results?
1920
@publish_verification_results
2021
end
22+
23+
def tags
24+
@tags
25+
end
2126
end
2227
end
2328
end

lib/pact/provider/configuration/service_provider_dsl.rb

+7-2
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ class ServiceProviderDSL
1313

1414
extend Pact::DSL
1515

16-
attr_accessor :name, :app_block, :application_version, :publish_verification_results
16+
attr_accessor :name, :app_block, :application_version, :tags, :publish_verification_results
1717

1818
CONFIG_RU_APP = lambda {
1919
unless File.exist? Pact.configuration.config_ru_path
@@ -25,6 +25,7 @@ class ServiceProviderDSL
2525
def initialize name
2626
@name = name
2727
@publish_verification_results = false
28+
@tags = []
2829
@app_block = CONFIG_RU_APP
2930
end
3031

@@ -37,6 +38,10 @@ def app_version application_version
3738
self.application_version = application_version
3839
end
3940

41+
def app_version_tags tags
42+
self.tags = tags
43+
end
44+
4045
def publish_verification_results publish_verification_results
4146
self.publish_verification_results = publish_verification_results
4247
Pact::RSpec.with_rspec_2 do
@@ -70,7 +75,7 @@ def application_version_blank?
7075
end
7176

7277
def create_service_provider
73-
Pact.configuration.provider = ServiceProviderConfig.new(application_version, publish_verification_results, &@app_block)
78+
Pact.configuration.provider = ServiceProviderConfig.new(application_version, tags, publish_verification_results, &@app_block)
7479
end
7580
end
7681
end

lib/pact/provider/verification_results/publish.rb

+39-7
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,9 @@
11
require 'json'
22
require 'pact/errors'
33

4+
# TODO move this to the pact broker client
5+
# TODO retries
6+
47
module Pact
58
module Provider
69
module VerificationResults
@@ -20,6 +23,11 @@ def initialize pact_source, verification_result
2023

2124
def call
2225
if Pact.configuration.provider.publish_verification_results?
26+
if tag_url('')
27+
tag
28+
else
29+
Pact.configuration.error_stream.puts "WARN: Cannot tag provider version as there is no link named pb:tag-version in the pact JSON."
30+
end
2331
if publication_url
2432
publish
2533
else
@@ -34,11 +42,35 @@ def publication_url
3442
@publication_url ||= pact_source.pact_hash.fetch('_links', {}).fetch('pb:publish-verification-results', {})['href']
3543
end
3644

45+
def tag_url tag
46+
href = pact_source.pact_hash.dig('_links', 'pb:tag-version', 'href')
47+
href ? href.gsub('{tag}', tag) : nil
48+
end
49+
50+
def tag
51+
Pact.configuration.provider.tags.each do | tag |
52+
uri = URI(tag_url(tag))
53+
request = build_request('Put', uri, nil, "Tagging provider version at")
54+
response = nil
55+
begin
56+
options = {:use_ssl => uri.scheme == 'https'}
57+
response = Net::HTTP.start(uri.host, uri.port, options) do |http|
58+
http.request request
59+
end
60+
rescue StandardError => e
61+
error_message = "Failed to tag provider version due to: #{e.class} #{e.message}"
62+
raise PublicationError.new(error_message)
63+
end
64+
65+
unless response.code.start_with?("2")
66+
raise PublicationError.new("Error returned from tagging request #{response.code} #{response.body}")
67+
end
68+
end
69+
end
70+
3771
def publish
38-
#TODO https
39-
#TODO username/password
4072
uri = URI(publication_url)
41-
request = build_request(uri)
73+
request = build_request('Post', uri, verification_result.to_json, "Publishing verification result #{verification_result.to_json} to")
4274
response = nil
4375
begin
4476
options = {:use_ssl => uri.scheme == 'https'}
@@ -55,16 +87,16 @@ def publish
5587
end
5688
end
5789

58-
def build_request uri
59-
request = Net::HTTP::Post.new(uri.path)
90+
def build_request meth, uri, body, action
91+
request = Net::HTTP.const_get(meth).new(uri.path)
6092
request['Content-Type'] = "application/json"
61-
request.body = verification_result.to_json
93+
request.body = body if body
6294
debug_uri = uri
6395
if pact_source.uri.basic_auth?
6496
request.basic_auth pact_source.uri.username, pact_source.uri.password
6597
debug_uri = URI(uri.to_s).tap { |x| x.userinfo="#{pact_source.uri.username}:*****"}
6698
end
67-
Pact.configuration.output_stream.puts "INFO: Publishing verification result #{verification_result.to_json} to #{debug_uri}"
99+
Pact.configuration.output_stream.puts "INFO: #{action} #{debug_uri}"
68100
request
69101
end
70102

lib/pact/provider/verification_results/publish_all.rb

+1
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ def initialize pact_sources, rspec_summary
1515
@rspec_summary = rspec_summary
1616
end
1717

18+
# TODO do not publish unless all interactions have been run
1819
def call
1920
verification_results.collect do | pair |
2021
Publish.call(pair.first, pair.last)

lib/tasks/pact.rake

-1
Original file line numberDiff line numberDiff line change
@@ -31,5 +31,4 @@ namespace :pact do
3131
require 'pact/provider/help/console_text'
3232
puts Pact::Provider::Help::ConsoleText.(args[:reports_dir])
3333
end
34-
3534
end

spec/lib/pact/provider/configuration/service_provider_config_spec.rb

+2-2
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ module Configuration
1010

1111
let(:app_block) { ->{ Object.new } }
1212

13-
subject { ServiceProviderConfig.new("1.2.3'", true, &app_block) }
13+
subject { ServiceProviderConfig.new("1.2.3'", [], true, &app_block) }
1414

1515
it "should execute the app_block each time" do
1616
expect(subject.app.object_id).to_not equal(subject.app.object_id)
@@ -20,4 +20,4 @@ module Configuration
2020
end
2121
end
2222
end
23-
end
23+
end

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

+37-2
Original file line numberDiff line numberDiff line change
@@ -6,13 +6,27 @@ module VerificationResults
66
describe Publish do
77
describe "call" do
88
let(:publish_verification_url) { nil }
9+
let(:tag_version_url) { 'http://tag-me/{tag}' }
910
let(:pact_source) { instance_double("Pact::Provider::PactSource", pact_hash: pact_hash, uri: pact_url)}
1011
let(:pact_url) { instance_double("Pact::Provider::PactURI", basic_auth?: basic_auth, username: 'username', password: 'password')}
1112
let(:basic_auth) { false }
12-
let(:pact_hash) { {'consumer' => {'name' => 'Foo'}, '_links' => {'pb:publish-verification-results'=> {'href' => publish_verification_url}}} }
13+
let(:pact_hash) do
14+
{
15+
'consumer' => {
16+
'name' => 'Foo'
17+
},
18+
'_links' => {
19+
'pb:publish-verification-results'=> {
20+
'href' => publish_verification_url
21+
},
22+
'pb:tag-version'=> {'href' => tag_version_url}
23+
}
24+
}
25+
end
1326
let(:app_version_set) { false }
1427
let(:verification_json) { '{"foo": "bar"}' }
1528
let(:publish_verification_results) { false }
29+
let(:tags) { [] }
1630
let(:verification) do
1731
instance_double("Pact::Verifications::Verification",
1832
to_json: verification_json,
@@ -21,13 +35,14 @@ module VerificationResults
2135
end
2236

2337
let(:provider_configuration) do
24-
double('provider config', publish_verification_results?: publish_verification_results)
38+
double('provider config', publish_verification_results?: publish_verification_results, tags: tags)
2539
end
2640

2741
before do
2842
allow($stdout).to receive(:puts)
2943
allow(Pact.configuration).to receive(:provider).and_return(provider_configuration)
3044
stub_request(:post, 'http://broker/verifications')
45+
stub_request(:put, /tag-me/)
3146
end
3247

3348
subject { Publish.call(pact_source, verification)}
@@ -50,6 +65,26 @@ module VerificationResults
5065
expect(WebMock).to have_requested(:post, publish_verification_url).with(body: verification_json, headers: {'Content-Type' => 'application/json'})
5166
end
5267

68+
context "with tags" do
69+
let(:tags) { ['foo'] }
70+
71+
it "tags the provider version" do
72+
subject
73+
expect(WebMock).to have_requested(:put, 'http://tag-me/foo').with(headers: {'Content-Type' => 'application/json'})
74+
end
75+
76+
context "when there is no pb:tag-version link" do
77+
before do
78+
pact_hash['_links'].delete('pb:tag-version')
79+
end
80+
81+
it "prints a warning" do
82+
expect($stderr).to receive(:puts).with /WARN: Cannot tag provider version/
83+
subject
84+
end
85+
end
86+
end
87+
5388
context "when basic auth is configured on the pact URL" do
5489
let(:basic_auth) { true }
5590
it "sets the username and password for the pubication URL" do

spec/support/bar_pact_helper.rb

+1
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@ def call env
1616
Pact.service_provider "Bar" do
1717
app { BarApp.new }
1818
app_version '1.2.3'
19+
app_version_tags ['master']
1920
publish_verification_results true
2021

2122
honours_pact_with 'Foo' do

tasks/foo-bar.rake

+1-2
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,10 @@
11
require 'pact/tasks/verification_task'
22
# Use for end to end manual debugging of issues.
33

4-
BROKER_BASE_URL = 'http://127.0.0.1:9292'
4+
BROKER_BASE_URL = 'http://localhost:9292'
55

66
RSpec::Core::RakeTask.new('pact:foobar:create') do | task |
77
task.pattern = "spec/features/foo_bar_spec.rb"
8-
98
end
109

1110
task 'pact:foobar:publish' do

0 commit comments

Comments
 (0)