From f908268d55edcee65a31e4b13d67840567b495ce Mon Sep 17 00:00:00 2001 From: Ivan Rabotyaga Date: Tue, 5 Oct 2021 15:54:58 +0300 Subject: [PATCH] Add participant label options to webhook create/update commands --- .../Pact Broker Client - Pact Broker.md | 112 ++++++++++++++++ .../client/cli/webhook_commands.rb | 17 ++- lib/pact_broker/client/webhooks/create.rb | 4 + .../cli/broker_run_webhook_commands_spec.rb | 55 ++++++++ .../pacts/pact_broker_client-pact_broker.json | 126 ++++++++++++++++++ .../service_providers/webhooks_create_spec.rb | 50 +++++++ 6 files changed, 363 insertions(+), 1 deletion(-) diff --git a/doc/pacts/markdown/Pact Broker Client - Pact Broker.md b/doc/pacts/markdown/Pact Broker Client - Pact Broker.md index e2e1ce9d..97b71fda 100644 --- a/doc/pacts/markdown/Pact Broker Client - Pact Broker.md +++ b/doc/pacts/markdown/Pact Broker Client - Pact Broker.md @@ -64,10 +64,14 @@ * [A request to create a webhook with a JSON body for a consumer and provider](#a_request_to_create_a_webhook_with_a_JSON_body_for_a_consumer_and_provider_given_the_'Pricing_Service'_and_'Condor'_already_exist_in_the_pact-broker) given the 'Pricing Service' and 'Condor' already exist in the pact-broker +* [A request to create a webhook with a JSON body for a consumer specified by a label](#a_request_to_create_a_webhook_with_a_JSON_body_for_a_consumer_specified_by_a_label_given_the_'Pricing_Service'_and_'Condor'_already_exist_in_the_pact-broker) given the 'Pricing Service' and 'Condor' already exist in the pact-broker + * [A request to create a webhook with a JSON body for a consumer that does not exist](#a_request_to_create_a_webhook_with_a_JSON_body_for_a_consumer_that_does_not_exist) * [A request to create a webhook with a JSON body for a provider](#a_request_to_create_a_webhook_with_a_JSON_body_for_a_provider_given_the_'Pricing_Service'_and_'Condor'_already_exist_in_the_pact-broker) given the 'Pricing Service' and 'Condor' already exist in the pact-broker +* [A request to create a webhook with a JSON body for a provider specified by a label](#a_request_to_create_a_webhook_with_a_JSON_body_for_a_provider_specified_by_a_label_given_the_'Pricing_Service'_and_'Condor'_already_exist_in_the_pact-broker) given the 'Pricing Service' and 'Condor' already exist in the pact-broker + * [A request to create a webhook with a non-JSON body for a consumer and provider](#a_request_to_create_a_webhook_with_a_non-JSON_body_for_a_consumer_and_provider_given_the_'Pricing_Service'_and_'Condor'_already_exist_in_the_pact-broker) given the 'Pricing Service' and 'Condor' already exist in the pact-broker * [A request to create a webhook with every possible event type](#a_request_to_create_a_webhook_with_every_possible_event_type_given_the_'Pricing_Service'_and_'Condor'_already_exist_in_the_pact-broker) given the 'Pricing Service' and 'Condor' already exist in the pact-broker @@ -1336,6 +1340,60 @@ Pact Broker will respond with: } } ``` + +Given **the 'Pricing Service' and 'Condor' already exist in the pact-broker**, upon receiving **a request to create a webhook with a JSON body for a consumer specified by a label** from Pact Broker Client, with +```json +{ + "method": "post", + "path": "/HAL-REL-PLACEHOLDER-PB-WEBHOOKS", + "headers": { + "Content-Type": "application/json", + "Accept": "application/hal+json" + }, + "body": { + "description": "a webhook", + "events": [ + { + "name": "contract_content_changed" + } + ], + "request": { + "url": "https://webhook", + "method": "POST", + "headers": { + "Foo": "bar", + "Bar": "foo" + }, + "body": { + "some": "body" + }, + "username": "username", + "password": "password" + }, + "consumer": { + "label": "consumer_label" + } + } +} +``` +Pact Broker will respond with: +```json +{ + "status": 201, + "headers": { + "Content-Type": "application/hal+json;charset=utf-8" + }, + "body": { + "description": "a webhook", + "_links": { + "self": { + "href": "http://localhost:1234/some-url", + "title": "A title" + } + } + } +} +``` Upon receiving **a request to create a webhook with a JSON body for a consumer that does not exist** from Pact Broker Client, with ```json @@ -1442,6 +1500,60 @@ Pact Broker will respond with: } } ``` + +Given **the 'Pricing Service' and 'Condor' already exist in the pact-broker**, upon receiving **a request to create a webhook with a JSON body for a provider specified by a label** from Pact Broker Client, with +```json +{ + "method": "post", + "path": "/HAL-REL-PLACEHOLDER-PB-WEBHOOKS", + "headers": { + "Content-Type": "application/json", + "Accept": "application/hal+json" + }, + "body": { + "description": "a webhook", + "events": [ + { + "name": "contract_content_changed" + } + ], + "request": { + "url": "https://webhook", + "method": "POST", + "headers": { + "Foo": "bar", + "Bar": "foo" + }, + "body": { + "some": "body" + }, + "username": "username", + "password": "password" + }, + "provider": { + "label": "provider_label" + } + } +} +``` +Pact Broker will respond with: +```json +{ + "status": 201, + "headers": { + "Content-Type": "application/hal+json;charset=utf-8" + }, + "body": { + "description": "a webhook", + "_links": { + "self": { + "href": "http://localhost:1234/some-url", + "title": "A title" + } + } + } +} +``` Given **the 'Pricing Service' and 'Condor' already exist in the pact-broker**, upon receiving **a request to create a webhook with a non-JSON body for a consumer and provider** from Pact Broker Client, with ```json diff --git a/lib/pact_broker/client/cli/webhook_commands.rb b/lib/pact_broker/client/cli/webhook_commands.rb index 86017cbd..5c940a44 100644 --- a/lib/pact_broker/client/cli/webhook_commands.rb +++ b/lib/pact_broker/client/cli/webhook_commands.rb @@ -15,7 +15,9 @@ def self.shared_options_for_webhook_commands method_option :data, aliases: "-d", desc: "Webhook payload (file or string)" method_option :user, aliases: "-u", desc: "Webhook basic auth username and password eg. username:password" method_option :consumer, desc: "Consumer name" + method_option :consumer_label, desc: "Consumer label, mutually exclusive with consumer name" method_option :provider, desc: "Provider name" + method_option :provider_label, desc: "Provider label, mutually exclusive with provider name" method_option :description, desc: "Webhook description" method_option :contract_content_changed, type: :boolean, desc: "Trigger this webhook when the pact content changes" method_option :contract_published, type: :boolean, desc: "Trigger this webhook when a pact is published" @@ -70,6 +72,7 @@ def parse_webhook_events end def parse_webhook_options(webhook_url) + validate_mutual_exclusiveness_of_participant_name_and_label_options events = parse_webhook_events # TODO update for contract_requiring_verification_published when released @@ -102,7 +105,9 @@ def parse_webhook_options(webhook_url) password: password, body: body, consumer: options.consumer, + consumer_label: options.consumer_label, provider: options.provider, + provider_label: options.provider_label, events: events, team_uuid: options.team_uuid } @@ -118,10 +123,20 @@ def run_webhook_commands webhook_url rescue PactBroker::Client::Error => e raise WebhookCreationError, "#{e.class} - #{e.message}" end + + def validate_mutual_exclusiveness_of_participant_name_and_label_options + if options.consumer && options.consumer_label + raise WebhookCreationError.new("Consumer name (--consumer) and label (--consumer_label) options are mutually exclusive") + end + + if options.provider && options.provider_label + raise WebhookCreationError.new("Provider name (--provider) and label (--provider_label) options are mutually exclusive") + end + end end end end end end end -end \ No newline at end of file +end diff --git a/lib/pact_broker/client/webhooks/create.rb b/lib/pact_broker/client/webhooks/create.rb index 5d97ebf4..f90381b2 100644 --- a/lib/pact_broker/client/webhooks/create.rb +++ b/lib/pact_broker/client/webhooks/create.rb @@ -81,10 +81,14 @@ def request_body_with_optional_consumer_and_provider if params.consumer body[:consumer] = { name: params.consumer } + elsif params.consumer_label + body[:consumer] = { label: params.consumer_label } end if params.provider body[:provider] = { name: params.provider } + elsif params.provider_label + body[:provider] = { label: params.provider_label } end if params.team_uuid diff --git a/spec/lib/pact_broker/client/cli/broker_run_webhook_commands_spec.rb b/spec/lib/pact_broker/client/cli/broker_run_webhook_commands_spec.rb index 22ac5442..7818becc 100644 --- a/spec/lib/pact_broker/client/cli/broker_run_webhook_commands_spec.rb +++ b/spec/lib/pact_broker/client/cli/broker_run_webhook_commands_spec.rb @@ -49,7 +49,9 @@ module CLI password: "password", body: "data", consumer: "consumer", + consumer_label: nil, provider: "provider", + provider_label: nil, events: ["contract_content_changed"], team_uuid: "1234" }.tap { |it| Pact::Fixture.add_fixture(:create_webhook_params, it) } @@ -223,6 +225,59 @@ module CLI expect { subject }.to raise_error(WebhookCreationError, /foo/) end end + + end + + context "when both consumer name and label options are specified" do + before do + options_hash[:consumer_label] = "consumer_label" + broker.options = OpenStruct.new(options_hash) + end + + it "raises a WebhookCreationError" do + expect { subject }.to raise_error( + WebhookCreationError, + "Consumer name (--consumer) and label (--consumer_label) options are mutually exclusive" + ) + end + end + + context "when both provider name and label options are specified" do + before do + options_hash[:provider_label] = "provider_label" + broker.options = OpenStruct.new(options_hash) + end + + it "raises a WebhookCreationError" do + expect { subject }.to raise_error( + WebhookCreationError, + "Provider name (--provider) and label (--provider_label) options are mutually exclusive" + ) + end + end + + context "when participant labels are specified" do + before do + options_hash.delete(:consumer) + options_hash.delete(:provider) + options_hash.merge!(consumer_label: 'consumer_label', provider_label: 'provider_label') + expected_params.merge!( + consumer: nil, + consumer_label: 'consumer_label', + provider: nil, + provider_label: 'provider_label' + ) + + broker.options = OpenStruct.new(options_hash) + end + + it "calls PactBroker::Client::Webhooks::Create with participant labels in params" do + expect(PactBroker::Client::Webhooks::Create).to receive(:call) do | params, _, _ | + expect(params).to eq expected_params + command_result + end + subject + end end end end diff --git a/spec/pacts/pact_broker_client-pact_broker.json b/spec/pacts/pact_broker_client-pact_broker.json index dfe19d95..f0bf1997 100644 --- a/spec/pacts/pact_broker_client-pact_broker.json +++ b/spec/pacts/pact_broker_client-pact_broker.json @@ -2344,6 +2344,69 @@ } } }, + { + "description": "a request to create a webhook with a JSON body for a consumer specified by a label", + "providerState": "the 'Pricing Service' and 'Condor' already exist in the pact-broker", + "request": { + "method": "post", + "path": "/HAL-REL-PLACEHOLDER-PB-WEBHOOKS", + "headers": { + "Content-Type": "application/json", + "Accept": "application/hal+json" + }, + "body": { + "description": "a webhook", + "events": [ + { + "name": "contract_content_changed" + } + ], + "request": { + "url": "https://webhook", + "method": "POST", + "headers": { + "Foo": "bar", + "Bar": "foo" + }, + "body": { + "some": "body" + }, + "username": "username", + "password": "password" + }, + "consumer": { + "label": "consumer_label" + } + } + }, + "response": { + "status": 201, + "headers": { + "Content-Type": "application/hal+json;charset=utf-8" + }, + "body": { + "description": "a webhook", + "_links": { + "self": { + "href": "http://localhost:1234/some-url", + "title": "A title" + } + } + }, + "matchingRules": { + "$.body.description": { + "match": "type" + }, + "$.body._links.self.href": { + "match": "regex", + "regex": "http:\\/\\/.*" + }, + "$.body._links.self.title": { + "match": "type" + } + } + } + }, { "description": "a request to create a webhook with a JSON body for a consumer that does not exist", "request": { @@ -2463,6 +2526,69 @@ } } }, + { + "description": "a request to create a webhook with a JSON body for a provider specified by a label", + "providerState": "the 'Pricing Service' and 'Condor' already exist in the pact-broker", + "request": { + "method": "post", + "path": "/HAL-REL-PLACEHOLDER-PB-WEBHOOKS", + "headers": { + "Content-Type": "application/json", + "Accept": "application/hal+json" + }, + "body": { + "description": "a webhook", + "events": [ + { + "name": "contract_content_changed" + } + ], + "request": { + "url": "https://webhook", + "method": "POST", + "headers": { + "Foo": "bar", + "Bar": "foo" + }, + "body": { + "some": "body" + }, + "username": "username", + "password": "password" + }, + "provider": { + "label": "provider_label" + } + } + }, + "response": { + "status": 201, + "headers": { + "Content-Type": "application/hal+json;charset=utf-8" + }, + "body": { + "description": "a webhook", + "_links": { + "self": { + "href": "http://localhost:1234/some-url", + "title": "A title" + } + } + }, + "matchingRules": { + "$.body.description": { + "match": "type" + }, + "$.body._links.self.href": { + "match": "regex", + "regex": "http:\\/\\/.*" + }, + "$.body._links.self.title": { + "match": "type" + } + } + } + }, { "description": "a request to create a global webhook with a JSON body", "request": { diff --git a/spec/service_providers/webhooks_create_spec.rb b/spec/service_providers/webhooks_create_spec.rb index ad9119d8..db95bfaf 100644 --- a/spec/service_providers/webhooks_create_spec.rb +++ b/spec/service_providers/webhooks_create_spec.rb @@ -207,6 +207,31 @@ end end + context "when consumer is specified using a label" do + before do + params.delete(:consumer) + params.delete(:provider) + params.merge!(consumer_label: "consumer_label") + request_body["consumer"] = { "label" => "consumer_label" } + mock_pact_broker_index(self) + + pact_broker + .given("the 'Pricing Service' and 'Condor' already exist in the pact-broker") + .upon_receiving("a request to create a webhook with a JSON body for a consumer specified by a label") + .with( + method: :post, + path: placeholder_path('pb:webhooks'), + headers: post_request_headers, + body: request_body) + .will_respond_with(success_response) + end + + it "returns a CommandResult with success = true" do + expect(subject.success).to be true + expect(subject.message).to eq "Webhook \"a webhook\" created" + end + end + context "when only a consumer is specified and it does not exist" do before do params.delete(:provider) @@ -257,6 +282,31 @@ end end + context "when provider is specified using a label" do + before do + params.delete(:consumer) + params.delete(:provider) + params.merge!(provider_label: "provider_label") + request_body["provider"] = { "label" => "provider_label" } + mock_pact_broker_index(self) + + pact_broker + .given("the 'Pricing Service' and 'Condor' already exist in the pact-broker") + .upon_receiving("a request to create a webhook with a JSON body for a provider specified by a label") + .with( + method: :post, + path: placeholder_path('pb:webhooks'), + headers: post_request_headers, + body: request_body) + .will_respond_with(success_response) + end + + it "returns a CommandResult with success = true" do + expect(subject.success).to be true + expect(subject.message).to eq "Webhook \"a webhook\" created" + end + end + context "when neither consumer nor provider are specified" do before do params.delete(:consumer)