|
|
@@ -14,20 +14,22 @@
|
|
|
|
|
|
-module(emqx_connection).
|
|
|
|
|
|
--behaviour(gen_server).
|
|
|
+-behaviour(gen_statem).
|
|
|
|
|
|
-include("emqx.hrl").
|
|
|
-include("emqx_mqtt.hrl").
|
|
|
-include("logger.hrl").
|
|
|
|
|
|
-export([start_link/3]).
|
|
|
--export([info/1, attrs/1, stats/1]).
|
|
|
+-export([info/1]).
|
|
|
+-export([attrs/1]).
|
|
|
+-export([stats/1]).
|
|
|
-export([kick/1]).
|
|
|
-export([session/1]).
|
|
|
|
|
|
-%% gen_server callbacks
|
|
|
--export([init/1, handle_call/3, handle_cast/2, handle_info/2, terminate/2,
|
|
|
- code_change/3]).
|
|
|
+%% gen_statem callbacks
|
|
|
+-export([idle/3, connected/3]).
|
|
|
+-export([init/1, callback_mode/0, code_change/4, terminate/3]).
|
|
|
|
|
|
-record(state, {
|
|
|
transport,
|
|
|
@@ -37,7 +39,7 @@
|
|
|
conn_state,
|
|
|
active_n,
|
|
|
proto_state,
|
|
|
- parser_state,
|
|
|
+ parse_state,
|
|
|
gc_state,
|
|
|
keepalive,
|
|
|
enable_stats,
|
|
|
@@ -48,28 +50,29 @@
|
|
|
idle_timeout
|
|
|
}).
|
|
|
|
|
|
--define(DEFAULT_ACTIVE_N, 100).
|
|
|
+-define(ACTIVE_N, 100).
|
|
|
+-define(HANDLE(T, C, D), handle((T), (C), (D))).
|
|
|
-define(SOCK_STATS, [recv_oct, recv_cnt, send_oct, send_cnt, send_pend]).
|
|
|
|
|
|
start_link(Transport, Socket, Options) ->
|
|
|
- {ok, proc_lib:spawn_link(?MODULE, init, [[Transport, Socket, Options]])}.
|
|
|
+ {ok, proc_lib:spawn_link(?MODULE, init, [{Transport, Socket, Options}])}.
|
|
|
|
|
|
%%------------------------------------------------------------------------------
|
|
|
%% API
|
|
|
%%------------------------------------------------------------------------------
|
|
|
|
|
|
-%% for debug
|
|
|
+%% For debug
|
|
|
info(CPid) when is_pid(CPid) ->
|
|
|
call(CPid, info);
|
|
|
|
|
|
-info(#state{transport = Transport,
|
|
|
- socket = Socket,
|
|
|
- peername = Peername,
|
|
|
- sockname = Sockname,
|
|
|
- conn_state = ConnState,
|
|
|
- active_n = ActiveN,
|
|
|
- rate_limit = RateLimit,
|
|
|
- pub_limit = PubLimit,
|
|
|
+info(#state{transport = Transport,
|
|
|
+ socket = Socket,
|
|
|
+ peername = Peername,
|
|
|
+ sockname = Sockname,
|
|
|
+ conn_state = ConnState,
|
|
|
+ active_n = ActiveN,
|
|
|
+ rate_limit = RateLimit,
|
|
|
+ pub_limit = PubLimit,
|
|
|
proto_state = ProtoState}) ->
|
|
|
ConnInfo = [{socktype, Transport:type(Socket)},
|
|
|
{peername, Peername},
|
|
|
@@ -81,10 +84,12 @@ info(#state{transport = Transport,
|
|
|
ProtoInfo = emqx_protocol:info(ProtoState),
|
|
|
lists:usort(lists:append(ConnInfo, ProtoInfo)).
|
|
|
|
|
|
-rate_limit_info(undefined) -> #{};
|
|
|
-rate_limit_info(Limit) -> esockd_rate_limit:info(Limit).
|
|
|
+rate_limit_info(undefined) ->
|
|
|
+ #{};
|
|
|
+rate_limit_info(Limit) ->
|
|
|
+ esockd_rate_limit:info(Limit).
|
|
|
|
|
|
-%% for dashboard
|
|
|
+%% For dashboard
|
|
|
attrs(CPid) when is_pid(CPid) ->
|
|
|
call(CPid, attrs);
|
|
|
|
|
|
@@ -100,277 +105,305 @@ attrs(#state{peername = Peername,
|
|
|
stats(CPid) when is_pid(CPid) ->
|
|
|
call(CPid, stats);
|
|
|
|
|
|
-stats(#state{transport = Transport,
|
|
|
- socket = Socket,
|
|
|
+stats(#state{transport = Transport,
|
|
|
+ socket = Socket,
|
|
|
proto_state = ProtoState}) ->
|
|
|
- lists:append([emqx_misc:proc_stats(),
|
|
|
- emqx_protocol:stats(ProtoState),
|
|
|
- case Transport:getstat(Socket, ?SOCK_STATS) of
|
|
|
- {ok, Ss} -> Ss;
|
|
|
- {error, _} -> []
|
|
|
- end]).
|
|
|
+ SockStats = case Transport:getstat(Socket, ?SOCK_STATS) of
|
|
|
+ {ok, Ss} -> Ss;
|
|
|
+ {error, _} -> []
|
|
|
+ end,
|
|
|
+ lists:append([SockStats,
|
|
|
+ emqx_misc:proc_stats(),
|
|
|
+ emqx_protocol:stats(ProtoState)]).
|
|
|
|
|
|
-kick(CPid) -> call(CPid, kick).
|
|
|
+kick(CPid) ->
|
|
|
+ call(CPid, kick).
|
|
|
|
|
|
-session(CPid) -> call(CPid, session).
|
|
|
+session(CPid) ->
|
|
|
+ call(CPid, session).
|
|
|
|
|
|
call(CPid, Req) ->
|
|
|
- gen_server:call(CPid, Req, infinity).
|
|
|
+ gen_statem:call(CPid, Req, infinity).
|
|
|
|
|
|
%%------------------------------------------------------------------------------
|
|
|
-%% gen_server callbacks
|
|
|
+%% gen_statem callbacks
|
|
|
%%------------------------------------------------------------------------------
|
|
|
|
|
|
-init([Transport, RawSocket, Options]) ->
|
|
|
- case Transport:wait(RawSocket) of
|
|
|
- {ok, Socket} ->
|
|
|
- Zone = proplists:get_value(zone, Options),
|
|
|
- {ok, Peername} = Transport:ensure_ok_or_exit(peername, [Socket]),
|
|
|
- {ok, Sockname} = Transport:ensure_ok_or_exit(sockname, [Socket]),
|
|
|
- Peercert = Transport:ensure_ok_or_exit(peercert, [Socket]),
|
|
|
- RateLimit = init_limiter(proplists:get_value(rate_limit, Options)),
|
|
|
- PubLimit = init_limiter(emqx_zone:get_env(Zone, publish_limit)),
|
|
|
- ActiveN = proplists:get_value(active_n, Options, ?DEFAULT_ACTIVE_N),
|
|
|
- EnableStats = emqx_zone:get_env(Zone, enable_stats, true),
|
|
|
- IdleTimout = emqx_zone:get_env(Zone, idle_timeout, 30000),
|
|
|
- SendFun = send_fun(Transport, Socket),
|
|
|
- ProtoState = emqx_protocol:init(#{peername => Peername,
|
|
|
- sockname => Sockname,
|
|
|
- peercert => Peercert,
|
|
|
- sendfun => SendFun}, Options),
|
|
|
- ParserState = emqx_protocol:parser(ProtoState),
|
|
|
- GcPolicy = emqx_zone:get_env(Zone, force_gc_policy, false),
|
|
|
- GcState = emqx_gc:init(GcPolicy),
|
|
|
- State = run_socket(#state{transport = Transport,
|
|
|
- socket = Socket,
|
|
|
- peername = Peername,
|
|
|
- conn_state = running,
|
|
|
- active_n = ActiveN,
|
|
|
- rate_limit = RateLimit,
|
|
|
- pub_limit = PubLimit,
|
|
|
- proto_state = ProtoState,
|
|
|
- parser_state = ParserState,
|
|
|
- gc_state = GcState,
|
|
|
- enable_stats = EnableStats,
|
|
|
- idle_timeout = IdleTimout
|
|
|
- }),
|
|
|
- ok = emqx_misc:init_proc_mng_policy(Zone),
|
|
|
- emqx_logger:set_metadata_peername(esockd_net:format(Peername)),
|
|
|
- gen_server:enter_loop(?MODULE, [{hibernate_after, IdleTimout}],
|
|
|
- State, self(), IdleTimout);
|
|
|
- {error, Reason} ->
|
|
|
- {stop, Reason}
|
|
|
- end.
|
|
|
+init({Transport, RawSocket, Options}) ->
|
|
|
+ {ok, Socket} = Transport:wait(RawSocket),
|
|
|
+ {ok, Peername} = Transport:ensure_ok_or_exit(peername, [Socket]),
|
|
|
+ {ok, Sockname} = Transport:ensure_ok_or_exit(sockname, [Socket]),
|
|
|
+ Peercert = Transport:ensure_ok_or_exit(peercert, [Socket]),
|
|
|
+ emqx_logger:set_metadata_peername(esockd_net:format(Peername)),
|
|
|
+ Zone = proplists:get_value(zone, Options),
|
|
|
+ RateLimit = init_limiter(proplists:get_value(rate_limit, Options)),
|
|
|
+ PubLimit = init_limiter(emqx_zone:get_env(Zone, publish_limit)),
|
|
|
+ ActiveN = proplists:get_value(active_n, Options, ?ACTIVE_N),
|
|
|
+ EnableStats = emqx_zone:get_env(Zone, enable_stats, true),
|
|
|
+ IdleTimout = emqx_zone:get_env(Zone, idle_timeout, 30000),
|
|
|
+ SendFun = fun(Data) -> Transport:async_send(Socket, Data) end,
|
|
|
+ ProtoState = emqx_protocol:init(#{peername => Peername,
|
|
|
+ sockname => Sockname,
|
|
|
+ peercert => Peercert,
|
|
|
+ sendfun => SendFun}, Options),
|
|
|
+ ParseState = emqx_protocol:parser(ProtoState),
|
|
|
+ GcPolicy = emqx_zone:get_env(Zone, force_gc_policy, false),
|
|
|
+ GcState = emqx_gc:init(GcPolicy),
|
|
|
+ State = #state{transport = Transport,
|
|
|
+ socket = Socket,
|
|
|
+ peername = Peername,
|
|
|
+ conn_state = running,
|
|
|
+ active_n = ActiveN,
|
|
|
+ rate_limit = RateLimit,
|
|
|
+ pub_limit = PubLimit,
|
|
|
+ proto_state = ProtoState,
|
|
|
+ parse_state = ParseState,
|
|
|
+ gc_state = GcState,
|
|
|
+ enable_stats = EnableStats,
|
|
|
+ idle_timeout = IdleTimout},
|
|
|
+ ok = emqx_misc:init_proc_mng_policy(Zone),
|
|
|
+ gen_statem:enter_loop(?MODULE, [{hibernate_after, 2 * IdleTimout}],
|
|
|
+ idle, State, self(), [IdleTimout]).
|
|
|
|
|
|
init_limiter(undefined) ->
|
|
|
undefined;
|
|
|
init_limiter({Rate, Burst}) ->
|
|
|
esockd_rate_limit:new(Rate, Burst).
|
|
|
|
|
|
-send_fun(Transport, Socket) ->
|
|
|
- fun(Packet, Options) ->
|
|
|
- Data = emqx_frame:serialize(Packet, Options),
|
|
|
- try Transport:async_send(Socket, Data) of
|
|
|
- ok ->
|
|
|
- emqx_metrics:trans(inc, 'bytes/sent', iolist_size(Data)),
|
|
|
- ok;
|
|
|
- Error -> Error
|
|
|
- catch
|
|
|
- error:Error ->
|
|
|
- {error, Error}
|
|
|
- end
|
|
|
- end.
|
|
|
-
|
|
|
-handle_call(info, _From, State) ->
|
|
|
- {reply, info(State), State};
|
|
|
+callback_mode() ->
|
|
|
+ [state_functions, state_enter].
|
|
|
|
|
|
-handle_call(attrs, _From, State) ->
|
|
|
- {reply, attrs(State), State};
|
|
|
+%%------------------------------------------------------------------------------
|
|
|
+%% Idle state
|
|
|
|
|
|
-handle_call(stats, _From, State) ->
|
|
|
- {reply, stats(State), State};
|
|
|
+idle(enter, _, State) ->
|
|
|
+ ok = activate_socket(State),
|
|
|
+ keep_state_and_data;
|
|
|
|
|
|
-handle_call(kick, _From, State) ->
|
|
|
- {stop, {shutdown, kicked}, ok, State};
|
|
|
+idle(timeout, _Timeout, State) ->
|
|
|
+ {stop, idle_timeout, State};
|
|
|
|
|
|
-handle_call(session, _From, State = #state{proto_state = ProtoState}) ->
|
|
|
- {reply, emqx_protocol:session(ProtoState), State};
|
|
|
+idle(cast, {incoming, Packet}, State) ->
|
|
|
+ handle_packet(Packet, fun(NState) ->
|
|
|
+ {next_state, connected, NState}
|
|
|
+ end, State);
|
|
|
|
|
|
-handle_call(Req, _From, State) ->
|
|
|
- ?LOG(error, "unexpected call: ~p", [Req]),
|
|
|
- {reply, ignored, State}.
|
|
|
+idle(EventType, Content, State) ->
|
|
|
+ ?HANDLE(EventType, Content, State).
|
|
|
|
|
|
-handle_cast(Msg, State) ->
|
|
|
- ?LOG(error, "unexpected cast: ~p", [Msg]),
|
|
|
- {noreply, State}.
|
|
|
-
|
|
|
-handle_info({deliver, PubOrAck}, State = #state{proto_state = ProtoState}) ->
|
|
|
+%%------------------------------------------------------------------------------
|
|
|
+%% Connected state
|
|
|
+
|
|
|
+connected(enter, _, _State) ->
|
|
|
+ %% What to do?
|
|
|
+ keep_state_and_data;
|
|
|
+
|
|
|
+%% Handle Input
|
|
|
+connected(cast, {incoming, Packet = ?PACKET(Type)}, State) ->
|
|
|
+ _ = emqx_metrics:received(Packet),
|
|
|
+ (Type == ?PUBLISH) andalso emqx_pd:update_counter(incoming_pubs, 1),
|
|
|
+ handle_packet(Packet, fun(NState) ->
|
|
|
+ {keep_state, NState}
|
|
|
+ end, State);
|
|
|
+
|
|
|
+%% Handle Output
|
|
|
+connected(info, {deliver, PubOrAck}, State = #state{proto_state = ProtoState}) ->
|
|
|
case emqx_protocol:deliver(PubOrAck, ProtoState) of
|
|
|
- {ok, ProtoState1} ->
|
|
|
- State1 = State#state{proto_state = ProtoState1},
|
|
|
- {noreply, maybe_gc(PubOrAck, ensure_stats_timer(State1))};
|
|
|
+ {ok, NProtoState} ->
|
|
|
+ NState = State#state{proto_state = NProtoState},
|
|
|
+ {keep_state, maybe_gc(PubOrAck, NState)};
|
|
|
{error, Reason} ->
|
|
|
shutdown(Reason, State)
|
|
|
end;
|
|
|
|
|
|
-handle_info({timeout, Timer, emit_stats},
|
|
|
- State = #state{stats_timer = Timer,
|
|
|
- proto_state = ProtoState,
|
|
|
- gc_state = GcState}) ->
|
|
|
+%% Start Keepalive
|
|
|
+connected(info, {keepalive, start, Interval},
|
|
|
+ State = #state{transport = Transport, socket = Socket}) ->
|
|
|
+ StatFun = fun() ->
|
|
|
+ case Transport:getstat(Socket, [recv_oct]) of
|
|
|
+ {ok, [{recv_oct, RecvOct}]} -> {ok, RecvOct};
|
|
|
+ Error -> Error
|
|
|
+ end
|
|
|
+ end,
|
|
|
+ case emqx_keepalive:start(StatFun, Interval, {keepalive, check}) of
|
|
|
+ {ok, KeepAlive} ->
|
|
|
+ {keep_state, State#state{keepalive = KeepAlive}};
|
|
|
+ {error, Error} ->
|
|
|
+ shutdown(Error, State)
|
|
|
+ end;
|
|
|
+
|
|
|
+%% Keepalive timer
|
|
|
+connected(info, {keepalive, check}, State = #state{keepalive = KeepAlive}) ->
|
|
|
+ case emqx_keepalive:check(KeepAlive) of
|
|
|
+ {ok, KeepAlive1} ->
|
|
|
+ {keep_state, State#state{keepalive = KeepAlive1}};
|
|
|
+ {error, timeout} ->
|
|
|
+ shutdown(keepalive_timeout, State);
|
|
|
+ {error, Error} ->
|
|
|
+ shutdown(Error, State)
|
|
|
+ end;
|
|
|
+
|
|
|
+connected(EventType, Content, State) ->
|
|
|
+ ?HANDLE(EventType, Content, State).
|
|
|
+
|
|
|
+%% Handle call
|
|
|
+handle({call, From}, info, State) ->
|
|
|
+ reply(From, info(State), State);
|
|
|
+
|
|
|
+handle({call, From}, attrs, State) ->
|
|
|
+ reply(From, attrs(State), State);
|
|
|
+
|
|
|
+handle({call, From}, stats, State) ->
|
|
|
+ reply(From, stats(State), State);
|
|
|
+
|
|
|
+handle({call, From}, kick, State) ->
|
|
|
+ ok = gen_statem:reply(From, ok),
|
|
|
+ shutdown(kicked, State);
|
|
|
+
|
|
|
+handle({call, From}, session, State = #state{proto_state = ProtoState}) ->
|
|
|
+ reply(From, emqx_protocol:session(ProtoState), State);
|
|
|
+
|
|
|
+handle({call, From}, Req, State) ->
|
|
|
+ ?LOG(error, "unexpected call: ~p", [Req]),
|
|
|
+ reply(From, ignored, State);
|
|
|
+
|
|
|
+%% Handle cast
|
|
|
+handle(cast, Msg, State) ->
|
|
|
+ ?LOG(error, "unexpected cast: ~p", [Msg]),
|
|
|
+ {keep_state, State};
|
|
|
+
|
|
|
+%% Handle Incoming
|
|
|
+handle(info, {Inet, _Sock, Data}, State) when Inet == tcp; Inet == ssl ->
|
|
|
+ Oct = iolist_size(Data),
|
|
|
+ ?LOG(debug, "RECV ~p", [Data]),
|
|
|
+ emqx_pd:update_counter(incoming_bytes, Oct),
|
|
|
+ emqx_metrics:trans(inc, 'bytes/received', Oct),
|
|
|
+ NState = ensure_stats_timer(maybe_gc({1, Oct}, State)),
|
|
|
+ process_incoming(Data, [], NState);
|
|
|
+
|
|
|
+handle(info, {Error, _Sock, Reason}, State)
|
|
|
+ when Error == tcp_error; Error == ssl_error ->
|
|
|
+ shutdown(Reason, State);
|
|
|
+
|
|
|
+handle(info, {Closed, _Sock}, State)
|
|
|
+ when Closed == tcp_closed; Closed == ssl_closed ->
|
|
|
+ shutdown(closed, State);
|
|
|
+
|
|
|
+handle(info, {tcp_passive, _Sock}, State) ->
|
|
|
+ %% Rate limit here:)
|
|
|
+ NState = ensure_rate_limit(State),
|
|
|
+ ok = activate_socket(NState),
|
|
|
+ {keep_state, NState};
|
|
|
+
|
|
|
+handle(info, activate_socket, State) ->
|
|
|
+ %% Rate limit timer expired.
|
|
|
+ ok = activate_socket(State),
|
|
|
+ {keep_state, State#state{conn_state = running, limit_timer = undefined}};
|
|
|
+
|
|
|
+handle(info, {inet_reply, _Sock, ok}, State) ->
|
|
|
+ %% something sent
|
|
|
+ {keep_state, ensure_stats_timer(State)};
|
|
|
+
|
|
|
+handle(info, {inet_reply, _Sock, {error, Reason}}, State) ->
|
|
|
+ shutdown(Reason, State);
|
|
|
+
|
|
|
+handle(info, {timeout, Timer, emit_stats},
|
|
|
+ State = #state{stats_timer = Timer,
|
|
|
+ proto_state = ProtoState,
|
|
|
+ gc_state = GcState}) ->
|
|
|
emqx_metrics:commit(),
|
|
|
emqx_cm:set_conn_stats(emqx_protocol:client_id(ProtoState), stats(State)),
|
|
|
- NewState = State#state{stats_timer = undefined},
|
|
|
+ NState = State#state{stats_timer = undefined},
|
|
|
Limits = erlang:get(force_shutdown_policy),
|
|
|
case emqx_misc:conn_proc_mng_policy(Limits) of
|
|
|
continue ->
|
|
|
- {noreply, NewState};
|
|
|
+ {keep_state, NState};
|
|
|
hibernate ->
|
|
|
%% going to hibernate, reset gc stats
|
|
|
GcState1 = emqx_gc:reset(GcState),
|
|
|
- {noreply, NewState#state{gc_state = GcState1}, hibernate};
|
|
|
+ {keep_state, NState#state{gc_state = GcState1}, hibernate};
|
|
|
{shutdown, Reason} ->
|
|
|
?LOG(warning, "shutdown due to ~p", [Reason]),
|
|
|
- shutdown(Reason, NewState)
|
|
|
+ shutdown(Reason, NState)
|
|
|
end;
|
|
|
|
|
|
-handle_info(timeout, State) ->
|
|
|
- shutdown(idle_timeout, State);
|
|
|
-
|
|
|
-handle_info({shutdown, Reason}, State) ->
|
|
|
- shutdown(Reason, State);
|
|
|
-
|
|
|
-handle_info({shutdown, discard, {ClientId, ByPid}}, State) ->
|
|
|
+handle(info, {shutdown, discard, {ClientId, ByPid}}, State) ->
|
|
|
?LOG(warning, "discarded by ~s:~p", [ClientId, ByPid]),
|
|
|
shutdown(discard, State);
|
|
|
|
|
|
-handle_info({shutdown, conflict, {ClientId, NewPid}}, State) ->
|
|
|
+handle(info, {shutdown, conflict, {ClientId, NewPid}}, State) ->
|
|
|
?LOG(warning, "clientid '~s' conflict with ~p", [ClientId, NewPid]),
|
|
|
shutdown(conflict, State);
|
|
|
|
|
|
-handle_info({TcpOrSsL, _Sock, Data}, State) when TcpOrSsL =:= tcp; TcpOrSsL =:= ssl ->
|
|
|
- process_incoming(Data, State);
|
|
|
-
|
|
|
-%% Rate limit here, cool:)
|
|
|
-handle_info({tcp_passive, _Sock}, State) ->
|
|
|
- {noreply, run_socket(ensure_rate_limit(State))};
|
|
|
-%% FIXME Later
|
|
|
-handle_info({ssl_passive, _Sock}, State) ->
|
|
|
- {noreply, run_socket(ensure_rate_limit(State))};
|
|
|
-
|
|
|
-handle_info({Err, _Sock, Reason}, State) when Err =:= tcp_error; Err =:= ssl_error ->
|
|
|
+handle(info, {shutdown, Reason}, State) ->
|
|
|
shutdown(Reason, State);
|
|
|
|
|
|
-handle_info({Closed, _Sock}, State) when Closed =:= tcp_closed; Closed =:= ssl_closed ->
|
|
|
- shutdown(closed, State);
|
|
|
-
|
|
|
-%% Rate limit timer
|
|
|
-handle_info(activate_sock, State) ->
|
|
|
- {noreply, run_socket(State#state{conn_state = running, limit_timer = undefined})};
|
|
|
-
|
|
|
-handle_info({inet_reply, _Sock, ok}, State) ->
|
|
|
- {noreply, State};
|
|
|
-
|
|
|
-handle_info({inet_reply, _Sock, {error, Reason}}, State) ->
|
|
|
- shutdown(Reason, State);
|
|
|
-
|
|
|
-handle_info({keepalive, start, Interval}, State = #state{transport = Transport, socket = Socket}) ->
|
|
|
- ?LOG(debug, "Keepalive at the interval of ~p", [Interval]),
|
|
|
- StatFun = fun() ->
|
|
|
- case Transport:getstat(Socket, [recv_oct]) of
|
|
|
- {ok, [{recv_oct, RecvOct}]} -> {ok, RecvOct};
|
|
|
- Error -> Error
|
|
|
- end
|
|
|
- end,
|
|
|
- case emqx_keepalive:start(StatFun, Interval, {keepalive, check}) of
|
|
|
- {ok, KeepAlive} ->
|
|
|
- {noreply, State#state{keepalive = KeepAlive}};
|
|
|
- {error, Error} ->
|
|
|
- shutdown(Error, State)
|
|
|
- end;
|
|
|
-
|
|
|
-handle_info({keepalive, check}, State = #state{keepalive = KeepAlive}) ->
|
|
|
- case emqx_keepalive:check(KeepAlive) of
|
|
|
- {ok, KeepAlive1} ->
|
|
|
- {noreply, State#state{keepalive = KeepAlive1}};
|
|
|
- {error, timeout} ->
|
|
|
- shutdown(keepalive_timeout, State);
|
|
|
- {error, Error} ->
|
|
|
- shutdown(Error, State)
|
|
|
- end;
|
|
|
-
|
|
|
-handle_info(Info, State) ->
|
|
|
+handle(info, Info, State) ->
|
|
|
?LOG(error, "unexpected info: ~p", [Info]),
|
|
|
- {noreply, State}.
|
|
|
+ {keep_state, State}.
|
|
|
+
|
|
|
+code_change(_Vsn, State, Data, _Extra) ->
|
|
|
+ {ok, State, Data}.
|
|
|
|
|
|
-terminate(Reason, #state{transport = Transport,
|
|
|
- socket = Socket,
|
|
|
- keepalive = KeepAlive,
|
|
|
- proto_state = ProtoState}) ->
|
|
|
+terminate(Reason, _StateName, #state{transport = Transport,
|
|
|
+ socket = Socket,
|
|
|
+ keepalive = KeepAlive,
|
|
|
+ proto_state = ProtoState}) ->
|
|
|
?LOG(debug, "Terminated for ~p", [Reason]),
|
|
|
Transport:fast_close(Socket),
|
|
|
emqx_keepalive:cancel(KeepAlive),
|
|
|
case {ProtoState, Reason} of
|
|
|
{undefined, _} -> ok;
|
|
|
{_, {shutdown, Error}} ->
|
|
|
- emqx_protocol:shutdown(Error, ProtoState);
|
|
|
+ emqx_protocol:terminate(Error, ProtoState);
|
|
|
{_, Reason} ->
|
|
|
- emqx_protocol:shutdown(Reason, ProtoState)
|
|
|
+ emqx_protocol:terminate(Reason, ProtoState)
|
|
|
end.
|
|
|
|
|
|
-code_change(_OldVsn, State, _Extra) ->
|
|
|
- {ok, State}.
|
|
|
-
|
|
|
-%%------------------------------------------------------------------------------
|
|
|
-%% Internals: process incoming, parse and handle packets
|
|
|
%%------------------------------------------------------------------------------
|
|
|
+%% Process incoming data
|
|
|
|
|
|
-process_incoming(Data, State) ->
|
|
|
- Oct = iolist_size(Data),
|
|
|
- ?LOG(debug, "RECV ~p", [Data]),
|
|
|
- emqx_pd:update_counter(incoming_bytes, Oct),
|
|
|
- emqx_metrics:trans(inc, 'bytes/received', Oct),
|
|
|
- case handle_packet(Data, State) of
|
|
|
- {noreply, State1} ->
|
|
|
- State2 = maybe_gc({1, Oct}, State1),
|
|
|
- {noreply, ensure_stats_timer(State2)};
|
|
|
- Shutdown -> Shutdown
|
|
|
- end.
|
|
|
+process_incoming(<<>>, Packets, State) ->
|
|
|
+ {keep_state, State, next_events(Packets)};
|
|
|
|
|
|
-%% Parse and handle packets
|
|
|
-handle_packet(<<>>, State) ->
|
|
|
- {noreply, State};
|
|
|
-
|
|
|
-handle_packet(Data, State = #state{proto_state = ProtoState,
|
|
|
- parser_state = ParserState,
|
|
|
- idle_timeout = IdleTimeout}) ->
|
|
|
- try emqx_frame:parse(Data, ParserState) of
|
|
|
- {more, ParserState1} ->
|
|
|
- {noreply, State#state{parser_state = ParserState1}, IdleTimeout};
|
|
|
- {ok, Packet = ?PACKET(Type), Rest} ->
|
|
|
- emqx_metrics:received(Packet),
|
|
|
- (Type == ?PUBLISH) andalso emqx_pd:update_counter(incoming_pubs, 1),
|
|
|
- case emqx_protocol:received(Packet, ProtoState) of
|
|
|
- {ok, ProtoState1} ->
|
|
|
- handle_packet(Rest, reset_parser(State#state{proto_state = ProtoState1}));
|
|
|
- {error, Reason} ->
|
|
|
- ?LOG(error, "Process packet error - ~p", [Reason]),
|
|
|
- shutdown(Reason, State);
|
|
|
- {error, Reason, ProtoState1} ->
|
|
|
- shutdown(Reason, State#state{proto_state = ProtoState1});
|
|
|
- {stop, Error, ProtoState1} ->
|
|
|
- stop(Error, State#state{proto_state = ProtoState1})
|
|
|
- end;
|
|
|
+process_incoming(Data, Packets, State = #state{parse_state = ParseState}) ->
|
|
|
+ try emqx_frame:parse(Data, ParseState) of
|
|
|
+ {ok, Packet, Rest} ->
|
|
|
+ process_incoming(Rest, [Packet|Packets], reset_parser(State));
|
|
|
+ {more, NewParseState} ->
|
|
|
+ {keep_state, State#state{parse_state = NewParseState}, next_events(Packets)};
|
|
|
{error, Reason} ->
|
|
|
- ?LOG(error, "Parse frame error - ~p", [Reason]),
|
|
|
shutdown(Reason, State)
|
|
|
catch
|
|
|
- _:Error ->
|
|
|
- ?LOG(error, "Parse failed for ~p~nError data:~p", [Error, Data]),
|
|
|
- shutdown(parse_error, State)
|
|
|
+ _:Error:Stk->
|
|
|
+ ?LOG(error, "Parse failed for ~p~nStacktrace:~p~nError data:~p", [Error, Stk, Data]),
|
|
|
+ shutdown(Error, State)
|
|
|
end.
|
|
|
|
|
|
reset_parser(State = #state{proto_state = ProtoState}) ->
|
|
|
- State#state{parser_state = emqx_protocol:parser(ProtoState)}.
|
|
|
+ State#state{parse_state = emqx_protocol:parser(ProtoState)}.
|
|
|
+
|
|
|
+next_events([]) ->
|
|
|
+ [];
|
|
|
+next_events([Packet]) ->
|
|
|
+ {next_event, cast, {incoming, Packet}};
|
|
|
+next_events(Packets) ->
|
|
|
+ [next_events([Packet]) || Packet <- lists:reverse(Packets)].
|
|
|
+
|
|
|
+%%------------------------------------------------------------------------------
|
|
|
+%% Handle incoming packet
|
|
|
+
|
|
|
+handle_packet(Packet, SuccFun, State = #state{proto_state = ProtoState}) ->
|
|
|
+ case emqx_protocol:received(Packet, ProtoState) of
|
|
|
+ {ok, NProtoState} ->
|
|
|
+ SuccFun(State#state{proto_state = NProtoState});
|
|
|
+ {error, Reason} ->
|
|
|
+ shutdown(Reason, State);
|
|
|
+ {error, Reason, NProtoState} ->
|
|
|
+ shutdown(Reason, State#state{proto_state = NProtoState});
|
|
|
+ {stop, Error, NProtoState} ->
|
|
|
+ stop(Error, State#state{proto_state = NProtoState})
|
|
|
+ end.
|
|
|
|
|
|
%%------------------------------------------------------------------------------
|
|
|
%% Ensure rate limit
|
|
|
@@ -389,27 +422,27 @@ ensure_rate_limit([{Rl, Pos, Cnt}|Limiters], State) ->
|
|
|
{0, Rl1} ->
|
|
|
ensure_rate_limit(Limiters, setelement(Pos, State, Rl1));
|
|
|
{Pause, Rl1} ->
|
|
|
- TRef = erlang:send_after(Pause, self(), activate_sock),
|
|
|
+ TRef = erlang:send_after(Pause, self(), activate_socket),
|
|
|
setelement(Pos, State#state{conn_state = blocked, limit_timer = TRef}, Rl1)
|
|
|
end.
|
|
|
|
|
|
%%------------------------------------------------------------------------------
|
|
|
%% Activate socket
|
|
|
|
|
|
-run_socket(State = #state{conn_state = blocked}) ->
|
|
|
- State;
|
|
|
+activate_socket(#state{conn_state = blocked}) ->
|
|
|
+ ok;
|
|
|
|
|
|
-run_socket(State = #state{transport = Transport, socket = Socket, active_n = N}) ->
|
|
|
+activate_socket(#state{transport = Transport, socket = Socket, active_n = N}) ->
|
|
|
TrueOrN = case Transport:is_ssl(Socket) of
|
|
|
true -> true; %% Cannot set '{active, N}' for SSL:(
|
|
|
false -> N
|
|
|
end,
|
|
|
- ensure_ok_or_exit(Transport:setopts(Socket, [{active, TrueOrN}])),
|
|
|
- State.
|
|
|
-
|
|
|
-ensure_ok_or_exit(ok) -> ok;
|
|
|
-ensure_ok_or_exit({error, Reason}) ->
|
|
|
- self() ! {shutdown, Reason}.
|
|
|
+ case Transport:setopts(Socket, [{active, TrueOrN}]) of
|
|
|
+ ok -> ok;
|
|
|
+ {error, Reason} ->
|
|
|
+ self() ! {shutdown, Reason},
|
|
|
+ ok
|
|
|
+ end.
|
|
|
|
|
|
%%------------------------------------------------------------------------------
|
|
|
%% Ensure stats timer
|
|
|
@@ -418,6 +451,7 @@ ensure_stats_timer(State = #state{enable_stats = true,
|
|
|
stats_timer = undefined,
|
|
|
idle_timeout = IdleTimeout}) ->
|
|
|
State#state{stats_timer = emqx_misc:start_timer(IdleTimeout, emit_stats)};
|
|
|
+
|
|
|
ensure_stats_timer(State) -> State.
|
|
|
|
|
|
%%------------------------------------------------------------------------------
|
|
|
@@ -425,20 +459,28 @@ ensure_stats_timer(State) -> State.
|
|
|
|
|
|
maybe_gc(_, State = #state{gc_state = undefined}) ->
|
|
|
State;
|
|
|
-maybe_gc({publish, _PacketId, #message{payload = Payload}}, State) ->
|
|
|
+maybe_gc({publish, _, #message{payload = Payload}}, State) ->
|
|
|
Oct = iolist_size(Payload),
|
|
|
maybe_gc({1, Oct}, State);
|
|
|
+maybe_gc(Packets, State) when is_list(Packets) ->
|
|
|
+ {Cnt, Oct} =
|
|
|
+ lists:unzip([{1, iolist_size(Payload)}
|
|
|
+ || {publish, _, #message{payload = Payload}} <- Packets]),
|
|
|
+ maybe_gc({lists:sum(Cnt), lists:sum(Oct)}, State);
|
|
|
maybe_gc({Cnt, Oct}, State = #state{gc_state = GCSt}) ->
|
|
|
{_, GCSt1} = emqx_gc:run(Cnt, Oct, GCSt),
|
|
|
State#state{gc_state = GCSt1};
|
|
|
-maybe_gc(_, State) ->
|
|
|
- State.
|
|
|
+maybe_gc(_, State) -> State.
|
|
|
|
|
|
%%------------------------------------------------------------------------------
|
|
|
-%% Shutdown or stop
|
|
|
+%% Helper functions
|
|
|
+
|
|
|
+reply(From, Reply, State) ->
|
|
|
+ {keep_state, State, [{reply, From, Reply}]}.
|
|
|
|
|
|
shutdown(Reason, State) ->
|
|
|
stop({shutdown, Reason}, State).
|
|
|
|
|
|
stop(Reason, State) ->
|
|
|
{stop, Reason, State}.
|
|
|
+
|