Skip to content

Commit 9f44a79

Browse files
committed
fix: validate & log invalid JSON response
1 parent e253364 commit 9f44a79

File tree

3 files changed

+124
-57
lines changed

3 files changed

+124
-57
lines changed

lib/postscript/request.ex

+46-16
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
defmodule Postscript.Request do
2-
alias Postscript.{ Config, Helpers, Operation, Response }
2+
require Logger
3+
4+
alias Postscript.{Config, Helpers, Operation, Response}
35

46
@type t ::
57
%__MODULE__{
@@ -10,20 +12,18 @@ defmodule Postscript.Request do
1012
url: String.t()
1113
}
1214

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

2121
@spec new(Operation.t(), Config.t()) :: t
2222
def new(operation, config) do
2323
body = Helpers.Body.encode!(operation, config)
2424

2525
headers = []
26-
headers = headers ++ [{ "content-type", "application/json" }]
26+
headers = headers ++ [{"content-type", "application/json"}]
2727
headers = headers ++ config.http_headers
2828

2929
url = Helpers.Url.to_string(operation, config)
@@ -45,19 +45,47 @@ defmodule Postscript.Request do
4545

4646
request
4747
|> config.http_client.send(config.http_client_opts)
48+
|> maybe_validate_json_decode(config)
4849
|> retry(request, config)
4950
|> finish(config)
5051
end
5152

52-
defp retry(response, _request, %_{ retry: retry }) when is_nil(retry) or retry == false do
53+
defp maybe_validate_json_decode({:ok, %{body: body, headers: headers}} = response, config) do
54+
headers
55+
|> Enum.reduce(%{}, fn {k, v}, acc -> Map.put(acc, String.downcase(k), v) end)
56+
|> case do
57+
%{"content-type" => "application/json"} ->
58+
case config.json_codec.decode(body) do
59+
{:ok, _decoded} ->
60+
response
61+
62+
{:error, decode_error} ->
63+
Logger.warning([
64+
inspect(__MODULE__),
65+
" received an invalid JSON response ",
66+
inspect(body)
67+
])
68+
69+
{:error, decode_error}
70+
end
71+
72+
_otherwise ->
73+
response
74+
end
75+
end
76+
77+
defp maybe_validate_json_decode(response, _config), do: response
78+
79+
defp retry(response, _request, %_{retry: retry}) when is_nil(retry) or retry == false do
5380
response
5481
end
5582

56-
defp retry({ :ok, %{ status_code: status_code } } = response, request, config) when status_code >= 500 do
83+
defp retry({:ok, %{status_code: status_code}} = response, request, config)
84+
when status_code >= 500 do
5785
do_retry(response, request, config)
5886
end
5987

60-
defp retry({ :error, _ } = response, request, config) do
88+
defp retry({:error, _} = response, request, config) do
6189
do_retry(response, request, config)
6290
end
6391

@@ -89,10 +117,12 @@ defmodule Postscript.Request do
89117

90118
defp finish(response, config) do
91119
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) }
120+
{:ok, %{status_code: status_code} = response} when status_code >= 400 ->
121+
{:error, Response.new(response, config)}
122+
123+
{:ok, %{status_code: status_code} = response} when status_code >= 200 ->
124+
{:ok, Response.new(response, config)}
125+
96126
otherwise ->
97127
otherwise
98128
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

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

4-
alias Postscript.{ Http, Operation, Response }
4+
import ExUnit.CaptureLog, only: [with_log: 2]
55

6-
@ok_resp %{ body: "{\"ok\":true}", headers: [], status_code: 200 }
6+
alias Postscript.{Http, Operation, Response}
77

8-
@not_ok_resp %{ body: "{\"ok\":false}", headers: [], status_code: 400 }
8+
@ok_resp %{body: "{\"ok\":true}", headers: [], status_code: 200}
9+
10+
@not_ok_resp %{body: "{\"ok\":false}", headers: [], status_code: 400}
11+
12+
@not_json_resp %{
13+
body: "not json",
14+
headers: [{"content-type", "application/json"}],
15+
status_code: 200
16+
}
917

1018
test "sends the proper HTTP method" do
1119
Http.Mock.start_link()
1220

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

1523
Http.Mock.put_response(response)
1624

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

1927
Postscript.request(operation, http_client: Http.Mock)
2028

@@ -24,11 +32,11 @@ defmodule PostscriptTest do
2432
test "uses the proper URL for GET requests" do
2533
Http.Mock.start_link()
2634

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

2937
Http.Mock.put_response(response)
3038

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

3341
Postscript.request(operation, http_client: Http.Mock)
3442

@@ -38,11 +46,11 @@ defmodule PostscriptTest do
3846
test "uses the proper URL for DELETE requests" do
3947
Http.Mock.start_link()
4048

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

4351
Http.Mock.put_response(response)
4452

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

4755
Postscript.request(operation, http_client: Http.Mock)
4856

@@ -52,11 +60,11 @@ defmodule PostscriptTest do
5260
test "uses the proper URL for non-GET requests" do
5361
Http.Mock.start_link()
5462

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

5765
Http.Mock.put_response(response)
5866

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

6169
Postscript.request(operation, http_client: Http.Mock)
6270

@@ -66,31 +74,36 @@ defmodule PostscriptTest do
6674
test "sends the proper HTTP headers" do
6775
Http.Mock.start_link()
6876

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

7179
Http.Mock.put_response(response)
7280

7381
operation = %Operation{}
7482
operation = Map.put(operation, :method, :get)
75-
operation = Map.put(operation, :params, [hello: "world"])
83+
operation = Map.put(operation, :params, hello: "world")
7684
operation = Map.put(operation, :path, "/fake")
7785

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()
86+
Postscript.request(operation,
87+
api_key: "thisisfake",
88+
http_client: Http.Mock,
89+
http_headers: [{"x-custom-header", "true"}],
90+
shop_token: "thisisfake"
91+
)
92+
93+
assert {"content-type", "application/json"} in Http.Mock.get_request_headers()
94+
assert {"authorization", "Bearer thisisfake"} in Http.Mock.get_request_headers()
95+
assert {"x-custom-header", "true"} in Http.Mock.get_request_headers()
96+
assert {"x-postscript-shop-token", "thisisfake"} in Http.Mock.get_request_headers()
8497
end
8598

8699
test "sends the proper body for GET requests" do
87100
Http.Mock.start_link()
88101

89-
response = { :ok, @ok_resp }
102+
response = {:ok, @ok_resp}
90103

91104
Http.Mock.put_response(response)
92105

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

95108
Postscript.request(operation, http_client: Http.Mock)
96109

@@ -100,11 +113,11 @@ defmodule PostscriptTest do
100113
test "sends the proper body for DELETE requests" do
101114
Http.Mock.start_link()
102115

103-
response = { :ok, @ok_resp }
116+
response = {:ok, @ok_resp}
104117

105118
Http.Mock.put_response(response)
106119

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

109122
Postscript.request(operation, http_client: Http.Mock)
110123

@@ -114,11 +127,11 @@ defmodule PostscriptTest do
114127
test "sends the proper body for non-GET requests" do
115128
Http.Mock.start_link()
116129

117-
response = { :ok, @ok_resp }
130+
response = {:ok, @ok_resp}
118131

119132
Http.Mock.put_response(response)
120133

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

123136
Postscript.request(operation, http_client: Http.Mock)
124137

@@ -128,39 +141,59 @@ defmodule PostscriptTest do
128141
test "returns :ok when the request is successful" do
129142
Http.Mock.start_link()
130143

131-
response = { :ok, @ok_resp }
144+
response = {:ok, @ok_resp}
132145

133146
Http.Mock.put_response(response)
134147

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

137150
result = Postscript.request(operation, http_client: Http.Mock)
138151

139-
assert { :ok, %Response{} } = result
152+
assert {:ok, %Response{}} = result
140153
end
141154

142155
test "returns :error when the request is not successful" do
143156
Http.Mock.start_link()
144157

145-
response = { :ok, @not_ok_resp }
158+
response = {:ok, @not_ok_resp}
146159

147160
Http.Mock.put_response(response)
148161

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

151164
result = Postscript.request(operation, http_client: Http.Mock)
152165

153-
assert { :error, %Response{} } = result
166+
assert {:error, %Response{}} = result
167+
end
168+
169+
test "logs warning and returns :error when response is not valid JSON" do
170+
Http.Mock.start_link()
171+
172+
response = {:ok, @not_json_resp}
173+
174+
Http.Mock.put_response(response)
175+
176+
operation = %Operation{method: :post, params: [hello: "world"], path: "/fake"}
177+
178+
assert {result, log} =
179+
with_log([level: :warning], fn ->
180+
Postscript.request(operation, http_client: Http.Mock)
181+
end)
182+
183+
assert {:error, _error} = result
184+
185+
assert log =~ "invalid JSON response"
186+
assert log =~ "\"not json\""
154187
end
155188

156189
test "passes the response through when unrecognized" do
157190
Http.Mock.start_link()
158191

159-
response = { :error, :timeout }
192+
response = {:error, :timeout}
160193

161194
Http.Mock.put_response(response)
162195

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

165198
result = Postscript.request(operation, http_client: Http.Mock)
166199

@@ -170,30 +203,30 @@ defmodule PostscriptTest do
170203
test "retries failed requests" do
171204
Http.Mock.start_link()
172205

173-
response_1 = { :error, :timeout }
174-
response_2 = { :ok, @ok_resp }
206+
response_1 = {:error, :timeout}
207+
response_2 = {:ok, @ok_resp}
175208

176209
Http.Mock.put_response(response_1)
177210
Http.Mock.put_response(response_2)
178211

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

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

183-
assert { :ok, %Response{} } = result
216+
assert {:ok, %Response{}} = result
184217
end
185218

186219
test "retries up to max attempts" do
187220
Http.Mock.start_link()
188221

189-
response = { :error, :timeout }
222+
response = {:error, :timeout}
190223

191224
Http.Mock.put_response(response)
192225
Http.Mock.put_response(response)
193226
Http.Mock.put_response(response)
194227
Http.Mock.put_response(response)
195228

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

198231
Postscript.request(operation, http_client: Http.Mock, retry: Postscript.Retry.Linear)
199232

0 commit comments

Comments
 (0)