kiola / lib /srh /http /base_router.ex
Scott Hiett
Better error handling for failure to connect to Redis server
bb41a4c
raw
history blame
2.9 kB
defmodule Srh.Http.BaseRouter do
use Plug.Router
alias Srh.Http.RequestValidator
alias Srh.Http.CommandHandler
alias Srh.Http.ResultEncoder
plug(:match)
plug(Plug.Parsers, parsers: [:json], pass: ["application/json"], json_decoder: Jason)
plug(:dispatch)
get "/" do
handle_response({:ok, "Welcome to Serverless Redis HTTP!"}, conn)
end
post "/" do
do_command_request(conn, &CommandHandler.handle_command(&1, &2))
end
post "/pipeline" do
do_command_request(conn, &CommandHandler.handle_command_array(&1, &2))
end
post "/multi-exec" do
do_command_request(conn, &CommandHandler.handle_command_transaction_array(&1, &2))
end
match _ do
send_resp(conn, 404, "Endpoint not found")
end
defp do_command_request(conn, success_lambda) do
encoding_enabled = handle_extract_encoding?(conn)
conn
|> handle_extract_auth(&success_lambda.(conn, &1))
|> handle_encoding_step(encoding_enabled)
|> handle_response(conn)
end
defp handle_extract_auth(conn, success_lambda) do
case conn
|> get_req_header("authorization")
|> RequestValidator.validate_bearer_header() do
{:ok, token} ->
success_lambda.(token)
{:error, _} ->
{:malformed_data, "Missing/Invalid authorization header"}
end
end
defp handle_extract_encoding?(conn) do
case conn
|> get_req_header("upstash-encoding")
|> RequestValidator.validate_encoding_header() do
{:ok, _encoding_enabled} -> true
# it's not required to be present
{:error, _} -> false
end
end
defp handle_encoding_step(response, encoding_enabled) do
case encoding_enabled do
true ->
# We need to use the encoder to
ResultEncoder.encode_response(response)
false ->
response
end
end
defp handle_response(response, conn) do
%{code: code, message: message, json: json} =
case response do
{:ok, data} ->
%{code: 200, message: Jason.encode!(data), json: true}
{:not_found, message} ->
%{code: 404, message: message, json: false}
{:malformed_data, message} ->
%{code: 400, message: message, json: false}
{:redis_error, data} ->
%{code: 400, message: Jason.encode!(data), json: true}
{:not_authorized, message} ->
%{code: 401, message: message, json: false}
{:connection_error, message} ->
%{code: 500, message: message, json: false}
{:server_error, _} ->
%{code: 500, message: "An error occurred internally", json: false}
_ ->
%{code: 500, message: "An error occurred internally", json: false}
end
case json do
true ->
conn
|> put_resp_header("content-type", "application/json")
false ->
conn
end
|> send_resp(code, message)
end
end