diff --git a/CHANGELOG.md b/CHANGELOG.md index 91963de..6f20f1d 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -2,6 +2,13 @@ This file tracks all the changes (https://keepachangelog.com/en/1.0.0/) made to the client. This allows parsers such as Dependabot to provide a clean overview in pull requests. +## [v0.12.0] - 2024-08-07 + +#### Added + +- Added `quotation!`, `invoice!`, `distribute_tax!`, `tax_area!` class methods to `VertexClient` that raise `Savon::SOAPFault` errors when errors are returned from the Vertex SOAP API + - Existing non-bang methods receive `nil` internally and return `fallback_response` defined in each resource + ## [v0.11.0] - 2024-04-03 #### Changed diff --git a/lib/vertex_client.rb b/lib/vertex_client.rb index 82ef774..30acb65 100644 --- a/lib/vertex_client.rb +++ b/lib/vertex_client.rb @@ -65,18 +65,34 @@ def quotation(payload) Resource::Quotation.new(payload).result end + def quotation!(payload) + Resource::Quotation.new(payload).result! + end + def invoice(payload) Resource::Invoice.new(payload).result end + def invoice!(payload) + Resource::Invoice.new(payload).result! + end + def distribute_tax(payload) Resource::DistributeTax.new(payload).result end + def distribute_tax!(payload) + Resource::DistributeTax.new(payload).result! + end + def tax_area(payload) Resource::TaxArea.new(payload).result end + def tax_area!(payload) + Resource::TaxArea.new(payload).result! + end + def circuit return unless circuit_configured? diff --git a/lib/vertex_client/connection.rb b/lib/vertex_client/connection.rb index 06c6d94..a505e6a 100644 --- a/lib/vertex_client/connection.rb +++ b/lib/vertex_client/connection.rb @@ -11,10 +11,13 @@ def initialize(endpoint, resource_key=nil) def request(payload) call_with_circuit_if_available do - client.call( - :vertex_envelope, - message: shell_with_auth.merge(payload) - ) + raw_request(payload) + end + end + + def request!(payload) + call_with_circuit_if_available(raise_exceptions: true) do + raw_request(payload) end end @@ -32,6 +35,13 @@ def client private + def raw_request(payload) + client.call( + :vertex_envelope, + message: shell_with_auth.merge(payload) + ) + end + def config @config ||= VertexClient.configuration end @@ -40,14 +50,20 @@ def resource_config config.resource_config[@resource_key] || {} end - def call_with_circuit_if_available + def call_with_circuit_if_available(raise_exceptions: false) if VertexClient.circuit - VertexClient.circuit.run(exception: false) { yield } + begin + VertexClient.circuit.run(exception: raise_exceptions) { yield } + rescue Circuitbox::ServiceFailureError => e + nil + raise e.original if raise_exceptions + end else begin yield - rescue => _e + rescue => e nil + raise e if raise_exceptions end end end diff --git a/lib/vertex_client/resources/base.rb b/lib/vertex_client/resources/base.rb index 26634a2..962c430 100644 --- a/lib/vertex_client/resources/base.rb +++ b/lib/vertex_client/resources/base.rb @@ -2,31 +2,44 @@ module VertexClient module Resource class Base + class NotImplementedError < StandardError; end def initialize(params) @payload = payload_type.new(params) end def result - @result ||= (response ? formatted_response : fallback_response) + @result ||= (formatted_response(response) || fallback_response) + end + + def result! + formatted_response(response!) end def config_key demodulized_class_name.underscore.to_sym end + def fallback_response + raise NotImplementedError + end + private def response - @response ||= connection.request(@payload.transform) + connection.request(@payload.transform) + end + + def response! + connection.request!(@payload.transform) end def connection @connection ||= Connection.new(self.class::ENDPOINT, config_key) end - def formatted_response - response_type.new(response) + def formatted_response(raw_response) + response_type.new(raw_response) if raw_response end def payload_type diff --git a/lib/vertex_client/version.rb b/lib/vertex_client/version.rb index 9646930..68c0311 100644 --- a/lib/vertex_client/version.rb +++ b/lib/vertex_client/version.rb @@ -1,3 +1,3 @@ module VertexClient - VERSION = '0.11.0' + VERSION = '0.12.0' end diff --git a/test/resources/base_test.rb b/test/resources/base_test.rb index c1a2be9..0c74a9d 100644 --- a/test/resources/base_test.rb +++ b/test/resources/base_test.rb @@ -3,6 +3,10 @@ describe VertexClient::Resource::Base do class VertexClient::Resource::MyTest < VertexClient::Resource::Base ENDPOINT = 'MyEndPoint'.freeze + + def fallback_response + { fall: :back } + end end class VertexClient::Payload::MyTest < VertexClient::Payload::Base def validate! @@ -51,11 +55,33 @@ class VertexClient::Response::MyTest < VertexClient::Response::Base; end; end end + describe 'result!' do + describe 'without exceptions' do + before do + resource.stubs(:response!).returns({ test: :ok }) + end + + it 'returns a formatted response' do + resource.expects(:formatted_response) + resource.result! + end + end + + describe 'with exceptions' do + before do + resource.stubs(:response!).raises(StandardError) + end + + it 'raises the exception' do + assert_raises(StandardError) { resource.result! } + end + end + end + describe 'formatted_response' do it 'returns a new response of the proper type' do - resource.expects(:response).returns({test: :ok}) VertexClient::Response::MyTest.expects(:new).with({test: :ok}) - resource.send(:formatted_response) + resource.send(:formatted_response, {test: :ok}) end end end diff --git a/test/vertex_client_test.rb b/test/vertex_client_test.rb index 4587a8a..8f91d74 100644 --- a/test/vertex_client_test.rb +++ b/test/vertex_client_test.rb @@ -23,6 +23,13 @@ assert VertexClient.quotation(working_quote_params) end + it 'does a quotation!' do + VertexClient::Resource::Quotation.expects(:new) + .with(working_quote_params) + .returns(stub(result!: true)) + assert VertexClient.quotation!(working_quote_params) + end + it 'does invoice' do VertexClient::Resource::Invoice.expects(:new) .with(working_quote_params) @@ -30,6 +37,13 @@ assert VertexClient.invoice(working_quote_params) end + it 'does invoice!' do + VertexClient::Resource::Invoice.expects(:new) + .with(working_quote_params) + .returns(stub(result!: true)) + assert VertexClient.invoice!(working_quote_params) + end + it 'does distribute_tax' do VertexClient::Resource::DistributeTax.expects(:new) .with(working_quote_params) @@ -37,6 +51,13 @@ assert VertexClient.distribute_tax(working_quote_params) end + it 'does distribute_tax!' do + VertexClient::Resource::DistributeTax.expects(:new) + .with(working_quote_params) + .returns(stub(result!: true)) + assert VertexClient.distribute_tax!(working_quote_params) + end + it 'does tax_area' do VertexClient::Resource::TaxArea.expects(:new) .with(working_quote_params) @@ -44,6 +65,13 @@ assert VertexClient.tax_area(working_quote_params) end + it 'does tax_area!' do + VertexClient::Resource::TaxArea.expects(:new) + .with(working_quote_params) + .returns(stub(result!: true)) + assert VertexClient.tax_area!(working_quote_params) + end + describe 'circuit' do it 'only exists if circuit_config is provided to configuration' do VertexClient.configuration.circuit_config = nil