Skip to content

Commit 55bb935

Browse files
authored
feat: use new 'pacts for verification' endpoint to retrieve pacts (#199)
1 parent 39e6221 commit 55bb935

20 files changed

+344
-206
lines changed

lib/pact/cli/run_pact_verification.rb

+11-5
Original file line numberDiff line numberDiff line change
@@ -44,15 +44,17 @@ def load_pact_helper
4444
end
4545

4646
def run_specs
47-
exit_code = if options[:pact_uri]
48-
run_with_pact_uri
47+
exit_code = if options[:pact_uri].is_a?(String)
48+
run_with_pact_url_string
49+
elsif options[:pact_uri]
50+
run_with_pact_uri_object # from pact-provider-verifier
4951
else
50-
run_with_configured_pacts
52+
run_with_configured_pacts_from_pact_helper
5153
end
5254
exit exit_code
5355
end
5456

55-
def run_with_pact_uri
57+
def run_with_pact_url_string
5658
pact_repository_uri_options = {}
5759
pact_repository_uri_options[:username] = options[:pact_broker_username] if options[:pact_broker_username]
5860
pact_repository_uri_options[:password] = options[:pact_broker_password] if options[:pact_broker_password]
@@ -61,7 +63,11 @@ def run_with_pact_uri
6163
Pact::Provider::PactSpecRunner.new([pact_uri], pact_spec_options).run
6264
end
6365

64-
def run_with_configured_pacts
66+
def run_with_pact_uri_object
67+
Pact::Provider::PactSpecRunner.new([options[:pact_uri]], pact_spec_options).run
68+
end
69+
70+
def run_with_configured_pacts_from_pact_helper
6571
pact_urls = Pact.provider_world.pact_urls
6672
raise "Please configure a pact to verify" if pact_urls.empty?
6773
Pact::Provider::PactSpecRunner.new(pact_urls, pact_spec_options).run

lib/pact/hal/http_client.rb

+5-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
require 'pact/retry'
22
require 'pact/hal/authorization_header_redactor'
33
require 'net/http'
4+
require 'rack'
45

56
module Pact
67
module Hal
@@ -15,9 +16,11 @@ def initialize options
1516
end
1617

1718
def get href, params = {}, headers = {}
18-
query = params.collect{ |(key, value)| "#{CGI::escape(key)}=#{CGI::escape(value)}" }.join("&")
1919
uri = URI(href)
20-
uri.query = query
20+
if params && params.any?
21+
existing_params = Rack::Utils.parse_nested_query(uri.query)
22+
uri.query = Rack::Utils.build_nested_query(existing_params.merge(params))
23+
end
2124
perform_request(create_request(uri, 'Get', nil, headers), uri)
2225
end
2326

lib/pact/hal/link.rb

+19-1
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,10 @@ def get(payload = {}, headers = {})
4040
wrap_response(href, @http_client.get(href, payload, headers))
4141
end
4242

43+
def get!(*args)
44+
get(*args).assert_success!
45+
end
46+
4347
def put(payload = nil, headers = {})
4448
wrap_response(href, @http_client.put(href, payload ? payload.to_json : nil, headers))
4549
end
@@ -51,11 +55,25 @@ def post(payload = nil, headers = {})
5155
def expand(params)
5256
expanded_url = expand_url(params, href)
5357
new_attrs = @attrs.merge('href' => expanded_url)
54-
Link.new(new_attrs, @http_client)
58+
Link.new(new_attrs, http_client)
59+
end
60+
61+
def with_query(query)
62+
if query && query.any?
63+
uri = URI(href)
64+
existing_query_params = Rack::Utils.parse_nested_query(uri.query)
65+
uri.query = Rack::Utils.build_nested_query(existing_query_params.merge(query))
66+
new_attrs = attrs.merge('href' => uri.to_s)
67+
Link.new(new_attrs, http_client)
68+
else
69+
self
70+
end
5571
end
5672

5773
private
5874

75+
attr_reader :attrs, :http_client
76+
5977
def wrap_response(href, http_response)
6078
require 'pact/hal/entity' # avoid circular reference
6179
if http_response.success?

lib/pact/pact_broker.rb

+10-4
Original file line numberDiff line numberDiff line change
@@ -1,19 +1,25 @@
11
require 'pact/pact_broker/fetch_pacts'
2-
require 'pact/pact_broker/fetch_pending_pacts'
2+
require 'pact/pact_broker/fetch_pact_uris_for_verification'
3+
require 'pact/provider/pact_uri'
34

45
#
5-
# @public Use by Pact Provider Verifier
6+
# @public Used by Pact Provider Verifier
67
#
78
module Pact
89
module PactBroker
910
extend self
1011

12+
# Keep for backwards compatibility with pact-provider-verifier < 1.23.1
1113
def fetch_pact_uris *args
1214
Pact::PactBroker::FetchPacts.call(*args).collect(&:uri)
1315
end
1416

15-
def fetch_pending_pact_uris *args
16-
Pact::PactBroker::FetchPendingPacts.call(*args).collect(&:uri)
17+
def fetch_pact_uris_for_verification *args
18+
Pact::PactBroker::FetchPactURIsForVerification.call(*args)
19+
end
20+
21+
def build_pact_uri(*args)
22+
Pact::Provider::PactURI.new(*args)
1723
end
1824
end
1925
end
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,96 @@
1+
require 'pact/hal/entity'
2+
require 'pact/hal/http_client'
3+
require 'pact/provider/pact_uri'
4+
require 'pact/errors'
5+
require 'pact/pact_broker/fetch_pacts'
6+
7+
module Pact
8+
module PactBroker
9+
class FetchPactURIsForVerification
10+
attr_reader :provider, :consumer_version_selectors, :provider_version_tags, :broker_base_url, :http_client_options, :http_client
11+
12+
PACTS_FOR_VERIFICATION_RELATION = 'beta:provider-pacts-for-verification'.freeze
13+
PACTS = 'pacts'.freeze
14+
HREF = 'href'.freeze
15+
LINKS = '_links'.freeze
16+
SELF = 'self'.freeze
17+
EMBEDDED = '_embedded'.freeze
18+
19+
def initialize(provider, consumer_version_selectors, provider_version_tags, broker_base_url, http_client_options)
20+
@provider = provider
21+
@consumer_version_selectors = consumer_version_selectors
22+
@provider_version_tags = provider_version_tags
23+
@http_client_options = http_client_options
24+
@broker_base_url = broker_base_url
25+
@http_client = Pact::Hal::HttpClient.new(http_client_options)
26+
end
27+
28+
def self.call(provider, consumer_version_selectors, provider_version_tags, broker_base_url, http_client_options)
29+
new(provider, consumer_version_selectors, provider_version_tags, broker_base_url, http_client_options).call
30+
end
31+
32+
def call
33+
if index.can?(PACTS_FOR_VERIFICATION_RELATION)
34+
log_message
35+
pacts_for_verification
36+
else
37+
# Fall back to old method of fetching pacts
38+
consumer_version_tags = consumer_version_selectors.collect{ | selector | selector[:tag] }
39+
FetchPacts.call(provider, consumer_version_tags, broker_base_url, http_client_options)
40+
end
41+
end
42+
43+
private
44+
45+
def index
46+
@index_entity ||= Pact::Hal::Link.new({ "href" => broker_base_url }, http_client).get.assert_success!
47+
end
48+
49+
def pacts_for_verification
50+
pacts_for_verification_entity.response.body[EMBEDDED][PACTS].collect do | pact |
51+
metadata = {
52+
pending: pact["verificationProperties"]["pending"],
53+
pending_reason: pact["verificationProperties"]["pendingReason"],
54+
inclusion_reason: pact["verificationProperties"]["inclusionReason"],
55+
}
56+
Pact::Provider::PactURI.new(pact[LINKS][SELF][HREF], http_client_options, metadata)
57+
end
58+
end
59+
60+
def pacts_for_verification_entity
61+
index
62+
._link(PACTS_FOR_VERIFICATION_RELATION)
63+
.expand(provider: provider)
64+
.with_query(query)
65+
.get!
66+
end
67+
68+
def query
69+
q = {}
70+
if consumer_version_selectors&.any?
71+
q["consumer_version_selectors"] = consumer_version_selectors
72+
end
73+
74+
if provider_version_tags&.any?
75+
q["provider_version_tags"] = provider_version_tags
76+
end
77+
q
78+
end
79+
80+
def log_message
81+
latest = consumer_version_selectors&.any? ? "" : "latest "
82+
message = "INFO: Fetching #{latest}pacts for #{provider} from #{broker_base_url}"
83+
if consumer_version_selectors&.any?
84+
desc = consumer_version_selectors.collect do |selector|
85+
all_or_latest = selector[:all] ? "all" : "latest"
86+
# TODO support fallback
87+
name = selector[:fallback] ? "#{selector[:tag]} (or #{selector[:fallback]} if not found)" : selector[:tag]
88+
"#{all_or_latest} #{name}"
89+
end.join(", ")
90+
message << " for tags: #{desc}"
91+
end
92+
Pact.configuration.output_stream.puts message
93+
end
94+
end
95+
end
96+
end

lib/pact/pact_broker/fetch_pending_pacts.rb

-58
This file was deleted.

lib/pact/provider/configuration/pact_verification_from_broker.rb

+22-4
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
require 'pact/shared/dsl'
22
require 'pact/provider/world'
3-
require 'pact/pact_broker/fetch_pacts'
3+
require 'pact/pact_broker/fetch_pact_uris_for_verification'
44
require 'pact/errors'
55

66
module Pact
@@ -14,10 +14,11 @@ class PactVerificationFromBroker
1414
# in parent scope, it will clash with these ones,
1515
# so put an underscore in front of the name to be safer.
1616

17-
attr_accessor :_provider_name, :_pact_broker_base_url, :_consumer_version_tags, :_basic_auth_options, :_verbose
17+
attr_accessor :_provider_name, :_pact_broker_base_url, :_consumer_version_tags, :_provider_version_tags, :_basic_auth_options, :_verbose
1818

19-
def initialize(provider_name)
19+
def initialize(provider_name, provider_version_tags)
2020
@_provider_name = provider_name
21+
@_provider_version_tags = provider_version_tags
2122
@_consumer_version_tags = []
2223
@_verbose = false
2324
end
@@ -45,10 +46,27 @@ def finalize
4546
private
4647

4748
def create_pact_verification
48-
fetch_pacts = Pact::PactBroker::FetchPacts.new(_provider_name, _consumer_version_tags, _pact_broker_base_url, _basic_auth_options.merge(verbose: _verbose))
49+
fetch_pacts = Pact::PactBroker::FetchPactURIsForVerification.new(
50+
_provider_name,
51+
consumer_version_selectors,
52+
_provider_version_tags,
53+
_pact_broker_base_url,
54+
_basic_auth_options.merge(verbose: _verbose)
55+
)
56+
4957
Pact.provider_world.add_pact_uri_source fetch_pacts
5058
end
5159

60+
def consumer_version_selectors
61+
# TODO support "all"
62+
_consumer_version_tags.collect do | tag |
63+
{
64+
tag: tag,
65+
latest: true
66+
}
67+
end
68+
end
69+
5270
def validate
5371
raise Pact::Error.new("Please provide a pact_broker_base_url from which to retrieve the pacts") unless _pact_broker_base_url
5472
end

lib/pact/provider/configuration/service_provider_dsl.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,7 @@ def create_pact_verification consumer_name, options, &block
6565
end
6666

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

7171
def finalize

lib/pact/provider/pact_uri.rb

+6-4
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,19 @@
11
module Pact
22
module Provider
33
class PactURI
4-
attr_reader :uri, :options
4+
attr_reader :uri, :options, :metadata
55

6-
def initialize (uri, options={})
6+
def initialize (uri, options = nil, metadata = nil)
77
@uri = uri
8-
@options = options
8+
@options = options || {}
9+
@metadata = metadata || {} # make sure it's not nil if nil is passed in
910
end
1011

1112
def == other
1213
other.is_a?(PactURI) &&
1314
uri == other.uri &&
14-
options == other.options
15+
options == other.options &&
16+
metadata == other.metadata
1517
end
1618

1719
def basic_auth?

lib/pact/provider/rspec.rb

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

2323
def honour_pactfile pact_uri, pact_json, options
24-
pact_description = options[:ignore_failures] ? "Pending pact" : "pact"
25-
Pact.configuration.output_stream.puts "INFO: Reading #{pact_description} at #{pact_uri}"
26-
Pact.configuration.output_stream.puts "INFO: Filtering interactions by: #{options[:criteria]}" if options[:criteria] && options[:criteria].any?
24+
Pact.configuration.output_stream.puts "INFO: Reading pact at #{pact_uri}"
25+
Pact.configuration.output_stream.puts("DEBUG: #{pact_uri.metadata[:inclusion_reason]}") if pact_uri.metadata[:inclusion_reason]
26+
Pact.configuration.output_stream.puts("DEBUG: #{pact_uri.metadata[:pending_reason]}") if pact_uri.metadata[:pending_reason]
27+
Pact.configuration.output_stream.puts "DEBUG: Filtering interactions by: #{options[:criteria]}" if options[:criteria] && options[:criteria].any?
2728
consumer_contract = Pact::ConsumerContract.from_json(pact_json)
28-
::RSpec.describe "Verifying a #{pact_description} between #{consumer_contract.consumer.name} and #{consumer_contract.provider.name}", pactfile_uri: pact_uri do
29+
suffix = pact_uri.metadata[:pending] ? " [PENDING]": ""
30+
::RSpec.describe "Verifying a pact between #{consumer_contract.consumer.name} and #{consumer_contract.provider.name}#{suffix}", pactfile_uri: pact_uri do
2931
honour_consumer_contract consumer_contract, options.merge(pact_json: pact_json, pact_uri: pact_uri)
3032
end
3133
end
@@ -74,7 +76,7 @@ def describe_interaction interaction, options
7476
pact_interaction: interaction,
7577
pact_interaction_example_description: interaction_description_for_rerun_command(interaction),
7678
pact_uri: options[:pact_uri],
77-
pact_ignore_failures: options[:ignore_failures]
79+
pact_ignore_failures: options[:pact_uri].metadata[:pending]
7880
}
7981

8082
describe description_for(interaction), metadata do

0 commit comments

Comments
 (0)