|
@@ -57,6 +57,7 @@
|
|
-type opts() :: #{
|
|
-type opts() :: #{
|
|
compress => boolean(),
|
|
compress => boolean(),
|
|
idle_timeout => timeout(),
|
|
idle_timeout => timeout(),
|
|
|
|
+ max_frame_size => non_neg_integer() | infinity,
|
|
req_filter => fun((cowboy_req:req()) -> map())
|
|
req_filter => fun((cowboy_req:req()) -> map())
|
|
}.
|
|
}.
|
|
-export_type([opts/0]).
|
|
-export_type([opts/0]).
|
|
@@ -71,6 +72,7 @@
|
|
timeout = infinity :: timeout(),
|
|
timeout = infinity :: timeout(),
|
|
timeout_ref = undefined :: undefined | reference(),
|
|
timeout_ref = undefined :: undefined | reference(),
|
|
compress = false :: boolean(),
|
|
compress = false :: boolean(),
|
|
|
|
+ max_frame_size :: non_neg_integer() | infinity,
|
|
messages = undefined :: undefined | {atom(), atom(), atom()},
|
|
messages = undefined :: undefined | {atom(), atom(), atom()},
|
|
hibernate = false :: boolean(),
|
|
hibernate = false :: boolean(),
|
|
frag_state = undefined :: cow_ws:frag_state(),
|
|
frag_state = undefined :: cow_ws:frag_state(),
|
|
@@ -95,12 +97,14 @@ upgrade(Req, Env, Handler, HandlerState) ->
|
|
%% @todo Error out if HTTP/2.
|
|
%% @todo Error out if HTTP/2.
|
|
upgrade(Req0, Env, Handler, HandlerState, Opts) ->
|
|
upgrade(Req0, Env, Handler, HandlerState, Opts) ->
|
|
Timeout = maps:get(idle_timeout, Opts, 60000),
|
|
Timeout = maps:get(idle_timeout, Opts, 60000),
|
|
|
|
+ MaxFrameSize = maps:get(max_frame_size, Opts, infinity),
|
|
Compress = maps:get(compress, Opts, false),
|
|
Compress = maps:get(compress, Opts, false),
|
|
FilteredReq = case maps:get(req_filter, Opts, undefined) of
|
|
FilteredReq = case maps:get(req_filter, Opts, undefined) of
|
|
undefined -> maps:with([method, version, scheme, host, port, path, qs, peer], Req0);
|
|
undefined -> maps:with([method, version, scheme, host, port, path, qs, peer], Req0);
|
|
FilterFun -> FilterFun(Req0)
|
|
FilterFun -> FilterFun(Req0)
|
|
end,
|
|
end,
|
|
- State0 = #state{handler=Handler, timeout=Timeout, compress=Compress, req=FilteredReq},
|
|
|
|
|
|
+ State0 = #state{handler=Handler, timeout=Timeout, compress=Compress,
|
|
|
|
+ max_frame_size=MaxFrameSize, req=FilteredReq},
|
|
try websocket_upgrade(State0, Req0) of
|
|
try websocket_upgrade(State0, Req0) of
|
|
{ok, State, Req} ->
|
|
{ok, State, Req} ->
|
|
websocket_handshake(State, Req, HandlerState, Env);
|
|
websocket_handshake(State, Req, HandlerState, Env);
|
|
@@ -291,12 +295,15 @@ parse(State, HandlerState, PS=#ps_payload{buffer=Buffer}, Data) ->
|
|
parse_payload(State, HandlerState, PS#ps_payload{buffer= <<>>},
|
|
parse_payload(State, HandlerState, PS#ps_payload{buffer= <<>>},
|
|
<<Buffer/binary, Data/binary>>).
|
|
<<Buffer/binary, Data/binary>>).
|
|
|
|
|
|
-parse_header(State=#state{frag_state=FragState, extensions=Extensions}, HandlerState,
|
|
|
|
- ParseState=#ps_header{buffer=Data}) ->
|
|
|
|
|
|
+parse_header(State=#state{max_frame_size=MaxFrameSize,
|
|
|
|
+ frag_state=FragState, extensions=Extensions},
|
|
|
|
+ HandlerState, ParseState=#ps_header{buffer=Data}) ->
|
|
case cow_ws:parse_header(Data, Extensions, FragState) of
|
|
case cow_ws:parse_header(Data, Extensions, FragState) of
|
|
%% All frames sent from the client to the server are masked.
|
|
%% All frames sent from the client to the server are masked.
|
|
{_, _, _, _, undefined, _} ->
|
|
{_, _, _, _, undefined, _} ->
|
|
websocket_close(State, HandlerState, {error, badframe});
|
|
websocket_close(State, HandlerState, {error, badframe});
|
|
|
|
+ {_, _, _, Len, _, _} when Len > MaxFrameSize ->
|
|
|
|
+ websocket_close(State, HandlerState, {error, badsize});
|
|
{Type, FragState2, Rsv, Len, MaskKey, Rest} ->
|
|
{Type, FragState2, Rsv, Len, MaskKey, Rest} ->
|
|
parse_payload(State#state{frag_state=FragState2}, HandlerState,
|
|
parse_payload(State#state{frag_state=FragState2}, HandlerState,
|
|
#ps_payload{type=Type, len=Len, mask_key=MaskKey, rsv=Rsv}, Rest);
|
|
#ps_payload{type=Type, len=Len, mask_key=MaskKey, rsv=Rsv}, Rest);
|
|
@@ -335,11 +342,14 @@ parse_payload(State=#state{frag_state=FragState, utf8_state=Incomplete, extensio
|
|
end.
|
|
end.
|
|
|
|
|
|
dispatch_frame(State=#state{socket=Socket, transport=Transport,
|
|
dispatch_frame(State=#state{socket=Socket, transport=Transport,
|
|
- frag_state=FragState, frag_buffer=SoFar, extensions=Extensions},
|
|
|
|
- HandlerState, #ps_payload{type=Type0, unmasked=Payload0, close_code=CloseCode0},
|
|
|
|
|
|
+ max_frame_size=MaxFrameSize, frag_state=FragState,
|
|
|
|
+ frag_buffer=SoFar, extensions=Extensions}, HandlerState,
|
|
|
|
+ #ps_payload{type=Type0, unmasked=Payload0, close_code=CloseCode0},
|
|
RemainingData) ->
|
|
RemainingData) ->
|
|
case cow_ws:make_frame(Type0, Payload0, CloseCode0, FragState) of
|
|
case cow_ws:make_frame(Type0, Payload0, CloseCode0, FragState) of
|
|
%% @todo Allow receiving fragments.
|
|
%% @todo Allow receiving fragments.
|
|
|
|
+ {fragment, _, _, Payload} when byte_size(Payload) + byte_size(SoFar) > MaxFrameSize ->
|
|
|
|
+ websocket_close(State, HandlerState, {error, badsize});
|
|
{fragment, nofin, _, Payload} ->
|
|
{fragment, nofin, _, Payload} ->
|
|
parse_header(State#state{frag_buffer= << SoFar/binary, Payload/binary >>},
|
|
parse_header(State#state{frag_buffer= << SoFar/binary, Payload/binary >>},
|
|
HandlerState, #ps_header{buffer=RemainingData});
|
|
HandlerState, #ps_header{buffer=RemainingData});
|
|
@@ -447,6 +457,8 @@ websocket_send_close(#state{socket=Socket, transport=Transport,
|
|
Transport:send(Socket, cow_ws:frame({close, 1002, <<>>}, Extensions));
|
|
Transport:send(Socket, cow_ws:frame({close, 1002, <<>>}, Extensions));
|
|
{error, badencoding} ->
|
|
{error, badencoding} ->
|
|
Transport:send(Socket, cow_ws:frame({close, 1007, <<>>}, Extensions));
|
|
Transport:send(Socket, cow_ws:frame({close, 1007, <<>>}, Extensions));
|
|
|
|
+ {error, badsize} ->
|
|
|
|
+ Transport:send(Socket, cow_ws:frame({close, 1009, <<>>}, Extensions));
|
|
{crash, _, _} ->
|
|
{crash, _, _} ->
|
|
Transport:send(Socket, cow_ws:frame({close, 1011, <<>>}, Extensions));
|
|
Transport:send(Socket, cow_ws:frame({close, 1011, <<>>}, Extensions));
|
|
remote ->
|
|
remote ->
|