diff --git a/lib/httparrot.ex b/lib/httparrot.ex index 19a89c6..b825cf8 100644 --- a/lib/httparrot.ex +++ b/lib/httparrot.ex @@ -19,6 +19,7 @@ defmodule HTTParrot do {'/basic-auth/:user/:passwd', HTTParrot.BasicAuthHandler, []}, {'/hidden-basic-auth/:user/:passwd', HTTParrot.HiddenBasicAuthHandler, []}, {'/stream/:n', HTTParrot.StreamHandler, []}, + {'/delay/:n', HTTParrot.DelayedHandler, []}, {'/html', :cowboy_static, {:priv_file, :httparrot, "html.html"}} ] } ]) {:ok, port} = :application.get_env(:httparrot, :port) diff --git a/lib/httparrot/delayed_handler.ex b/lib/httparrot/delayed_handler.ex new file mode 100644 index 0000000..3a987bb --- /dev/null +++ b/lib/httparrot/delayed_handler.ex @@ -0,0 +1,37 @@ +defmodule HTTParrot.DelayedHandler do + alias HTTParrot.GeneralRequestInfo + + def init(_transport, _req, _opts) do + {:upgrade, :protocol, :cowboy_rest} + end + + def allowed_methods(req, state) do + {["GET"], req, state} + end + + def malformed_request(req, state) do + {n, req} = :cowboy_req.binding(:n, req) + try do + n = n |> binary_to_integer |> min(10) |> max(0) + {false, req, n} + rescue + ArgumentError -> {true, req, state} + end + end + + def content_types_provided(req, state) do + {[{{"application", "json", []}, :get_json}], req, state} + end + + def get_json(req, n) do + {info, req} = GeneralRequestInfo.retrieve(req) + :timer.sleep(n*1000) + {response(info), req, n} + end + + defp response(info) do + info |> JSEX.encode! + end + + def terminate(_, _, _), do: :ok +end diff --git a/test/delayed_handler_test.exs b/test/delayed_handler_test.exs new file mode 100644 index 0000000..a879907 --- /dev/null +++ b/test/delayed_handler_test.exs @@ -0,0 +1,57 @@ +defmodule HTTParrot.DelayedHandlerTest do + use ExUnit.Case + import :meck + import HTTParrot.DelayedHandler + + setup do + new HTTParrot.GeneralRequestInfo + new JSEX + end + + teardown do + unload HTTParrot.GeneralRequestInfo + unload JSEX + end + + test "malformed_request returns false if it's not an integer" do + expect(:cowboy_req, :binding, [{[:n, :req1], {"a2B=", :req2}}]) + + assert malformed_request(:req1, :state) == {true, :req2, :state} + + assert validate :cowboy_req + end + + test "malformed_request returns false if it's an integer" do + expect(:cowboy_req, :binding, [{[:n, :req1], {"2", :req2}}]) + + assert malformed_request(:req1, :state) == {false, :req2, 2} + + assert validate :cowboy_req + end + + test "malformed_request returns 0 if 'n' is less than 0" do + expect(:cowboy_req, :binding, [{[:n, :req1], {"-1", :req2}}]) + + assert malformed_request(:req1, :state) == {false, :req2, 0} + + assert validate :cowboy_req + end + + test "malformed_request returns 10 if 'n' is greater than 10" do + expect(:cowboy_req, :binding, [{[:n, :req1], {"20", :req2}}]) + + assert malformed_request(:req1, :state) == {false, :req2, 10} + + assert validate :cowboy_req + end + + test "returns json with query values, headers, url and origin" do + expect(HTTParrot.GeneralRequestInfo, :retrieve, 1, {:info, :req2}) + expect(JSEX, :encode!, [{[:info], :json}]) + + assert get_json(:req1, 0) == {:json, :req2, 0} + + assert validate HTTParrot.GeneralRequestInfo + assert validate JSEX + end +end