require 'term/ansicolor' require 'pact_broker/client' require 'pact_broker/client/retry' require 'pact_broker/client/pact_file' require 'pact_broker/client/pact_hash' require 'pact_broker/client/merge_pacts' module PactBroker module Client class PublishPacts def self.call(pact_broker_base_url, pact_file_paths, consumer_version, tags, pact_broker_client_options={}) new(pact_broker_base_url, pact_file_paths, consumer_version, tags, pact_broker_client_options).call end def initialize pact_broker_base_url, pact_file_paths, consumer_version, tags, pact_broker_client_options={} @pact_broker_base_url = pact_broker_base_url @pact_file_paths = pact_file_paths @consumer_version = consumer_version @tags = tags @pact_broker_client_options = pact_broker_client_options end def call validate $stdout.puts("") result = apply_tags && publish_pacts $stdout.puts("") result end private attr_reader :pact_broker_base_url, :pact_file_paths, :consumer_version, :tags, :pact_broker_client_options def pact_broker_client @pact_broker_client ||= PactBroker::Client::PactBrokerClient.new(base_url: pact_broker_base_url, client_options: pact_broker_client_options) end def publish_pacts pact_files.group_by(&:pact_name).collect do | pact_name, pact_files | $stdout.puts "Merging #{pact_files.collect(&:path).join(", ")}" if pact_files.size > 1 publish_pact(PactHash[merge_contents(pact_files)]) end.all? end def merge_contents(pact_files) MergePacts.call(pact_files.collect(&:pact_hash)) end def pact_files @pact_files ||= pact_file_paths.collect{ |pact_file_path| PactFile.new(pact_file_path) } end def consumer_names pact_files.collect(&:consumer_name).uniq end def publish_pact pact begin $stdout.puts "Publishing #{pact.pact_name} to pact broker at #{pact_broker_base_url}" publish_pact_contents pact rescue => e $stderr.puts "Failed to publish #{pact.pact_name} due to error: #{e.class} - #{e}" false end end def apply_tags return true if tags.nil? || tags.empty? tags.all? do | tag | tag_consumer_version tag end end def tag_consumer_version tag versions = pact_broker_client.pacticipants.versions Retry.while_error do consumer_names.collect do | consumer_name | versions.tag(pacticipant: consumer_name, version: consumer_version, tag: tag) $stdout.puts "Tagged version #{consumer_version} of #{consumer_name} as #{tag.inspect}" true end end rescue => e $stderr.puts "Failed to tag version due to error: #{e.class} - #{e}" false end def publish_pact_contents(pact) Retry.while_error do pacts = pact_broker_client.pacticipants.versions.pacts if pacts.version_published?(consumer: pact.consumer_name, provider: pact.provider_name, consumer_version: consumer_version) $stdout.puts ::Term::ANSIColor.yellow("A pact for this consumer version is already published. Overwriting. (Note: Overwriting pacts is not recommended as it can lead to race conditions. Best practice is to provide a unique consumer version number for each publication.)") end latest_pact_url = pacts.publish(pact_hash: pact, consumer_version: consumer_version) $stdout.puts "The latest version of this pact can be accessed at the following URL:\n#{latest_pact_url}" true end end def validate raise PactBroker::Client::Error.new("Please specify the consumer_version") unless (consumer_version && consumer_version.to_s.strip.size > 0) raise PactBroker::Client::Error.new("Please specify the pact_broker_base_url") unless (pact_broker_base_url && pact_broker_base_url.to_s.strip.size > 0) raise PactBroker::Client::Error.new("No pact files found") unless (pact_file_paths && pact_file_paths.any?) end end end end