Skip to content

Commit ef97898

Browse files
committed
feat: allow pacts to be dynamically fetched from a pact broker by provider name and tags
1 parent fca52f2 commit ef97898

File tree

7 files changed

+167
-73
lines changed

7 files changed

+167
-73
lines changed

lib/pact/provider/configuration/pact_verification_from_broker.rb

+19-20
Original file line numberDiff line numberDiff line change
@@ -1,30 +1,33 @@
1-
require 'pact/provider/pact_verification'
2-
require 'pact/provider/pact_uri'
31
require 'pact/shared/dsl'
42
require 'pact/provider/world'
3+
require 'pact/pact_broker/fetch_pacts'
54

65
module Pact
76
module Provider
8-
97
module Configuration
10-
118
class PactVerificationFromBroker
129

1310
extend Pact::DSL
1411

15-
attr_accessor :name, :pact_broker_base_url, :tags
12+
# If user declares a variable with the same name as one of these attributes
13+
# in parent scope, it will clash with these ones,
14+
# so put an underscore in front of the name to be safer.
1615

17-
def initialize(name, options = {})
18-
@tags = options.fetch(:consumer_version_tags) || []
19-
@pact_broker_base_url = options.fetch(:pact_broker_base_url) || ''
20-
@provider_name = name
21-
@options = options
16+
attr_accessor :_provider_name, :_pact_broker_base_url, :_consumer_version_tags, :_basic_auth_options
17+
18+
def initialize(provider_name)
19+
@_provider_name = provider_name
20+
@_consumer_version_tags = []
2221
end
2322

2423
dsl do
25-
def pact_broker_base_url pact_broker_base_url, options
26-
@pact_broker_base_url = URI(pact_broker_base_url)
27-
@pact_broker_base_url.set_user(options[:username]) if options[:username] # not sure about this exactly, I'll work it out when I get there.
24+
def pact_broker_base_url pact_broker_base_url, basic_auth_options = {}
25+
self._pact_broker_base_url = pact_broker_base_url
26+
self._basic_auth_options = basic_auth_options
27+
end
28+
29+
def consumer_version_tags consumer_version_tags
30+
self._consumer_version_tags = *consumer_version_tags
2831
end
2932
end
3033

@@ -36,17 +39,13 @@ def finalize
3639
private
3740

3841
def create_pact_verification
39-
pacts = Pact::PactBroker::FetchPacts.call(@provider_name, tags, pact_broker_base_url, @options)
40-
pacts.each do |pact_uri|
41-
verification = Pact::Provider::PactVerification.new(nil, pact_uri, nil)
42-
Pact.provider_world.add_pact_verification verification
43-
end
42+
fetch_pacts = Pact::PactBroker::FetchPacts.new(_provider_name, _consumer_version_tags, _pact_broker_base_url, _basic_auth_options)
43+
Pact.provider_world.add_pact_uri_source fetch_pacts
4444
end
4545

4646
def validate
47-
raise "Please provide a pact_broker_base_url for the verification" unless pact_broker_base_url
47+
raise Pact::Error.new("Please provide a pact_broker_base_url from which to retrieve the pacts") unless _pact_broker_base_url
4848
end
49-
5049
end
5150
end
5251
end

lib/pact/provider/configuration/service_provider_dsl.rb

+4-4
Original file line numberDiff line numberDiff line change
@@ -55,17 +55,17 @@ def honours_pact_with consumer_name, options = {}, &block
5555
create_pact_verification consumer_name, options, &block
5656
end
5757

58-
def honours_pacts_from_pact_broker options = {}, &block
59-
create_pact_verification_from_broker options, &block
58+
def honours_pacts_from_pact_broker &block
59+
create_pact_verification_from_broker &block
6060
end
6161
end
6262

6363
def create_pact_verification consumer_name, options, &block
6464
PactVerification.build(consumer_name, options, &block)
6565
end
6666

67-
def create_pact_verification_from_broker(options, &block)
68-
PactVerificationFromBroker.build(name, options, &block)
67+
def create_pact_verification_from_broker(&block)
68+
PactVerificationFromBroker.build(name, &block)
6969
end
7070

7171
def finalize

lib/pact/provider/world.rb

+16-2
Original file line numberDiff line numberDiff line change
@@ -29,8 +29,22 @@ def pact_verifications
2929
end
3030

3131
def pact_urls
32-
pact_verifications.collect(&:uri)
32+
(pact_verifications.collect(&:uri) + pact_uris_from_pact_uri_sources).compact
33+
end
34+
35+
def add_pact_uri_source pact_uri_source
36+
pact_uri_sources << pact_uri_source
37+
end
38+
39+
private
40+
41+
def pact_uri_sources
42+
@pact_uri_sources ||= []
43+
end
44+
45+
def pact_uris_from_pact_uri_sources
46+
pact_uri_sources.collect{| pact_uri_source| pact_uri_source.call }.flatten
3347
end
3448
end
3549
end
36-
end
50+
end
Original file line numberDiff line numberDiff line change
@@ -1,52 +1,92 @@
1-
require 'spec_helper'
21
require 'pact/provider/configuration/pact_verification'
3-
require 'pact/pact_broker/fetch_pacts'
42

53
module Pact
64
module Provider
75
module Configuration
8-
describe PactVerification do
9-
10-
describe 'create_verification' do
11-
let(:url) { 'http://some/uri' }
6+
describe PactVerificationFromBroker do
7+
describe 'build' do
128
let(:provider_name) {'provider-name'}
13-
let(:pact_repository_uri_options) do
9+
let(:base_url) { "http://broker.org" }
10+
let(:basic_auth_options) do
1411
{
1512
username: 'pact_broker_username',
1613
password: 'pact_broker_password'
1714
}
1815
end
19-
let(:tag) do
20-
{
21-
name: 'tag-name',
22-
all: false,
23-
fallback: 'master'
24-
}
25-
end
16+
let(:tags) { ['master'] }
2617

27-
let(:options) do
28-
{
29-
pact_broker_base_url: url,
30-
consumer_version_tags: [tag]
31-
}
18+
before do
19+
allow(Pact::PactBroker::FetchPacts).to receive(:new).and_return(fetch_pacts)
20+
allow(Pact.provider_world).to receive(:add_pact_uri_source)
3221
end
22+
3323
context "with valid values" do
3424
subject do
35-
PactVerificationFromBroker.build(provider_name, options) do
25+
PactVerificationFromBroker.build(provider_name) do
26+
pact_broker_base_url base_url, basic_auth_options
27+
consumer_version_tags tags
28+
end
29+
end
30+
31+
let(:fetch_pacts) { double('FetchPacts') }
32+
33+
it "creates a instance of Pact::PactBroker::FetchPacts" do
34+
expect(Pact::PactBroker::FetchPacts).to receive(:new).with(provider_name, tags, base_url, basic_auth_options)
35+
subject
36+
end
37+
38+
it "adds a pact_uri_source to the provider world" do
39+
expect(Pact.provider_world).to receive(:add_pact_uri_source).with(fetch_pacts)
40+
subject
41+
end
42+
end
43+
44+
context "with a missing base url" do
45+
subject do
46+
PactVerificationFromBroker.build(provider_name) do
47+
48+
end
49+
end
50+
51+
let(:fetch_pacts) { double('FetchPacts') }
52+
53+
it "raises an error" do
54+
expect { subject }.to raise_error Pact::Error, /Please provide a pact_broker_base_url/
55+
end
56+
end
57+
58+
context "with a non array object for consumer_version_tags" do
59+
subject do
60+
PactVerificationFromBroker.build(provider_name) do
61+
pact_broker_base_url base_url
62+
consumer_version_tags "master"
63+
end
64+
end
65+
66+
let(:fetch_pacts) { double('FetchPacts') }
67+
68+
it "coerces the value into an array" do
69+
expect(Pact::PactBroker::FetchPacts).to receive(:new).with(anything, ["master"], anything, anything)
70+
subject
71+
end
72+
end
73+
74+
context "when no consumer_version_tags are provided" do
75+
subject do
76+
PactVerificationFromBroker.build(provider_name) do
77+
pact_broker_base_url base_url
3678
end
3779
end
3880

39-
it "creates a Verification" do
40-
allow(Pact::PactBroker::FetchPacts).to receive(:call).and_return(['pact-urls'])
81+
let(:fetch_pacts) { double('FetchPacts') }
4182

42-
tags = [tag]
43-
expect(Pact::PactBroker::FetchPacts).to receive(:call).with(provider_name, tags, url, options)
44-
expect(Pact::Provider::PactVerification).to receive(:new).with(nil, 'pact-urls', nil)
83+
it "creates an instance of FetchPacts with an emtpy array for the consumer_version_tags" do
84+
expect(Pact::PactBroker::FetchPacts).to receive(:new).with(anything, [], anything, anything)
4585
subject
4686
end
4787
end
4888
end
4989
end
5090
end
5191
end
52-
end
92+
end

spec/lib/pact/provider/world_spec.rb

+36-1
Original file line numberDiff line numberDiff line change
@@ -26,16 +26,51 @@ module Provider
2626
describe World do
2727

2828
subject { World.new }
29+
2930
describe "provider_states" do
3031
it "returns a provider state proxy" do
3132
expect(subject.provider_states).to be_instance_of State::ProviderStateProxy
3233
end
3334
it "returns the same object each time" do
3435
expect(subject.provider_states).to be subject.provider_states
3536
end
36-
3737
end
3838

39+
describe "pact_urls" do
40+
context "with pact_uri_sources" do
41+
before do
42+
subject.add_pact_uri_source(pact_uri_source_1)
43+
subject.add_pact_uri_source(pact_uri_source_2)
44+
end
45+
46+
let(:pact_uri_source_1) { double('pact_uri_source_1', call: ["uri-1"]) }
47+
let(:pact_uri_source_2) { double('pact_uri_source_2', call: ["uri-2"]) }
48+
49+
let(:pact_urls) { subject.pact_urls }
50+
51+
it "invokes call on the pact_uri_sources" do
52+
expect(pact_uri_source_1).to receive(:call)
53+
expect(pact_uri_source_2).to receive(:call)
54+
pact_urls
55+
end
56+
57+
it "concatenates the results" do
58+
expect(pact_urls).to eq ["uri-1", "uri-2"]
59+
end
60+
61+
context "with a pact_verification" do
62+
before do
63+
subject.add_pact_verification(pact_verification)
64+
end
65+
66+
let(:pact_verification) { double('PactVerification', uri: "uri-3") }
67+
68+
it "concatenates the results with those of the pact_uri_sources" do
69+
expect(pact_urls).to eq ["uri-3", "uri-1", "uri-2"]
70+
end
71+
end
72+
end
73+
end
3974
end
4075
end
4176
end

spec/support/bar_pact_helper.rb

+24-19
Original file line numberDiff line numberDiff line change
@@ -2,26 +2,31 @@
22
require 'pact/provider/rspec'
33

44
module Pact
5-
module Test
6-
class BarApp
7-
def call env
8-
[200, {'Content-Type' => 'application/json'}, [{"status" => "5"},{"status" => "6"}].to_json]
9-
end
10-
end
5+
module Test
6+
class BarApp
7+
def call env
8+
[200, {'Content-Type' => 'application/json'}, [{"status" => "5"},{"status" => "6"}].to_json]
9+
end
10+
end
1111

12-
Pact.configure do | config |
13-
config.logger.level = Logger::DEBUG
14-
end
12+
Pact.configure do | config |
13+
config.logger.level = Logger::DEBUG
14+
end
1515

16-
Pact.service_provider "Bar" do
17-
app { BarApp.new }
18-
app_version '1.2.3'
19-
app_version_tags ['master']
20-
publish_verification_results true
16+
Pact.service_provider "Bar" do
17+
app { BarApp.new }
18+
app_version '1.2.3'
19+
app_version_tags ['master']
20+
publish_verification_results true
2121

22-
honours_pact_with 'Foo' do
23-
pact_uri './spec/support/foo-bar.json'
24-
end
25-
end
26-
end
22+
honours_pacts_from_pact_broker do
23+
pact_broker_base_url "http://localhost:9292"
24+
consumer_version_tags ["prod"]
25+
end
26+
27+
honours_pact_with 'Foo' do
28+
pact_uri './spec/support/foo-bar.json'
29+
end
30+
end
31+
end
2732
end

tasks/foo-bar.rake

+2-1
Original file line numberDiff line numberDiff line change
@@ -24,8 +24,9 @@ task 'pact:foobar:publish' do
2424
puts response.code unless response.code == '200'
2525
end
2626

27+
#'./spec/pacts/foo-bar.json'
2728
Pact::VerificationTask.new('foobar') do | pact |
28-
pact.uri './spec/pacts/foo-bar.json', pact_helper: './spec/support/bar_pact_helper.rb'
29+
pact.uri nil, pact_helper: './spec/support/bar_pact_helper.rb'
2930
end
3031

3132

0 commit comments

Comments
 (0)