Skip to content

Commit bb5aa39

Browse files
authored
fix: [ENG-1383] Validate & log invalid JSON response (#69)
* fix: don't throw exception on invalid JSON response, return error tuple
1 parent 58c59f5 commit bb5aa39

File tree

3 files changed

+67
-57
lines changed

3 files changed

+67
-57
lines changed

lib/postscript/request.ex

+17-16
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
defmodule Postscript.Request do
2-
alias Postscript.{ Config, Helpers, Operation, Response }
2+
alias Postscript.{Config, Helpers, Operation, Response}
33

44
@type t ::
55
%__MODULE__{
@@ -10,20 +10,18 @@ defmodule Postscript.Request do
1010
url: String.t()
1111
}
1212

13-
defstruct [
14-
body: nil,
15-
headers: [],
16-
method: nil,
17-
private: %{},
18-
url: nil
19-
]
13+
defstruct body: nil,
14+
headers: [],
15+
method: nil,
16+
private: %{},
17+
url: nil
2018

2119
@spec new(Operation.t(), Config.t()) :: t
2220
def new(operation, config) do
2321
body = Helpers.Body.encode!(operation, config)
2422

2523
headers = []
26-
headers = headers ++ [{ "content-type", "application/json" }]
24+
headers = headers ++ [{"content-type", "application/json"}]
2725
headers = headers ++ config.http_headers
2826

2927
url = Helpers.Url.to_string(operation, config)
@@ -49,15 +47,16 @@ defmodule Postscript.Request do
4947
|> finish(config)
5048
end
5149

52-
defp retry(response, _request, %_{ retry: retry }) when is_nil(retry) or retry == false do
50+
defp retry(response, _request, %_{retry: retry}) when is_nil(retry) or retry == false do
5351
response
5452
end
5553

56-
defp retry({ :ok, %{ status_code: status_code } } = response, request, config) when status_code >= 500 do
54+
defp retry({:ok, %{status_code: status_code}} = response, request, config)
55+
when status_code >= 500 do
5756
do_retry(response, request, config)
5857
end
5958

60-
defp retry({ :error, _ } = response, request, config) do
59+
defp retry({:error, _} = response, request, config) do
6160
do_retry(response, request, config)
6261
end
6362

@@ -89,10 +88,12 @@ defmodule Postscript.Request do
8988

9089
defp finish(response, config) do
9190
case response do
92-
{ :ok, %{ status_code: status_code } = response } when status_code >= 400 ->
93-
{ :error, Response.new(response, config) }
94-
{ :ok, %{ status_code: status_code } = response } when status_code >= 200 ->
95-
{ :ok, Response.new(response, config) }
91+
{:ok, %{status_code: status_code} = response} when status_code >= 400 ->
92+
{:error, Response.new(response, config)}
93+
94+
{:ok, %{status_code: status_code} = response} when status_code >= 200 ->
95+
{:ok, Response.new(response, config)}
96+
9697
otherwise ->
9798
otherwise
9899
end

lib/postscript/response.ex

+6-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
defmodule Postscript.Response do
2-
alias Postscript.{ Config, Http }
2+
alias Postscript.{Config, Http}
33

44
@type t ::
55
%__MODULE__{
@@ -15,7 +15,11 @@ defmodule Postscript.Response do
1515
body =
1616
response
1717
|> Map.get(:body)
18-
|> config.json_codec.decode!()
18+
|> config.json_codec.decode()
19+
|> case do
20+
{:ok, json} -> json
21+
{:error, _decode_error} -> Map.get(response, :body)
22+
end
1923

2024
%__MODULE__{}
2125
|> Map.put(:body, body)

test/postscript_test.exs

+44-39
Original file line numberDiff line numberDiff line change
@@ -1,20 +1,20 @@
11
defmodule PostscriptTest do
22
use ExUnit.Case, async: true
33

4-
alias Postscript.{ Http, Operation, Response }
4+
alias Postscript.{Http, Operation, Response}
55

6-
@ok_resp %{ body: "{\"ok\":true}", headers: [], status_code: 200 }
6+
@ok_resp %{body: "{\"ok\":true}", headers: [], status_code: 200}
77

8-
@not_ok_resp %{ body: "{\"ok\":false}", headers: [], status_code: 400 }
8+
@not_ok_resp %{body: "{\"ok\":false}", headers: [], status_code: 400}
99

1010
test "sends the proper HTTP method" do
1111
Http.Mock.start_link()
1212

13-
response = { :ok, @ok_resp }
13+
response = {:ok, @ok_resp}
1414

1515
Http.Mock.put_response(response)
1616

17-
operation = %Operation{ method: :get, params: [hello: "world"], path: "/fake" }
17+
operation = %Operation{method: :get, params: [hello: "world"], path: "/fake"}
1818

1919
Postscript.request(operation, http_client: Http.Mock)
2020

@@ -24,11 +24,11 @@ defmodule PostscriptTest do
2424
test "uses the proper URL for GET requests" do
2525
Http.Mock.start_link()
2626

27-
response = { :ok, @ok_resp }
27+
response = {:ok, @ok_resp}
2828

2929
Http.Mock.put_response(response)
3030

31-
operation = %Operation{ method: :get, params: [hello: "world"], path: "/fake" }
31+
operation = %Operation{method: :get, params: [hello: "world"], path: "/fake"}
3232

3333
Postscript.request(operation, http_client: Http.Mock)
3434

@@ -38,11 +38,11 @@ defmodule PostscriptTest do
3838
test "uses the proper URL for DELETE requests" do
3939
Http.Mock.start_link()
4040

41-
response = { :ok, @ok_resp }
41+
response = {:ok, @ok_resp}
4242

4343
Http.Mock.put_response(response)
4444

45-
operation = %Operation{ method: :delete, params: [hello: "world"], path: "/fake" }
45+
operation = %Operation{method: :delete, params: [hello: "world"], path: "/fake"}
4646

4747
Postscript.request(operation, http_client: Http.Mock)
4848

@@ -52,11 +52,11 @@ defmodule PostscriptTest do
5252
test "uses the proper URL for non-GET requests" do
5353
Http.Mock.start_link()
5454

55-
response = { :ok, @ok_resp }
55+
response = {:ok, @ok_resp}
5656

5757
Http.Mock.put_response(response)
5858

59-
operation = %Operation{ method: :post, params: [hello: "world"], path: "/fake" }
59+
operation = %Operation{method: :post, params: [hello: "world"], path: "/fake"}
6060

6161
Postscript.request(operation, http_client: Http.Mock)
6262

@@ -66,31 +66,36 @@ defmodule PostscriptTest do
6666
test "sends the proper HTTP headers" do
6767
Http.Mock.start_link()
6868

69-
response = { :ok, @ok_resp }
69+
response = {:ok, @ok_resp}
7070

7171
Http.Mock.put_response(response)
7272

7373
operation = %Operation{}
7474
operation = Map.put(operation, :method, :get)
75-
operation = Map.put(operation, :params, [hello: "world"])
75+
operation = Map.put(operation, :params, hello: "world")
7676
operation = Map.put(operation, :path, "/fake")
7777

78-
Postscript.request(operation, api_key: "thisisfake", http_client: Http.Mock, http_headers: [{ "x-custom-header", "true" }], shop_token: "thisisfake")
79-
80-
assert { "content-type", "application/json" } in Http.Mock.get_request_headers()
81-
assert { "authorization", "Bearer thisisfake" } in Http.Mock.get_request_headers()
82-
assert { "x-custom-header", "true" } in Http.Mock.get_request_headers()
83-
assert { "x-postscript-shop-token", "thisisfake" } in Http.Mock.get_request_headers()
78+
Postscript.request(operation,
79+
api_key: "thisisfake",
80+
http_client: Http.Mock,
81+
http_headers: [{"x-custom-header", "true"}],
82+
shop_token: "thisisfake"
83+
)
84+
85+
assert {"content-type", "application/json"} in Http.Mock.get_request_headers()
86+
assert {"authorization", "Bearer thisisfake"} in Http.Mock.get_request_headers()
87+
assert {"x-custom-header", "true"} in Http.Mock.get_request_headers()
88+
assert {"x-postscript-shop-token", "thisisfake"} in Http.Mock.get_request_headers()
8489
end
8590

8691
test "sends the proper body for GET requests" do
8792
Http.Mock.start_link()
8893

89-
response = { :ok, @ok_resp }
94+
response = {:ok, @ok_resp}
9095

9196
Http.Mock.put_response(response)
9297

93-
operation = %Operation{ method: :get, params: [hello: "world"], path: "/fake" }
98+
operation = %Operation{method: :get, params: [hello: "world"], path: "/fake"}
9499

95100
Postscript.request(operation, http_client: Http.Mock)
96101

@@ -100,11 +105,11 @@ defmodule PostscriptTest do
100105
test "sends the proper body for DELETE requests" do
101106
Http.Mock.start_link()
102107

103-
response = { :ok, @ok_resp }
108+
response = {:ok, @ok_resp}
104109

105110
Http.Mock.put_response(response)
106111

107-
operation = %Operation{ method: :delete, params: [hello: "world"], path: "/fake" }
112+
operation = %Operation{method: :delete, params: [hello: "world"], path: "/fake"}
108113

109114
Postscript.request(operation, http_client: Http.Mock)
110115

@@ -114,11 +119,11 @@ defmodule PostscriptTest do
114119
test "sends the proper body for non-GET requests" do
115120
Http.Mock.start_link()
116121

117-
response = { :ok, @ok_resp }
122+
response = {:ok, @ok_resp}
118123

119124
Http.Mock.put_response(response)
120125

121-
operation = %Operation{ method: :post, params: [hello: "world"], path: "/fake" }
126+
operation = %Operation{method: :post, params: [hello: "world"], path: "/fake"}
122127

123128
Postscript.request(operation, http_client: Http.Mock)
124129

@@ -128,39 +133,39 @@ defmodule PostscriptTest do
128133
test "returns :ok when the request is successful" do
129134
Http.Mock.start_link()
130135

131-
response = { :ok, @ok_resp }
136+
response = {:ok, @ok_resp}
132137

133138
Http.Mock.put_response(response)
134139

135-
operation = %Operation{ method: :post, params: [hello: "world"], path: "/fake" }
140+
operation = %Operation{method: :post, params: [hello: "world"], path: "/fake"}
136141

137142
result = Postscript.request(operation, http_client: Http.Mock)
138143

139-
assert { :ok, %Response{} } = result
144+
assert {:ok, %Response{}} = result
140145
end
141146

142147
test "returns :error when the request is not successful" do
143148
Http.Mock.start_link()
144149

145-
response = { :ok, @not_ok_resp }
150+
response = {:ok, @not_ok_resp}
146151

147152
Http.Mock.put_response(response)
148153

149-
operation = %Operation{ method: :post, params: [hello: "world"], path: "/fake" }
154+
operation = %Operation{method: :post, params: [hello: "world"], path: "/fake"}
150155

151156
result = Postscript.request(operation, http_client: Http.Mock)
152157

153-
assert { :error, %Response{} } = result
158+
assert {:error, %Response{}} = result
154159
end
155160

156161
test "passes the response through when unrecognized" do
157162
Http.Mock.start_link()
158163

159-
response = { :error, :timeout }
164+
response = {:error, :timeout}
160165

161166
Http.Mock.put_response(response)
162167

163-
operation = %Operation{ method: :post, params: [hello: "world"], path: "/fake" }
168+
operation = %Operation{method: :post, params: [hello: "world"], path: "/fake"}
164169

165170
result = Postscript.request(operation, http_client: Http.Mock)
166171

@@ -170,30 +175,30 @@ defmodule PostscriptTest do
170175
test "retries failed requests" do
171176
Http.Mock.start_link()
172177

173-
response_1 = { :error, :timeout }
174-
response_2 = { :ok, @ok_resp }
178+
response_1 = {:error, :timeout}
179+
response_2 = {:ok, @ok_resp}
175180

176181
Http.Mock.put_response(response_1)
177182
Http.Mock.put_response(response_2)
178183

179-
operation = %Operation{ method: :post, params: [hello: "world"], path: "/fake" }
184+
operation = %Operation{method: :post, params: [hello: "world"], path: "/fake"}
180185

181186
result = Postscript.request(operation, http_client: Http.Mock, retry: Postscript.Retry.Linear)
182187

183-
assert { :ok, %Response{} } = result
188+
assert {:ok, %Response{}} = result
184189
end
185190

186191
test "retries up to max attempts" do
187192
Http.Mock.start_link()
188193

189-
response = { :error, :timeout }
194+
response = {:error, :timeout}
190195

191196
Http.Mock.put_response(response)
192197
Http.Mock.put_response(response)
193198
Http.Mock.put_response(response)
194199
Http.Mock.put_response(response)
195200

196-
operation = %Operation{ method: :post, params: [hello: "world"], path: "/fake" }
201+
operation = %Operation{method: :post, params: [hello: "world"], path: "/fake"}
197202

198203
Postscript.request(operation, http_client: Http.Mock, retry: Postscript.Retry.Linear)
199204

0 commit comments

Comments
 (0)