From 57c317f3adaf7bbae1d5ace77ca2d811a5747fcb Mon Sep 17 00:00:00 2001 From: Ben Thorner Date: Fri, 15 Mar 2019 17:06:43 +0000 Subject: [PATCH 1/4] Add unreserve_path to Pub API adapter This adds a new method to the Pub API adapter to complement the new endpoint to unreserve a path in the Publishing API. --- lib/gds_api/json_client.rb | 4 +- lib/gds_api/publishing_api.rb | 9 ++++ lib/gds_api/rummager.rb | 1 + test/json_client_test.rb | 3 +- test/publishing_api_test.rb | 85 +++++++++++++++++++++++++++++++++++ 5 files changed, 99 insertions(+), 3 deletions(-) diff --git a/lib/gds_api/json_client.rb b/lib/gds_api/json_client.rb index 6e7e98e6..7f9368cf 100644 --- a/lib/gds_api/json_client.rb +++ b/lib/gds_api/json_client.rb @@ -64,8 +64,8 @@ def patch_json(url, params, additional_headers = {}) do_json_request(:patch, url, params, additional_headers) end - def delete_json(url, additional_headers = {}) - do_json_request(:delete, url, nil, additional_headers) + def delete_json(url, params = {}, additional_headers = {}) + do_json_request(:delete, url, params, additional_headers) end def delete_json_with_params!(url, params, additional_headers = {}) diff --git a/lib/gds_api/publishing_api.rb b/lib/gds_api/publishing_api.rb index 6bd41915..32faff22 100644 --- a/lib/gds_api/publishing_api.rb +++ b/lib/gds_api/publishing_api.rb @@ -12,8 +12,17 @@ def destroy_intent(base_path) e end + def unreserve_path(base_path, publishing_app) + payload = { publishing_app: publishing_app } + delete_json(unreserve_url(base_path), payload) + end + private + def unreserve_url(base_path) + "#{endpoint}/paths#{base_path}" + end + def intent_url(base_path) "#{endpoint}/publish-intent#{base_path}" end diff --git a/lib/gds_api/rummager.rb b/lib/gds_api/rummager.rb index 479eab86..81c61b1a 100644 --- a/lib/gds_api/rummager.rb +++ b/lib/gds_api/rummager.rb @@ -19,6 +19,7 @@ def add_document(type, id, document) def delete_document(type, id) delete_json( "#{documents_url}/#{id}", + nil, _type: type, ) end diff --git a/test/json_client_test.rb b/test/json_client_test.rb index 62313f14..62ee81d0 100644 --- a/test/json_client_test.rb +++ b/test/json_client_test.rb @@ -425,7 +425,8 @@ def test_client_can_set_custom_headers_on_deletes stub_request(:delete, "http://some.other.endpoint/some.json").to_return(status: 200) GdsApi::JsonClient.new.delete_json("http://some.other.endpoint/some.json", - "HEADER-A" => "B", "HEADER-C" => "D") + {}, + "HEADER-A" => "B", "HEADER-C" => "D") assert_requested(:delete, %r{/some.json}) do |request| headers_with_uppercase_names = Hash[request.headers.collect { |key, value| [key.upcase, value] }] diff --git a/test/publishing_api_test.rb b/test/publishing_api_test.rb index b5a31eac..04d0e31f 100644 --- a/test/publishing_api_test.rb +++ b/test/publishing_api_test.rb @@ -11,6 +11,91 @@ @api_client = GdsApi::PublishingApi.new(publishing_api_host) end + describe "#unreserve_path" do + it "responds with 200 OK if reservation is owned by the app" do + publishing_app = "publisher" + base_path = "/test-item" + + publishing_api + .given("/test-item has been reserved by the Publisher application") + .upon_receiving("a request to unreserve a path") + .with( + method: :delete, + path: "/paths#{base_path}", + body: { publishing_app: publishing_app }, + headers: { + "Content-Type" => "application/json" + }, + ) + .will_respond_with( + status: 200, + body: {}, + headers: { + "Content-Type" => "application/json; charset=utf-8" + }, + ) + + response = @api_client.unreserve_path(base_path, publishing_app) + assert_equal 200, response.code + end + + it "raises an error if the reservation does not exist" do + publishing_app = "publisher" + base_path = "/test-item" + + publishing_api + .given("no content exists") + .upon_receiving("a request to unreserve a non-existant path") + .with( + method: :delete, + path: "/paths#{base_path}", + body: { publishing_app: publishing_app }, + headers: { + "Content-Type" => "application/json" + }, + ) + .will_respond_with( + status: 404, + body: {}, + headers: { + "Content-Type" => "application/json; charset=utf-8" + }, + ) + + assert_raises GdsApi::HTTPNotFound do + @api_client.unreserve_path(base_path, publishing_app) + end + end + + it "raises an error if the reservation is with another app" do + publishing_app = "whitehall" + base_path = "/test-item" + + publishing_api + .given("/test-item has been reserved by the Publisher application") + .upon_receiving("a request to unreserve a path owned by another app") + .with( + method: :delete, + path: "/paths#{base_path}", + body: { publishing_app: "whitehall" }, + headers: { + "Content-Type" => "application/json" + }, + ) + .will_respond_with( + status: 422, + body: {}, + headers: { + "Content-Type" => "application/json; charset=utf-8" + }, + ) + + assert_raises GdsApi::HTTPUnprocessableEntity do + @api_client.unreserve_path(base_path, publishing_app) + end + end + end + describe "#put_intent" do it "responds with 200 OK if publish intent is valid" do base_path = "/test-intent" From c7be09afd85145b34d9efa4432041c35499dc2b0 Mon Sep 17 00:00:00 2001 From: Ben Thorner Date: Fri, 15 Mar 2019 17:10:26 +0000 Subject: [PATCH 2/4] Add test helpers to complement new adapter method --- lib/gds_api/test_helpers/publishing_api.rb | 18 ++++++++++++++ test/publishing_api_test.rb | 12 +++++----- test/test_helpers/publishing_api_test.rb | 28 ++++++++++++++++++++++ 3 files changed, 52 insertions(+), 6 deletions(-) diff --git a/lib/gds_api/test_helpers/publishing_api.rb b/lib/gds_api/test_helpers/publishing_api.rb index 0da22ebc..96350408 100644 --- a/lib/gds_api/test_helpers/publishing_api.rb +++ b/lib/gds_api/test_helpers/publishing_api.rb @@ -11,6 +11,18 @@ module PublishingApi PUBLISHING_API_ENDPOINT = Plek.current.find('publishing-api') + def stub_publishing_api_unreserve_path(base_path, publishing_app) + stub_publishing_api_unreserve_path_with_code(base_path, publishing_app, 200) + end + + def stub_publishing_api_unreserve_path_not_found(base_path, publishing_app) + stub_publishing_api_unreserve_path_with_code(base_path, publishing_app, 404) + end + + def stub_publishing_api_unreserve_path_invalid(base_path, publishing_app) + stub_publishing_api_unreserve_path_with_code(base_path, publishing_app, 422) + end + def stub_publishing_api_put_intent(base_path, body = intent_for_publishing_api(base_path)) url = PUBLISHING_API_ENDPOINT + "/publish-intent" + base_path body = body.to_json unless body.is_a?(String) @@ -101,6 +113,12 @@ def stub_publishing_api_returns_path_reservation_validation_error_for(path, erro private + def stub_publishing_api_unreserve_path_with_code(base_path, publishing_app, code) + url = PUBLISHING_API_ENDPOINT + "/paths" + base_path + body = { publishing_app: publishing_app } + stub_request(:delete, url).with(body: body).to_return(status: code, body: '{}', headers: { "Content-Type" => "application/json; charset=utf-8" }) + end + def values_match_recursively(expected_value, actual_value) case expected_value when Hash diff --git a/test/publishing_api_test.rb b/test/publishing_api_test.rb index 04d0e31f..6b40a38f 100644 --- a/test/publishing_api_test.rb +++ b/test/publishing_api_test.rb @@ -62,9 +62,9 @@ }, ) - assert_raises GdsApi::HTTPNotFound do - @api_client.unreserve_path(base_path, publishing_app) - end + assert_raises GdsApi::HTTPNotFound do + @api_client.unreserve_path(base_path, publishing_app) + end end it "raises an error if the reservation is with another app" do @@ -90,9 +90,9 @@ }, ) - assert_raises GdsApi::HTTPUnprocessableEntity do - @api_client.unreserve_path(base_path, publishing_app) - end + assert_raises GdsApi::HTTPUnprocessableEntity do + @api_client.unreserve_path(base_path, publishing_app) + end end end diff --git a/test/test_helpers/publishing_api_test.rb b/test/test_helpers/publishing_api_test.rb index af56c993..83987d93 100644 --- a/test/test_helpers/publishing_api_test.rb +++ b/test/test_helpers/publishing_api_test.rb @@ -7,6 +7,34 @@ let(:base_api_url) { Plek.current.find("publishing-api") } let(:publishing_api) { GdsApi::PublishingApi.new(base_api_url) } + describe "#stub_publishing_api_unreserve_path" do + it "stubs the unreserve path API call" do + stub_publishing_api_unreserve_path("/foo", "myapp") + api_response = publishing_api.unreserve_path("/foo", "myapp") + assert_equal(api_response.code, 200) + end + end + + describe "#stub_publishing_api_unreserve_path_not_found" do + it "stubs the unreserve path API call" do + stub_publishing_api_unreserve_path_not_found("/foo", "myapp") + + assert_raises GdsApi::HTTPNotFound do + publishing_api.unreserve_path("/foo", "myapp") + end + end + end + + describe "#stub_publishing_api_unreserve_path_invalid" do + it "stubs the unreserve path API call" do + stub_publishing_api_unreserve_path_invalid("/foo", "myapp") + + assert_raises GdsApi::HTTPUnprocessableEntity do + publishing_api.unreserve_path("/foo", "myapp") + end + end + end + describe '#request_json_matching predicate' do describe "nested required attribute" do let(:matcher) { request_json_matching("a" => { "b" => 1 }) } From b3e48d8c2ae130e5ae555f9d251699db545c6b4c Mon Sep 17 00:00:00 2001 From: Ben Thorner Date: Fri, 15 Mar 2019 18:05:51 +0000 Subject: [PATCH 3/4] Remove redundant deprecation warning This was introduced in 2e19c7a0, but the referenced RFC has since been superceded, and it is now accepted to have a body in a DELETE request. https://tools.ietf.org/html/rfc7231#page-29 --- lib/gds_api/base.rb | 1 - lib/gds_api/json_client.rb | 12 ------------ 2 files changed, 13 deletions(-) diff --git a/lib/gds_api/base.rb b/lib/gds_api/base.rb index d077542f..1ab267a7 100644 --- a/lib/gds_api/base.rb +++ b/lib/gds_api/base.rb @@ -23,7 +23,6 @@ def create_client :put_json, :patch_json, :delete_json, - :delete_json_with_params!, :get_raw, :get_raw!, :put_multipart, :post_multipart diff --git a/lib/gds_api/json_client.rb b/lib/gds_api/json_client.rb index 7f9368cf..e1d26936 100644 --- a/lib/gds_api/json_client.rb +++ b/lib/gds_api/json_client.rb @@ -68,18 +68,6 @@ def delete_json(url, params = {}, additional_headers = {}) do_json_request(:delete, url, params, additional_headers) end - def delete_json_with_params!(url, params, additional_headers = {}) - warn <<-doc - DEPRECATION NOTICE: Delete requests should not include parameters. - - Do not use this method as the ability to do this will be removed. - - Called from: #{caller[2]} - doc - - do_json_request(:delete, url, params, additional_headers) - end - def post_multipart(url, params) r = do_raw_request(:post, url, params.merge(multipart: true)) Response.new(r) From bc4eaa72d80b20bdadb5ea7fa8688e9898d4e18b Mon Sep 17 00:00:00 2001 From: Ben Thorner Date: Tue, 19 Mar 2019 12:09:28 +0000 Subject: [PATCH 4/4] Make publishing_app optional in unreserve stub This reduces the amount of boilerplate we need to write when testing against this feature. --- lib/gds_api/test_helpers/publishing_api.rb | 6 +++--- test/test_helpers/publishing_api_test.rb | 22 ++++++++++++++++++++++ 2 files changed, 25 insertions(+), 3 deletions(-) diff --git a/lib/gds_api/test_helpers/publishing_api.rb b/lib/gds_api/test_helpers/publishing_api.rb index 96350408..bc860fab 100644 --- a/lib/gds_api/test_helpers/publishing_api.rb +++ b/lib/gds_api/test_helpers/publishing_api.rb @@ -11,15 +11,15 @@ module PublishingApi PUBLISHING_API_ENDPOINT = Plek.current.find('publishing-api') - def stub_publishing_api_unreserve_path(base_path, publishing_app) + def stub_publishing_api_unreserve_path(base_path, publishing_app = /.*/) stub_publishing_api_unreserve_path_with_code(base_path, publishing_app, 200) end - def stub_publishing_api_unreserve_path_not_found(base_path, publishing_app) + def stub_publishing_api_unreserve_path_not_found(base_path, publishing_app = /.*/) stub_publishing_api_unreserve_path_with_code(base_path, publishing_app, 404) end - def stub_publishing_api_unreserve_path_invalid(base_path, publishing_app) + def stub_publishing_api_unreserve_path_invalid(base_path, publishing_app = /.*/) stub_publishing_api_unreserve_path_with_code(base_path, publishing_app, 422) end diff --git a/test/test_helpers/publishing_api_test.rb b/test/test_helpers/publishing_api_test.rb index 83987d93..bda872a9 100644 --- a/test/test_helpers/publishing_api_test.rb +++ b/test/test_helpers/publishing_api_test.rb @@ -13,6 +13,12 @@ api_response = publishing_api.unreserve_path("/foo", "myapp") assert_equal(api_response.code, 200) end + + it "stubs for any app if not specified" do + stub_publishing_api_unreserve_path("/foo") + api_response = publishing_api.unreserve_path("/foo", "myapp") + assert_equal(api_response.code, 200) + end end describe "#stub_publishing_api_unreserve_path_not_found" do @@ -23,6 +29,14 @@ publishing_api.unreserve_path("/foo", "myapp") end end + + it "stubs for any app if not specified" do + stub_publishing_api_unreserve_path_not_found("/foo") + + assert_raises GdsApi::HTTPNotFound do + publishing_api.unreserve_path("/foo", "myapp") + end + end end describe "#stub_publishing_api_unreserve_path_invalid" do @@ -33,6 +47,14 @@ publishing_api.unreserve_path("/foo", "myapp") end end + + it "stubs for any app if not specified" do + stub_publishing_api_unreserve_path_invalid("/foo") + + assert_raises GdsApi::HTTPUnprocessableEntity do + publishing_api.unreserve_path("/foo", "myapp") + end + end end describe '#request_json_matching predicate' do