dudeswave2/gun/test/ws_SUITE.erl

271 lines
9.6 KiB
Erlang

%% Copyright (c) 2018-2023, Loïc Hoguin <essen@ninenines.eu>
%%
%% Permission to use, copy, modify, and/or distribute this software for any
%% purpose with or without fee is hereby granted, provided that the above
%% copyright notice and this permission notice appear in all copies.
%%
%% THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
%% WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
%% MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
%% ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
%% WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
%% ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
%% OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
-module(ws_SUITE).
-compile(export_all).
-compile(nowarn_export_all).
-import(ct_helper, [config/2]).
-import(ct_helper, [doc/1]).
%% ct.
all() ->
[{group, http}, {group, http2}].
groups() ->
Tests = ct_helper:all(?MODULE),
HTTP1Tests = [
http10_upgrade_error,
http11_request_error,
http11_keepalive,
http11_keepalive_default_silence_pings
],
[
{http, [], Tests},
{http2, [], Tests -- HTTP1Tests}
].
init_per_suite(Config) ->
Routes = [
{"/", ws_echo_h, []},
{"/reject", ws_reject_h, []},
{"/subprotocol", ws_subprotocol_h, []}
],
{ok, _} = cowboy:start_clear(ws, [], #{
enable_connect_protocol => true,
env => #{dispatch => cowboy_router:compile([{'_', Routes}])}
}),
Port = ranch:get_port(ws),
[{port, Port}|Config].
end_per_suite(_) ->
cowboy:stop_listener(ws).
%% Tests.
await(Config) ->
doc("Ensure gun:await/2 can be used to receive Websocket frames."),
Protocol = config(name, config(tc_group_properties, Config)),
{ok, ConnPid} = gun:open("localhost", config(port, Config), #{
protocols => [Protocol],
http2_opts => #{notify_settings_changed => true}
}),
{ok, Protocol} = gun:await_up(ConnPid),
do_await_enable_connect_protocol(Protocol, ConnPid),
StreamRef = gun:ws_upgrade(ConnPid, "/", []),
{upgrade, [<<"websocket">>], _} = gun:await(ConnPid, StreamRef),
Frame = {text, <<"Hello!">>},
gun:ws_send(ConnPid, StreamRef, Frame),
{ws, Frame} = gun:await(ConnPid, StreamRef),
gun:close(ConnPid).
headers_normalized_upgrade(Config) ->
doc("Headers passed to ws_upgrade are normalized before being used."),
Protocol = config(name, config(tc_group_properties, Config)),
{ok, ConnPid} = gun:open("localhost", config(port, Config), #{
protocols => [Protocol],
http2_opts => #{notify_settings_changed => true}
}),
{ok, Protocol} = gun:await_up(ConnPid),
do_await_enable_connect_protocol(Protocol, ConnPid),
StreamRef = gun:ws_upgrade(ConnPid, "/", #{
atom_header_name => <<"value">>,
"string_header_name" => <<"value">>
}),
{upgrade, [<<"websocket">>], _} = gun:await(ConnPid, StreamRef),
gun:close(ConnPid).
http10_upgrade_error(Config) ->
doc("Attempting to upgrade HTTP/1.0 to Websocket produces an error."),
{ok, ConnPid} = gun:open("localhost", config(port, Config), #{
http_opts => #{version => 'HTTP/1.0'}
}),
{ok, _} = gun:await_up(ConnPid),
StreamRef = gun:ws_upgrade(ConnPid, "/", []),
receive
{gun_error, ConnPid, StreamRef, {badstate, _}} ->
gun:close(ConnPid);
Msg ->
error({fail, Msg})
after 1000 ->
error(timeout)
end.
http11_keepalive(Config) ->
doc("Ensure that Gun automatically sends ping frames."),
{ok, ConnPid} = gun:open("localhost", config(port, Config), #{
ws_opts => #{
keepalive => 100,
silence_pings => false
}
}),
{ok, _} = gun:await_up(ConnPid),
StreamRef = gun:ws_upgrade(ConnPid, "/", []),
{upgrade, [<<"websocket">>], _} = gun:await(ConnPid, StreamRef),
%% Gun sent a ping automatically, we therefore receive a pong.
{ws, pong} = gun:await(ConnPid, StreamRef),
gun:close(ConnPid).
http11_keepalive_default_silence_pings(Config) ->
doc("Ensure that Gun does not forward ping/pong by default."),
{ok, ConnPid} = gun:open("localhost", config(port, Config), #{
ws_opts => #{keepalive => 100}
}),
{ok, _} = gun:await_up(ConnPid),
StreamRef = gun:ws_upgrade(ConnPid, "/", []),
{upgrade, [<<"websocket">>], _} = gun:await(ConnPid, StreamRef),
%% Gun sent a ping automatically, but we silence ping/pong by default.
{error, timeout} = gun:await(ConnPid, StreamRef, 1000),
gun:close(ConnPid).
http11_request_error(Config) ->
doc("Ensure that HTTP/1.1 requests are rejected while using Websocket."),
{ok, ConnPid} = gun:open("localhost", config(port, Config)),
{ok, _} = gun:await_up(ConnPid),
StreamRef1 = gun:ws_upgrade(ConnPid, "/", []),
{upgrade, [<<"websocket">>], _} = gun:await(ConnPid, StreamRef1),
StreamRef2 = gun:get(ConnPid, "/"),
{error, {connection_error, {badstate, _}}} = gun:await(ConnPid, StreamRef2),
gun:close(ConnPid).
reject_upgrade(Config) ->
doc("Ensure Websocket connections can be rejected."),
Protocol = config(name, config(tc_group_properties, Config)),
{ok, ConnPid} = gun:open("localhost", config(port, Config), #{
protocols => [Protocol],
http2_opts => #{notify_settings_changed => true}
}),
{ok, Protocol} = gun:await_up(ConnPid),
do_await_enable_connect_protocol(Protocol, ConnPid),
StreamRef = gun:ws_upgrade(ConnPid, "/reject", []),
receive
{gun_response, ConnPid, StreamRef, nofin, 400, _} ->
{ok, <<"Upgrade rejected">>} = gun:await_body(ConnPid, StreamRef, 1000),
gun:close(ConnPid);
Msg ->
error({fail, Msg})
after 1000 ->
error(timeout)
end.
reply_to(Config) ->
doc("Ensure the reply_to request option is respected."),
Self = self(),
Frame = {text, <<"Hello!">>},
ReplyTo = spawn(fun() ->
{ConnPid, StreamRef} = receive
{C, S} when is_pid(C), is_reference(S) -> {C, S}
after 1000 ->
error(timeout)
end,
{upgrade, [<<"websocket">>], _} = gun:await(ConnPid, StreamRef),
Self ! {self(), ready},
{ws, Frame} = gun:await(ConnPid, StreamRef),
Self ! {self(), ok}
end),
Protocol = config(name, config(tc_group_properties, Config)),
{ok, ConnPid} = gun:open("localhost", config(port, Config), #{
protocols => [Protocol],
http2_opts => #{notify_settings_changed => true}
}),
{ok, Protocol} = gun:await_up(ConnPid),
do_await_enable_connect_protocol(Protocol, ConnPid),
StreamRef = gun:ws_upgrade(ConnPid, "/", [], #{reply_to => ReplyTo}),
ReplyTo ! {ConnPid, StreamRef},
receive {ReplyTo, ready} -> gun:ws_send(ConnPid, StreamRef, Frame) after 1000 -> error(timeout) end,
receive {ReplyTo, ok} -> gun:close(ConnPid) after 1000 -> error(timeout) end.
send_many(Config) ->
doc("Ensure we can send a list of frames in one gun:ws_send call."),
Protocol = config(name, config(tc_group_properties, Config)),
{ok, ConnPid} = gun:open("localhost", config(port, Config), #{
protocols => [Protocol],
http2_opts => #{notify_settings_changed => true}
}),
{ok, Protocol} = gun:await_up(ConnPid),
do_await_enable_connect_protocol(Protocol, ConnPid),
StreamRef = gun:ws_upgrade(ConnPid, "/", []),
{upgrade, [<<"websocket">>], _} = gun:await(ConnPid, StreamRef),
Frame1 = {text, <<"Hello!">>},
Frame2 = {binary, <<"World!">>},
gun:ws_send(ConnPid, StreamRef, [Frame1, Frame2]),
{ws, Frame1} = gun:await(ConnPid, StreamRef),
{ws, Frame2} = gun:await(ConnPid, StreamRef),
gun:close(ConnPid).
send_many_close(Config) ->
doc("Ensure we can send a list of frames in one gun:ws_send call, including a close frame."),
Protocol = config(name, config(tc_group_properties, Config)),
{ok, ConnPid} = gun:open("localhost", config(port, Config), #{
protocols => [Protocol],
http2_opts => #{notify_settings_changed => true}
}),
{ok, Protocol} = gun:await_up(ConnPid),
do_await_enable_connect_protocol(Protocol, ConnPid),
StreamRef = gun:ws_upgrade(ConnPid, "/", []),
{upgrade, [<<"websocket">>], _} = gun:await(ConnPid, StreamRef),
Frame1 = {text, <<"Hello!">>},
Frame2 = {binary, <<"World!">>},
gun:ws_send(ConnPid, StreamRef, [Frame1, Frame2, close]),
{ws, Frame1} = gun:await(ConnPid, StreamRef),
{ws, Frame2} = gun:await(ConnPid, StreamRef),
{ws, close} = gun:await(ConnPid, StreamRef),
gun:close(ConnPid).
subprotocol_match(Config) ->
doc("Websocket subprotocol successfully negotiated."),
Protocols = [{P, gun_ws_h} || P <- [<<"dummy">>, <<"echo">>, <<"junk">>]],
{ok, ConnPid} = gun:open("localhost", config(port, Config)),
{ok, _} = gun:await_up(ConnPid),
StreamRef = gun:ws_upgrade(ConnPid, "/subprotocol", [], #{
protocols => Protocols
}),
{upgrade, [<<"websocket">>], _} = gun:await(ConnPid, StreamRef),
Frame = {text, <<"Hello!">>},
gun:ws_send(ConnPid, StreamRef, Frame),
{ws, Frame} = gun:await(ConnPid, StreamRef),
gun:close(ConnPid).
subprotocol_nomatch(Config) ->
doc("Websocket subprotocol negotiation failure."),
Protocols = [{P, gun_ws_h} || P <- [<<"dummy">>, <<"junk">>]],
{ok, ConnPid} = gun:open("localhost", config(port, Config)),
{ok, _} = gun:await_up(ConnPid),
StreamRef = gun:ws_upgrade(ConnPid, "/subprotocol", [], #{
protocols => Protocols
}),
{response, nofin, 400, _} = gun:await(ConnPid, StreamRef),
{ok, <<"nomatch">>} = gun:await_body(ConnPid, StreamRef),
gun:close(ConnPid).
subprotocol_required_but_missing(Config) ->
doc("Websocket subprotocol not negotiated but required by the server."),
{ok, ConnPid} = gun:open("localhost", config(port, Config)),
{ok, _} = gun:await_up(ConnPid),
StreamRef = gun:ws_upgrade(ConnPid, "/subprotocol", []),
{response, nofin, 400, _} = gun:await(ConnPid, StreamRef),
{ok, <<"undefined">>} = gun:await_body(ConnPid, StreamRef),
gun:close(ConnPid).
%% Internal.
do_await_enable_connect_protocol(http, _) ->
ok;
%% We cannot do a CONNECT :protocol request until the server tells us we can.
do_await_enable_connect_protocol(http2, ConnPid) ->
{notify, settings_changed, #{enable_connect_protocol := true}}
= gun:await(ConnPid, undefined), %% @todo Maybe have a gun:await/1?
ok.