mirror of
https://github.com/ninenines/cowboy.git
synced 2025-07-14 12:20:24 +00:00
Add tests for the streams shutdown mechanism
This commit is contained in:
parent
58b70a594b
commit
a2facaf2da
2 changed files with 145 additions and 9 deletions
|
@ -9,21 +9,56 @@
|
||||||
-export([terminate/3]).
|
-export([terminate/3]).
|
||||||
-export([early_error/5]).
|
-export([early_error/5]).
|
||||||
|
|
||||||
init(StreamID, Req, Opts) ->
|
-record(state, {
|
||||||
%% @todo Vary behavior depending on x-test-case.
|
pid,
|
||||||
Pid = list_to_pid(binary_to_list(cowboy_req:header(<<"x-test-pid">>, Req))),
|
test
|
||||||
Pid ! {Pid, self(), init, StreamID, Req, Opts},
|
}).
|
||||||
{[{headers, 200, #{}}], Pid}.
|
|
||||||
|
|
||||||
data(StreamID, IsFin, Data, State=Pid) ->
|
init(StreamID, Req, Opts) ->
|
||||||
|
Pid = list_to_pid(binary_to_list(cowboy_req:header(<<"x-test-pid">>, Req))),
|
||||||
|
Test = binary_to_atom(cowboy_req:header(<<"x-test-case">>, Req), latin1),
|
||||||
|
State = #state{pid=Pid, test=Test},
|
||||||
|
Pid ! {Pid, self(), init, StreamID, Req, Opts},
|
||||||
|
{init_commands(StreamID, Req, State), State}.
|
||||||
|
|
||||||
|
init_commands(_, _, State=#state{test=shutdown_on_stream_stop}) ->
|
||||||
|
Spawn = init_process(false, State),
|
||||||
|
[{headers, 200, #{}}, {spawn, Spawn, 5000}, stop];
|
||||||
|
init_commands(_, _, State=#state{test=shutdown_on_socket_close}) ->
|
||||||
|
Spawn = init_process(false, State),
|
||||||
|
[{headers, 200, #{}}, {spawn, Spawn, 5000}];
|
||||||
|
init_commands(_, _, State=#state{test=shutdown_timeout_on_stream_stop}) ->
|
||||||
|
Spawn = init_process(true, State),
|
||||||
|
[{headers, 200, #{}}, {spawn, Spawn, 2000}, stop];
|
||||||
|
init_commands(_, _, State=#state{test=shutdown_timeout_on_socket_close}) ->
|
||||||
|
Spawn = init_process(true, State),
|
||||||
|
[{headers, 200, #{}}, {spawn, Spawn, 2000}];
|
||||||
|
init_commands(_, _, _) ->
|
||||||
|
[{headers, 200, #{}}].
|
||||||
|
|
||||||
|
init_process(TrapExit, #state{pid=Pid}) ->
|
||||||
|
Self = self(),
|
||||||
|
Spawn = spawn_link(fun() ->
|
||||||
|
process_flag(trap_exit, TrapExit),
|
||||||
|
Pid ! {Pid, Self, spawned, self()},
|
||||||
|
receive {Pid, ready} -> ok after 1000 -> error(timeout) end,
|
||||||
|
Self ! {self(), ready},
|
||||||
|
receive after 5000 ->
|
||||||
|
Pid ! {Pid, Self, still_alive, self()}
|
||||||
|
end
|
||||||
|
end),
|
||||||
|
receive {Spawn, ready} -> ok after 1000 -> error(timeout) end,
|
||||||
|
Spawn.
|
||||||
|
|
||||||
|
data(StreamID, IsFin, Data, State=#state{pid=Pid}) ->
|
||||||
Pid ! {Pid, self(), data, StreamID, IsFin, Data, State},
|
Pid ! {Pid, self(), data, StreamID, IsFin, Data, State},
|
||||||
{[], State}.
|
{[], State}.
|
||||||
|
|
||||||
info(StreamID, Info, State=Pid) ->
|
info(StreamID, Info, State=#state{pid=Pid}) ->
|
||||||
Pid ! {Pid, self(), info, StreamID, Info, State},
|
Pid ! {Pid, self(), info, StreamID, Info, State},
|
||||||
{[], State}.
|
{[], State}.
|
||||||
|
|
||||||
terminate(StreamID, Reason, State=Pid) ->
|
terminate(StreamID, Reason, State=#state{pid=Pid}) ->
|
||||||
Pid ! {Pid, self(), terminate, StreamID, Reason, State},
|
Pid ! {Pid, self(), terminate, StreamID, Reason, State},
|
||||||
ok.
|
ok.
|
||||||
|
|
||||||
|
|
|
@ -59,13 +59,114 @@ end_per_group(Name, _) ->
|
||||||
|
|
||||||
%% Tests.
|
%% Tests.
|
||||||
|
|
||||||
|
shutdown_on_stream_stop(Config) ->
|
||||||
|
doc("Confirm supervised processes are shutdown when stopping the stream."),
|
||||||
|
Self = self(),
|
||||||
|
ConnPid = gun_open(Config),
|
||||||
|
Ref = gun:get(ConnPid, "/long_polling", [
|
||||||
|
{<<"accept-encoding">>, <<"gzip">>},
|
||||||
|
{<<"x-test-case">>, <<"shutdown_on_stream_stop">>},
|
||||||
|
{<<"x-test-pid">>, pid_to_list(Self)}
|
||||||
|
]),
|
||||||
|
%% Confirm init/3 is called.
|
||||||
|
Pid = receive {Self, P, init, _, _, _} -> P after 1000 -> error(timeout) end,
|
||||||
|
%% Receive the pid of the newly started process and monitor it.
|
||||||
|
Spawn = receive {Self, Pid, spawned, S} -> S after 1000 -> error(timeout) end,
|
||||||
|
MRef = monitor(process, Spawn),
|
||||||
|
Spawn ! {Self, ready},
|
||||||
|
%% Confirm terminate/3 is called, indicating the stream ended.
|
||||||
|
receive {Self, Pid, terminate, _, _, _} -> ok after 1000 -> error(timeout) end,
|
||||||
|
%% We should receive a DOWN message soon after (or before) because the stream
|
||||||
|
%% handler is stopping the stream immediately after the process started.
|
||||||
|
receive {'DOWN', MRef, process, Spawn, shutdown} -> ok after 1000 -> error(timeout) end,
|
||||||
|
%% The response is still sent.
|
||||||
|
{response, nofin, 200, _} = gun:await(ConnPid, Ref),
|
||||||
|
{ok, <<>>} = gun:await_body(ConnPid, Ref),
|
||||||
|
ok.
|
||||||
|
|
||||||
|
shutdown_on_socket_close(Config) ->
|
||||||
|
doc("Confirm supervised processes are shutdown when the socket closes."),
|
||||||
|
Self = self(),
|
||||||
|
ConnPid = gun_open(Config),
|
||||||
|
Ref = gun:get(ConnPid, "/long_polling", [
|
||||||
|
{<<"accept-encoding">>, <<"gzip">>},
|
||||||
|
{<<"x-test-case">>, <<"shutdown_on_socket_close">>},
|
||||||
|
{<<"x-test-pid">>, pid_to_list(Self)}
|
||||||
|
]),
|
||||||
|
%% Confirm init/3 is called.
|
||||||
|
Pid = receive {Self, P, init, _, _, _} -> P after 1000 -> error(timeout) end,
|
||||||
|
%% Receive the pid of the newly started process and monitor it.
|
||||||
|
Spawn = receive {Self, Pid, spawned, S} -> S after 1000 -> error(timeout) end,
|
||||||
|
MRef = monitor(process, Spawn),
|
||||||
|
Spawn ! {Self, ready},
|
||||||
|
%% Close the socket.
|
||||||
|
ok = gun:close(ConnPid),
|
||||||
|
%% Confirm terminate/3 is called, indicating the stream ended.
|
||||||
|
receive {Self, Pid, terminate, _, _, _} -> ok after 1000 -> error(timeout) end,
|
||||||
|
%% Confirm we receive a DOWN message for the child process.
|
||||||
|
receive {'DOWN', MRef, process, Spawn, shutdown} -> ok after 1000 -> error(timeout) end,
|
||||||
|
ok.
|
||||||
|
|
||||||
|
shutdown_timeout_on_stream_stop(Config) ->
|
||||||
|
doc("Confirm supervised processes are killed "
|
||||||
|
"when the shutdown timeout triggers after stopping the stream."),
|
||||||
|
Self = self(),
|
||||||
|
ConnPid = gun_open(Config),
|
||||||
|
Ref = gun:get(ConnPid, "/long_polling", [
|
||||||
|
{<<"accept-encoding">>, <<"gzip">>},
|
||||||
|
{<<"x-test-case">>, <<"shutdown_timeout_on_stream_stop">>},
|
||||||
|
{<<"x-test-pid">>, pid_to_list(Self)}
|
||||||
|
]),
|
||||||
|
%% Confirm init/3 is called.
|
||||||
|
Pid = receive {Self, P, init, _, _, _} -> P after 1000 -> error(timeout) end,
|
||||||
|
%% Receive the pid of the newly started process and monitor it.
|
||||||
|
Spawn = receive {Self, Pid, spawned, S} -> S after 1000 -> error(timeout) end,
|
||||||
|
MRef = monitor(process, Spawn),
|
||||||
|
Spawn ! {Self, ready},
|
||||||
|
%% Confirm terminate/3 is called, indicating the stream ended.
|
||||||
|
receive {Self, Pid, terminate, _, _, _} -> ok after 1000 -> error(timeout) end,
|
||||||
|
%% We should NOT receive a DOWN message immediately.
|
||||||
|
receive {'DOWN', MRef, process, Spawn, killed} -> error(killed) after 1500 -> ok end,
|
||||||
|
%% We should received it now.
|
||||||
|
receive {'DOWN', MRef, process, Spawn, killed} -> ok after 1000 -> error(timeout) end,
|
||||||
|
%% The response is still sent.
|
||||||
|
{response, nofin, 200, _} = gun:await(ConnPid, Ref),
|
||||||
|
{ok, <<>>} = gun:await_body(ConnPid, Ref),
|
||||||
|
ok.
|
||||||
|
|
||||||
|
shutdown_timeout_on_socket_close(Config) ->
|
||||||
|
doc("Confirm supervised processes are killed "
|
||||||
|
"when the shutdown timeout triggers after the socket has closed."),
|
||||||
|
Self = self(),
|
||||||
|
ConnPid = gun_open(Config),
|
||||||
|
Ref = gun:get(ConnPid, "/long_polling", [
|
||||||
|
{<<"accept-encoding">>, <<"gzip">>},
|
||||||
|
{<<"x-test-case">>, <<"shutdown_timeout_on_socket_close">>},
|
||||||
|
{<<"x-test-pid">>, pid_to_list(Self)}
|
||||||
|
]),
|
||||||
|
%% Confirm init/3 is called.
|
||||||
|
Pid = receive {Self, P, init, _, _, _} -> P after 1000 -> error(timeout) end,
|
||||||
|
%% Receive the pid of the newly started process and monitor it.
|
||||||
|
Spawn = receive {Self, Pid, spawned, S} -> S after 1000 -> error(timeout) end,
|
||||||
|
MRef = monitor(process, Spawn),
|
||||||
|
Spawn ! {Self, ready},
|
||||||
|
%% Close the socket.
|
||||||
|
ok = gun:close(ConnPid),
|
||||||
|
%% Confirm terminate/3 is called, indicating the stream ended.
|
||||||
|
receive {Self, Pid, terminate, _, _, _} -> ok after 1000 -> error(timeout) end,
|
||||||
|
%% We should NOT receive a DOWN message immediately.
|
||||||
|
receive {'DOWN', MRef, process, Spawn, killed} -> error(killed) after 1500 -> ok end,
|
||||||
|
%% We should received it now.
|
||||||
|
receive {'DOWN', MRef, process, Spawn, killed} -> ok after 1000 -> error(timeout) end,
|
||||||
|
ok.
|
||||||
|
|
||||||
terminate_on_socket_close(Config) ->
|
terminate_on_socket_close(Config) ->
|
||||||
doc("Confirm terminate/3 is called when the socket gets closed brutally."),
|
doc("Confirm terminate/3 is called when the socket gets closed brutally."),
|
||||||
Self = self(),
|
Self = self(),
|
||||||
ConnPid = gun_open(Config),
|
ConnPid = gun_open(Config),
|
||||||
Ref = gun:get(ConnPid, "/long_polling", [
|
Ref = gun:get(ConnPid, "/long_polling", [
|
||||||
{<<"accept-encoding">>, <<"gzip">>},
|
{<<"accept-encoding">>, <<"gzip">>},
|
||||||
{<<"x-test-case">>, <<"stream">>},
|
{<<"x-test-case">>, <<"terminate_on_socket_close">>},
|
||||||
{<<"x-test-pid">>, pid_to_list(Self)}
|
{<<"x-test-pid">>, pid_to_list(Self)}
|
||||||
]),
|
]),
|
||||||
%% Confirm init/3 is called and receive the beginning of the response.
|
%% Confirm init/3 is called and receive the beginning of the response.
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue