First working iteration of shortnr api

This commit is contained in:
mitchell 2019-12-08 13:44:04 -05:00
parent 7ff70d452a
commit 9ee9eddbfa
8 changed files with 147 additions and 12 deletions

3
.gitignore vendored
View File

@ -24,3 +24,6 @@ service-*.tar
# Ignore elixir-ls artifacts # Ignore elixir-ls artifacts
/.elixir_ls/ /.elixir_ls/
# Ignore development DETS file
urls

View File

@ -4,32 +4,45 @@ defmodule Shortnr.Router do
require Logger require Logger
alias Shortnr.Transport.{Json, Http} alias Shortnr.Transport.{Text, Http}
alias Shortnr.Url alias Shortnr.URL
plug(Plug.Logger, log: :debug) plug(Plug.Logger, log: :debug)
plug(:match) plug(:match)
plug(:dispatch) plug(:dispatch)
post "/urls" do post "/urls/:url" do
conn {:ok, url, conn}
|> Json.decode_request(Url.CreateRequest) |> Http.handle(&URL.create(&1, URL.Repo.DETS))
|> Http.handle(&Url.create(&1, Url.Repo.DETS)) |> Text.encode_response()
|> Json.encode_response()
|> Http.send(:created, conn) |> Http.send(:created, conn)
end end
get "/urls" do
{:ok, :ignore, conn}
|> Http.handle(fn -> URL.list(URL.Repo.DETS) end)
|> Text.encode_response()
|> Http.send(:ok, conn)
end
get "/:id" do
{:ok, id, conn}
|> Http.handle(&URL.get(&1, URL.Repo.DETS))
|> Text.encode_response()
|> Http.send(:found, conn)
end
match _ do match _ do
{:error, {:not_found, "route not found"}, conn} {:error, {:not_found, "route not found"}, conn}
|> Json.encode_response() |> Text.encode_response()
|> Http.send(:ignore, conn) |> Http.send(:ignore, conn)
end end
def handle_errors(conn, %{kind: _kind, reason: reason, stack: stack}) do def handle_errors(conn, %{kind: _kind, reason: reason, stack: stack}) do
Logger.error(inspect(reason), stack: stack) Logger.error(inspect(reason), stack: inspect(stack))
{:error, {:internal_server_error, "internal server error"}, conn} {:error, {:internal_server_error, "internal server error"}, conn}
|> Json.encode_response() |> Text.encode_response()
|> Http.send(:ignore, conn) |> Http.send(:ignore, conn)
end end
end end

View File

@ -14,6 +14,8 @@ defmodule Shortnr do
{Plug.Cowboy, scheme: :http, plug: Shortnr.Router, options: [port: port]} {Plug.Cowboy, scheme: :http, plug: Shortnr.Router, options: [port: port]}
] ]
:dets.open_file(:urls, type: :set)
Logger.info("server starting", port: port) Logger.info("server starting", port: port)
Supervisor.start_link(children, strategy: :one_for_one) Supervisor.start_link(children, strategy: :one_for_one)
end end

View File

@ -4,16 +4,23 @@ defmodule Shortnr.Transport.Http do
@type ok_error :: {:ok, term(), Plug.Conn.t()} | error() @type ok_error :: {:ok, term(), Plug.Conn.t()} | error()
@type error :: {:error, {atom(), String.t()}, Plug.Conn.t()} @type error :: {:error, {atom(), String.t()}, Plug.Conn.t()}
@spec handle(ok_error(), (term() -> ok_error())) :: ok_error() @spec handle(ok_error(), (... -> ok_error())) :: ok_error()
def handle(error = {:error, _sub_error, _conn}, _func), do: error def handle(error = {:error, _sub_error, _conn}, _func), do: error
def handle({:ok, request, conn}, func) do def handle({:ok, request, conn}, func) when is_function(func, 1) do
case func.(request) do case func.(request) do
{:ok, response} -> {:ok, response, conn} {:ok, response} -> {:ok, response, conn}
{:error, error_value} -> convert_error(error_value, conn) {:error, error_value} -> convert_error(error_value, conn)
end end
end end
def handle({:ok, _request, conn}, func) when is_function(func, 0) do
case func.() do
{:ok, response} -> {:ok, response, conn}
{:error, error_value} -> convert_error(error_value, conn)
end
end
defp convert_error({:invalid_argument, message}, conn), defp convert_error({:invalid_argument, message}, conn),
do: {:error, {:bad_request, message}, conn} do: {:error, {:bad_request, message}, conn}
@ -24,12 +31,18 @@ defmodule Shortnr.Transport.Http do
send_resp(original_conn, status_to_code(status), body) send_resp(original_conn, status_to_code(status), body)
end end
def send({:ok, body, conn}, :found, _original_conn) do
conn = put_resp_header(conn, "Location", URI.to_string(body))
send_resp(conn, status_to_code(:found), URI.to_string(body))
end
def send({:ok, body, conn}, success_status, _original_conn) do def send({:ok, body, conn}, success_status, _original_conn) do
send_resp(conn, status_to_code(success_status), body) send_resp(conn, status_to_code(success_status), body)
end end
defp status_to_code(:ok), do: 200 defp status_to_code(:ok), do: 200
defp status_to_code(:created), do: 201 defp status_to_code(:created), do: 201
defp status_to_code(:found), do: 302
defp status_to_code(:bad_request), do: 400 defp status_to_code(:bad_request), do: 400
defp status_to_code(:not_found), do: 404 defp status_to_code(:not_found), do: 404
defp status_to_code(:internal_server_error), do: 500 defp status_to_code(:internal_server_error), do: 500

30
lib/transport/text.ex Normal file
View File

@ -0,0 +1,30 @@
defmodule Shortnr.Transport.Text do
import Plug.Conn
alias Shortnr.Transport.Http
alias Shortnr.URL
@spec decode_request(Plug.Conn.t()) :: Http.ok_error()
def decode_request(conn) do
{:ok, body, conn} = read_body(conn)
{:ok, body, conn}
end
@spec encode_response(Http.ok_error()) :: Http.ok_error()
def encode_response(ok_error = {:error, _, _}), do: ok_error
def encode_response({:ok, [], conn}) do
{:ok, "", conn}
end
def encode_response({:ok, body, conn}) when is_list(body) do
{:ok, for(item <- body, into: "", do: "#{item}\n"), conn}
end
def encode_response({:ok, %URL{url: url}, conn}) do
{:ok, url, conn}
end
def encode_response({:ok, body, conn}) do
{:ok, "#{body}", conn}
end
end

20
lib/url/repo/dets.ex Normal file
View File

@ -0,0 +1,20 @@
defmodule Shortnr.URL.Repo.DETS do
@behaviour Shortnr.URL.Repo
@impl true
def get(key) do
{:ok, :dets.lookup(:urls, key) |> List.first() |> elem(1)}
end
@impl true
def put(url) do
:ok = :dets.insert(:urls, {url.id, url})
:ok
end
@impl true
def list() do
resp = :dets.select(:urls, [{:"$1", [], [:"$1"]}])
{:ok, resp |> Enum.map(&elem(&1, 1))}
end
end

8
lib/url/repo/repo.ex Normal file
View File

@ -0,0 +1,8 @@
defmodule Shortnr.URL.Repo do
alias Shortnr.URL
alias Shortnr.Transport
@callback put(URL.t()) :: :ok | Transport.error()
@callback get(String.t()) :: {:ok, URL.t()} | Transport.error()
@callback list() :: {:ok, list(URL.t())} | Transport.error()
end

46
lib/url/url.ex Normal file
View File

@ -0,0 +1,46 @@
defmodule Shortnr.URL do
alias Shortnr.Transport
alias Shortnr.URL
defstruct id:
for(
_ <- 0..7,
into: "",
do:
Enum.random(
String.codepoints(
"abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXWYZ0123456789"
)
)
),
created_at: DateTime.utc_now(),
updated_at: DateTime.utc_now(),
url: %URI{}
@type t :: %__MODULE__{id: String.t(), url: URI.t()}
@spec create(String.t(), module()) :: {:ok, String.t()} | Transport.error()
def create(url, repo) do
url_struct = %URL{url: URI.parse(url)}
:ok = repo.put(url_struct)
{:ok, url_struct.id}
end
@spec get(String.t(), module()) :: {:ok, URL.t()} | Transport.error()
def get(key, repo) do
{:ok, _} = repo.get(key)
end
@spec list(module()) :: {:ok, list(URL.t())} | Transport.error()
def list(repo) do
{:ok, _} = repo.list
end
defimpl String.Chars do
alias Shortnr.URL
@spec to_string(URL.t()) :: String.t()
def to_string(url) do
"id=#{url.id} created=#{url.created_at} updated=#{url.updated_at} url=#{url.url}"
end
end
end