0
Fork 0
mirror of https://github.com/ninenines/cowboy.git synced 2025-07-14 12:20:24 +00:00

Websocket: Allow setting the max_frame_size option dynamically

This can be used to limit the maximum frame size before
some authentication or other validation is completed.
This commit is contained in:
Loïc Hoguin 2025-01-16 14:40:37 +01:00
parent 818b448ae9
commit fc9ba13938
No known key found for this signature in database
GPG key ID: 8A9DF795F6FED764
3 changed files with 59 additions and 11 deletions

View file

@ -615,14 +615,16 @@ commands([{active, Active}|Tail], State0=#state{active=Active0}, Data) when is_b
commands(Tail, State#state{active=Active}, Data); commands(Tail, State#state{active=Active}, Data);
commands([{deflate, Deflate}|Tail], State, Data) when is_boolean(Deflate) -> commands([{deflate, Deflate}|Tail], State, Data) when is_boolean(Deflate) ->
commands(Tail, State#state{deflate=Deflate}, Data); commands(Tail, State#state{deflate=Deflate}, Data);
commands([{set_options, SetOpts}|Tail], State0=#state{opts=Opts}, Data) -> commands([{set_options, SetOpts}|Tail], State0, Data) ->
State = case SetOpts of State = maps:fold(fun
#{idle_timeout := IdleTimeout} -> (idle_timeout, IdleTimeout, StateF=#state{opts=Opts}) ->
%% We reset the number of ticks when changing the idle_timeout option. %% We reset the number of ticks when changing the idle_timeout option.
set_idle_timeout(State0#state{opts=Opts#{idle_timeout => IdleTimeout}}, 0); set_idle_timeout(StateF#state{opts=Opts#{idle_timeout => IdleTimeout}}, 0);
_ -> (max_frame_size, MaxFrameSize, StateF=#state{opts=Opts}) ->
State0 StateF#state{opts=Opts#{max_frame_size => MaxFrameSize}};
end, (_, _, StateF) ->
StateF
end, State0, SetOpts),
commands(Tail, State, Data); commands(Tail, State, Data);
commands([{shutdown_reason, ShutdownReason}|Tail], State, Data) -> commands([{shutdown_reason, ShutdownReason}|Tail], State, Data) ->
commands(Tail, State#state{shutdown_reason=ShutdownReason}, Data); commands(Tail, State#state{shutdown_reason=ShutdownReason}, Data);

View file

@ -11,10 +11,21 @@ init(Req, RunOrHibernate) ->
{cowboy_websocket, Req, RunOrHibernate, {cowboy_websocket, Req, RunOrHibernate,
#{idle_timeout => infinity}}. #{idle_timeout => infinity}}.
websocket_handle(Frame={text, <<"idle_timeout_short">>}, State=run) -> %% Set the idle_timeout option dynamically.
{[{set_options, #{idle_timeout => 500}}, Frame], State}; websocket_handle({text, <<"idle_timeout_short">>}, State=run) ->
websocket_handle(Frame={text, <<"idle_timeout_short">>}, State=hibernate) -> {[{set_options, #{idle_timeout => 500}}], State};
{[{set_options, #{idle_timeout => 500}}, Frame], State, hibernate}. websocket_handle({text, <<"idle_timeout_short">>}, State=hibernate) ->
{[{set_options, #{idle_timeout => 500}}], State, hibernate};
%% Set the max_frame_size option dynamically.
websocket_handle({text, <<"max_frame_size_small">>}, State=run) ->
{[{set_options, #{max_frame_size => 1000}}], State};
websocket_handle({text, <<"max_frame_size_small">>}, State=hibernate) ->
{[{set_options, #{max_frame_size => 1000}}], State, hibernate};
%% We just echo binary frames.
websocket_handle(Frame={binary, _}, State=run) ->
{[Frame], State};
websocket_handle(Frame={binary, _}, State=hibernate) ->
{[Frame], State, hibernate}.
websocket_info(_Info, State) -> websocket_info(_Info, State) ->
{[], State}. {[], State}.

View file

@ -296,6 +296,41 @@ websocket_set_options_idle_timeout(Config) ->
error(timeout) error(timeout)
end. end.
websocket_set_options_max_frame_size(Config) ->
doc("The max_frame_size option can be modified using the "
"command {set_options, Opts} at runtime."),
ConnPid = gun_open(Config),
StreamRef = gun:ws_upgrade(ConnPid, "/set_options"),
receive
{gun_upgrade, ConnPid, StreamRef, [<<"websocket">>], _} ->
ok;
{gun_response, ConnPid, _, _, Status, Headers} ->
exit({ws_upgrade_failed, Status, Headers});
{gun_error, ConnPid, StreamRef, Reason} ->
exit({ws_upgrade_failed, Reason})
after 1000 ->
error(timeout)
end,
%% We first send a 1MB frame to confirm that yes, we can
%% send a frame that large. The default max_frame_size is infinity.
gun:ws_send(ConnPid, StreamRef, {binary, <<0:8000000>>}),
{ws, {binary, <<0:8000000>>}} = gun:await(ConnPid, StreamRef),
%% Trigger the change in max_frame_size. From now on we will
%% only allow frames of up to 1000 bytes.
gun:ws_send(ConnPid, StreamRef, {text, <<"max_frame_size_small">>}),
%% Confirm that we can send frames of up to 1000 bytes.
gun:ws_send(ConnPid, StreamRef, {binary, <<0:8000>>}),
{ws, {binary, <<0:8000>>}} = gun:await(ConnPid, StreamRef),
%% Confirm that sending frames larger than 1000 bytes
%% results in the closing of the connection.
gun:ws_send(ConnPid, StreamRef, {binary, <<0:8008>>}),
receive
{gun_down, ConnPid, _, _, _} ->
ok
after 2000 ->
error(timeout)
end.
websocket_shutdown_reason(Config) -> websocket_shutdown_reason(Config) ->
doc("The command {shutdown_reason, any()} can be used to " doc("The command {shutdown_reason, any()} can be used to "
"change the shutdown reason of a Websocket connection."), "change the shutdown reason of a Websocket connection."),