mirror of
https://github.com/processone/ejabberd.git
synced 2024-11-22 16:20:52 +01:00
XMPP Ping support (thanks to Brian Cully)
SVN Revision: 2401
This commit is contained in:
parent
a528c62bba
commit
458b28eeff
@ -33,6 +33,8 @@
|
|||||||
-export([start_link/0]).
|
-export([start_link/0]).
|
||||||
|
|
||||||
-export([route/3,
|
-export([route/3,
|
||||||
|
route_iq/4,
|
||||||
|
process_iq_reply/3,
|
||||||
register_iq_handler/4,
|
register_iq_handler/4,
|
||||||
register_iq_handler/5,
|
register_iq_handler/5,
|
||||||
register_iq_response_handler/4,
|
register_iq_response_handler/4,
|
||||||
@ -51,10 +53,13 @@
|
|||||||
|
|
||||||
-record(state, {}).
|
-record(state, {}).
|
||||||
|
|
||||||
-record(iq_response, {id, module, function}).
|
-record(iq_response, {id, module, function, timer}).
|
||||||
|
|
||||||
-define(IQTABLE, local_iqtable).
|
-define(IQTABLE, local_iqtable).
|
||||||
|
|
||||||
|
%% This value is used in SIP and Megaco for a transaction lifetime.
|
||||||
|
-define(IQ_TIMEOUT, 32000).
|
||||||
|
|
||||||
%%====================================================================
|
%%====================================================================
|
||||||
%% API
|
%% API
|
||||||
%%====================================================================
|
%%====================================================================
|
||||||
@ -89,36 +94,24 @@ process_iq(From, To, Packet) ->
|
|||||||
ejabberd_router:route(To, From, Err)
|
ejabberd_router:route(To, From, Err)
|
||||||
end;
|
end;
|
||||||
reply ->
|
reply ->
|
||||||
process_iq_reply(From, To, Packet);
|
IQReply = jlib:iq_query_or_response_info(Packet),
|
||||||
|
process_iq_reply(From, To, IQReply);
|
||||||
_ ->
|
_ ->
|
||||||
Err = jlib:make_error_reply(Packet, ?ERR_BAD_REQUEST),
|
Err = jlib:make_error_reply(Packet, ?ERR_BAD_REQUEST),
|
||||||
ejabberd_router:route(To, From, Err),
|
ejabberd_router:route(To, From, Err),
|
||||||
ok
|
ok
|
||||||
end.
|
end.
|
||||||
|
|
||||||
process_iq_reply(From, To, Packet) ->
|
process_iq_reply(From, To, #iq{id = ID} = IQ) ->
|
||||||
IQ = jlib:iq_query_or_response_info(Packet),
|
case get_iq_callback(ID) of
|
||||||
#iq{id = ID} = IQ,
|
{ok, undefined, Function} ->
|
||||||
case catch mnesia:dirty_read(iq_response, ID) of
|
Function(IQ),
|
||||||
[] ->
|
ok;
|
||||||
|
{ok, Module, Function} ->
|
||||||
|
Module:Function(From, To, IQ),
|
||||||
ok;
|
ok;
|
||||||
_ ->
|
_ ->
|
||||||
F = fun() ->
|
nothing
|
||||||
case mnesia:read({iq_response, ID}) of
|
|
||||||
[] ->
|
|
||||||
nothing;
|
|
||||||
[#iq_response{module = Module,
|
|
||||||
function = Function}] ->
|
|
||||||
mnesia:delete({iq_response, ID}),
|
|
||||||
{Module, Function}
|
|
||||||
end
|
|
||||||
end,
|
|
||||||
case mnesia:transaction(F) of
|
|
||||||
{atomic, {Module, Function}} ->
|
|
||||||
Module:Function(From, To, IQ);
|
|
||||||
_ ->
|
|
||||||
ok
|
|
||||||
end
|
|
||||||
end.
|
end.
|
||||||
|
|
||||||
route(From, To, Packet) ->
|
route(From, To, Packet) ->
|
||||||
@ -130,8 +123,21 @@ route(From, To, Packet) ->
|
|||||||
ok
|
ok
|
||||||
end.
|
end.
|
||||||
|
|
||||||
|
route_iq(From, To, #iq{type = Type} = IQ, F) when is_function(F) ->
|
||||||
|
Packet = if Type == set; Type == get ->
|
||||||
|
ID = randoms:get_string(),
|
||||||
|
Host = From#jid.lserver,
|
||||||
|
register_iq_response_handler(Host, ID, undefined, F),
|
||||||
|
jlib:iq_to_xml(IQ#iq{id = ID});
|
||||||
|
true ->
|
||||||
|
jlib:iq_to_xml(IQ)
|
||||||
|
end,
|
||||||
|
ejabberd_router:route(From, To, Packet).
|
||||||
|
|
||||||
register_iq_response_handler(Host, ID, Module, Fun) ->
|
register_iq_response_handler(Host, ID, Module, Fun) ->
|
||||||
ejabberd_local ! {register_iq_response_handler, Host, ID, Module, Fun}.
|
gen_server:call(ejabberd_local,
|
||||||
|
{register_iq_response_handler,
|
||||||
|
Host, ID, Module, Fun}).
|
||||||
|
|
||||||
register_iq_handler(Host, XMLNS, Module, Fun) ->
|
register_iq_handler(Host, XMLNS, Module, Fun) ->
|
||||||
ejabberd_local ! {register_iq_handler, Host, XMLNS, Module, Fun}.
|
ejabberd_local ! {register_iq_handler, Host, XMLNS, Module, Fun}.
|
||||||
@ -139,8 +145,9 @@ register_iq_handler(Host, XMLNS, Module, Fun) ->
|
|||||||
register_iq_handler(Host, XMLNS, Module, Fun, Opts) ->
|
register_iq_handler(Host, XMLNS, Module, Fun, Opts) ->
|
||||||
ejabberd_local ! {register_iq_handler, Host, XMLNS, Module, Fun, Opts}.
|
ejabberd_local ! {register_iq_handler, Host, XMLNS, Module, Fun, Opts}.
|
||||||
|
|
||||||
unregister_iq_response_handler(Host, ID) ->
|
unregister_iq_response_handler(_Host, ID) ->
|
||||||
ejabberd_local ! {unregister_iq_response_handler, Host, ID}.
|
catch get_iq_callback(ID),
|
||||||
|
ok.
|
||||||
|
|
||||||
unregister_iq_handler(Host, XMLNS) ->
|
unregister_iq_handler(Host, XMLNS) ->
|
||||||
ejabberd_local ! {unregister_iq_handler, Host, XMLNS}.
|
ejabberd_local ! {unregister_iq_handler, Host, XMLNS}.
|
||||||
@ -172,6 +179,7 @@ init([]) ->
|
|||||||
?MODULE, bounce_resource_packet, 100)
|
?MODULE, bounce_resource_packet, 100)
|
||||||
end, ?MYHOSTS),
|
end, ?MYHOSTS),
|
||||||
catch ets:new(?IQTABLE, [named_table, public]),
|
catch ets:new(?IQTABLE, [named_table, public]),
|
||||||
|
update_table(),
|
||||||
mnesia:create_table(iq_response,
|
mnesia:create_table(iq_response,
|
||||||
[{ram_copies, [node()]},
|
[{ram_copies, [node()]},
|
||||||
{attributes, record_info(fields, iq_response)}]),
|
{attributes, record_info(fields, iq_response)}]),
|
||||||
@ -187,6 +195,14 @@ init([]) ->
|
|||||||
%% {stop, Reason, State}
|
%% {stop, Reason, State}
|
||||||
%% Description: Handling call messages
|
%% Description: Handling call messages
|
||||||
%%--------------------------------------------------------------------
|
%%--------------------------------------------------------------------
|
||||||
|
handle_call({register_iq_response_handler, _Host,
|
||||||
|
ID, Module, Function}, _From, State) ->
|
||||||
|
TRef = erlang:start_timer(?IQ_TIMEOUT, self(), ID),
|
||||||
|
mnesia:dirty_write(#iq_response{id = ID,
|
||||||
|
module = Module,
|
||||||
|
function = Function,
|
||||||
|
timer = TRef}),
|
||||||
|
{reply, ok, State};
|
||||||
handle_call(_Request, _From, State) ->
|
handle_call(_Request, _From, State) ->
|
||||||
Reply = ok,
|
Reply = ok,
|
||||||
{reply, Reply, State}.
|
{reply, Reply, State}.
|
||||||
@ -215,12 +231,6 @@ handle_info({route, From, To, Packet}, State) ->
|
|||||||
ok
|
ok
|
||||||
end,
|
end,
|
||||||
{noreply, State};
|
{noreply, State};
|
||||||
handle_info({register_iq_response_handler, _Host, ID, Module, Function}, State) ->
|
|
||||||
mnesia:dirty_write(#iq_response{id = ID, module = Module, function = Function}),
|
|
||||||
{noreply, State};
|
|
||||||
handle_info({unregister_iq_response_handler, _Host, ID}, State) ->
|
|
||||||
mnesia:dirty_delete({iq_response, ID}),
|
|
||||||
{noreply, State};
|
|
||||||
handle_info({register_iq_handler, Host, XMLNS, Module, Function}, State) ->
|
handle_info({register_iq_handler, Host, XMLNS, Module, Function}, State) ->
|
||||||
ets:insert(?IQTABLE, {{XMLNS, Host}, Module, Function}),
|
ets:insert(?IQTABLE, {{XMLNS, Host}, Module, Function}),
|
||||||
catch mod_disco:register_feature(Host, XMLNS),
|
catch mod_disco:register_feature(Host, XMLNS),
|
||||||
@ -252,6 +262,9 @@ handle_info(refresh_iq_handlers, State) ->
|
|||||||
end
|
end
|
||||||
end, ets:tab2list(?IQTABLE)),
|
end, ets:tab2list(?IQTABLE)),
|
||||||
{noreply, State};
|
{noreply, State};
|
||||||
|
handle_info({timeout, _TRef, ID}, State) ->
|
||||||
|
process_iq_timeout(ID),
|
||||||
|
{noreply, State};
|
||||||
handle_info(_Info, State) ->
|
handle_info(_Info, State) ->
|
||||||
{noreply, State}.
|
{noreply, State}.
|
||||||
|
|
||||||
@ -305,3 +318,52 @@ do_route(From, To, Packet) ->
|
|||||||
end
|
end
|
||||||
end.
|
end.
|
||||||
|
|
||||||
|
update_table() ->
|
||||||
|
case catch mnesia:table_info(iq_response, attributes) of
|
||||||
|
[id, module, function] ->
|
||||||
|
mnesia:delete_table(iq_response);
|
||||||
|
[id, module, function, timer] ->
|
||||||
|
ok;
|
||||||
|
{'EXIT', _} ->
|
||||||
|
ok
|
||||||
|
end.
|
||||||
|
|
||||||
|
get_iq_callback(ID) ->
|
||||||
|
case mnesia:dirty_read(iq_response, ID) of
|
||||||
|
[#iq_response{module = Module, timer = TRef,
|
||||||
|
function = Function}] ->
|
||||||
|
cancel_timer(TRef),
|
||||||
|
mnesia:dirty_delete(iq_response, ID),
|
||||||
|
{ok, Module, Function};
|
||||||
|
_ ->
|
||||||
|
error
|
||||||
|
end.
|
||||||
|
|
||||||
|
process_iq_timeout(ID) ->
|
||||||
|
spawn(fun process_iq_timeout/0) ! ID.
|
||||||
|
|
||||||
|
process_iq_timeout() ->
|
||||||
|
receive
|
||||||
|
ID ->
|
||||||
|
case get_iq_callback(ID) of
|
||||||
|
{ok, undefined, Function} ->
|
||||||
|
Function(timeout);
|
||||||
|
_ ->
|
||||||
|
ok
|
||||||
|
end
|
||||||
|
after 5000 ->
|
||||||
|
ok
|
||||||
|
end.
|
||||||
|
|
||||||
|
cancel_timer(TRef) ->
|
||||||
|
case erlang:cancel_timer(TRef) of
|
||||||
|
false ->
|
||||||
|
receive
|
||||||
|
{timeout, TRef, _} ->
|
||||||
|
ok
|
||||||
|
after 0 ->
|
||||||
|
ok
|
||||||
|
end;
|
||||||
|
_ ->
|
||||||
|
ok
|
||||||
|
end.
|
||||||
|
222
src/mod_ping.erl
Normal file
222
src/mod_ping.erl
Normal file
@ -0,0 +1,222 @@
|
|||||||
|
%%%-------------------------------------------------------------------
|
||||||
|
%%% @doc Implements support for XEP-0199 (XMPP Ping) and periodic
|
||||||
|
%%% keepalives.
|
||||||
|
%%%
|
||||||
|
%%% <p>When enabled (see below), ejabberd will respond correctly to
|
||||||
|
%%% ping packets, as defined in XEP-0199.</p>
|
||||||
|
%%%
|
||||||
|
%%% <p>In addition you can have the server generate pings to clients
|
||||||
|
%%% as a method of keeping them alive or checking
|
||||||
|
%%% availibility. However, this feature is disabled by default since
|
||||||
|
%%% it is mostly not needed and consumes resources. For "interesting"
|
||||||
|
%%% uses it can be enabled in the config (see below).</p>
|
||||||
|
%%%
|
||||||
|
%%% <p>To use this module simply include it in the modules section of
|
||||||
|
%%% your ejabberd config.</p>
|
||||||
|
%%%
|
||||||
|
%%% <p>Configuration options:</p>
|
||||||
|
%%% <dl>
|
||||||
|
%%% <dt>{send_pings, true | false}</dt>
|
||||||
|
%%% <dd>Whether to send pings to connected clients.</dd>
|
||||||
|
%%% <dt>{ping_interval, Seconds}</dt>
|
||||||
|
%%% <dd>How often to send pings to connected clients.</dd>
|
||||||
|
%%% </dl>
|
||||||
|
%%%
|
||||||
|
%%% @reference <a
|
||||||
|
%%% href="http://xmpp.org/extensions/xep-0199.html">XEP-0199</a>
|
||||||
|
%%% @end
|
||||||
|
%%% -------------------------------------------------------------------
|
||||||
|
-module(mod_ping).
|
||||||
|
-author('bjc@kublai.com').
|
||||||
|
|
||||||
|
-behavior(gen_mod).
|
||||||
|
-behavior(gen_server).
|
||||||
|
|
||||||
|
-include("ejabberd.hrl").
|
||||||
|
-include("jlib.hrl").
|
||||||
|
|
||||||
|
-define(SUPERVISOR, ejabberd_sup).
|
||||||
|
-define(NS_PING, "urn:xmpp:ping").
|
||||||
|
-define(DEFAULT_SEND_PINGS, false). % bool()
|
||||||
|
-define(DEFAULT_PING_INTERVAL, 60). % seconds
|
||||||
|
|
||||||
|
-define(DICT, dict).
|
||||||
|
|
||||||
|
%% API
|
||||||
|
-export([start_link/2, start_ping/2, stop_ping/2]).
|
||||||
|
|
||||||
|
%% gen_mod callbacks
|
||||||
|
-export([start/2, stop/1]).
|
||||||
|
|
||||||
|
%% gen_server callbacks
|
||||||
|
-export([init/1, terminate/2, handle_call/3, handle_cast/2,
|
||||||
|
handle_info/2, code_change/3]).
|
||||||
|
|
||||||
|
%% Hook callbacks
|
||||||
|
-export([iq_ping/3, user_online/3, user_offline/3, user_send/3]).
|
||||||
|
|
||||||
|
-record(state, {host = "",
|
||||||
|
send_pings = ?DEFAULT_SEND_PINGS,
|
||||||
|
ping_interval = ?DEFAULT_PING_INTERVAL,
|
||||||
|
timers = ?DICT:new()}).
|
||||||
|
|
||||||
|
%%====================================================================
|
||||||
|
%% API
|
||||||
|
%%====================================================================
|
||||||
|
start_link(Host, Opts) ->
|
||||||
|
Proc = gen_mod:get_module_proc(Host, ?MODULE),
|
||||||
|
gen_server:start_link({local, Proc}, ?MODULE, [Host, Opts], []).
|
||||||
|
|
||||||
|
start_ping(Host, JID) ->
|
||||||
|
Proc = gen_mod:get_module_proc(Host, ?MODULE),
|
||||||
|
gen_server:cast(Proc, {start_ping, JID}).
|
||||||
|
|
||||||
|
stop_ping(Host, JID) ->
|
||||||
|
Proc = gen_mod:get_module_proc(Host, ?MODULE),
|
||||||
|
gen_server:cast(Proc, {stop_ping, JID}).
|
||||||
|
|
||||||
|
%%====================================================================
|
||||||
|
%% gen_mod callbacks
|
||||||
|
%%====================================================================
|
||||||
|
start(Host, Opts) ->
|
||||||
|
Proc = gen_mod:get_module_proc(Host, ?MODULE),
|
||||||
|
PingSpec = {Proc, {?MODULE, start_link, [Host, Opts]},
|
||||||
|
transient, 2000, worker, [?MODULE]},
|
||||||
|
supervisor:start_child(?SUPERVISOR, PingSpec).
|
||||||
|
|
||||||
|
stop(Host) ->
|
||||||
|
Proc = gen_mod:get_module_proc(Host, ?MODULE),
|
||||||
|
gen_server:call(Proc, stop),
|
||||||
|
supervisor:delete_child(?SUPERVISOR, Proc).
|
||||||
|
|
||||||
|
%%====================================================================
|
||||||
|
%% gen_server callbacks
|
||||||
|
%%====================================================================
|
||||||
|
init([Host, Opts]) ->
|
||||||
|
SendPings = gen_mod:get_opt(send_pings, Opts, ?DEFAULT_SEND_PINGS),
|
||||||
|
PingInterval = gen_mod:get_opt(ping_interval, Opts, ?DEFAULT_PING_INTERVAL),
|
||||||
|
IQDisc = gen_mod:get_opt(iqdisc, Opts, one_queue),
|
||||||
|
mod_disco:register_feature(Host, ?NS_PING),
|
||||||
|
gen_iq_handler:add_iq_handler(ejabberd_sm, Host, ?NS_PING,
|
||||||
|
?MODULE, iq_ping, IQDisc),
|
||||||
|
gen_iq_handler:add_iq_handler(ejabberd_local, Host, ?NS_PING,
|
||||||
|
?MODULE, iq_ping, IQDisc),
|
||||||
|
case SendPings of
|
||||||
|
true ->
|
||||||
|
ejabberd_hooks:add(sm_register_connection_hook, Host,
|
||||||
|
?MODULE, user_online, 100),
|
||||||
|
ejabberd_hooks:add(sm_remove_connection_hook, Host,
|
||||||
|
?MODULE, user_offline, 100),
|
||||||
|
ejabberd_hooks:add(user_send_packet, Host,
|
||||||
|
?MODULE, user_send, 100);
|
||||||
|
_ ->
|
||||||
|
ok
|
||||||
|
end,
|
||||||
|
{ok, #state{host = Host,
|
||||||
|
send_pings = SendPings,
|
||||||
|
ping_interval = PingInterval,
|
||||||
|
timers = ?DICT:new()}}.
|
||||||
|
|
||||||
|
terminate(_Reason, #state{host = Host}) ->
|
||||||
|
ejabberd_hooks:delete(sm_remove_connection_hook, Host,
|
||||||
|
?MODULE, user_offline, 100),
|
||||||
|
ejabberd_hooks:delete(sm_register_connection_hook, Host,
|
||||||
|
?MODULE, user_online, 100),
|
||||||
|
ejabberd_hooks:delete(user_send_packet, Host,
|
||||||
|
?MODULE, user_send, 100),
|
||||||
|
gen_iq_handler:remove_iq_handler(ejabberd_local, Host, ?NS_PING),
|
||||||
|
gen_iq_handler:remove_iq_handler(ejabberd_sm, Host, ?NS_PING),
|
||||||
|
mod_disco:unregister_feature(Host, ?NS_PING).
|
||||||
|
|
||||||
|
handle_call(stop, _From, State) ->
|
||||||
|
{stop, normal, ok, State};
|
||||||
|
handle_call(_Req, _From, State) ->
|
||||||
|
{reply, {error, badarg}, State}.
|
||||||
|
|
||||||
|
handle_cast({start_ping, JID}, State) ->
|
||||||
|
Timers = add_timer(JID, State#state.ping_interval, State#state.timers),
|
||||||
|
{noreply, State#state{timers = Timers}};
|
||||||
|
handle_cast({stop_ping, JID}, State) ->
|
||||||
|
Timers = del_timer(JID, State#state.timers),
|
||||||
|
{noreply, State#state{timers = Timers}};
|
||||||
|
handle_cast({iq_pong, JID, timeout}, State) ->
|
||||||
|
Timers = del_timer(JID, State#state.timers),
|
||||||
|
ejabberd_hooks:run(user_ping_timeout, State#state.host, [JID]),
|
||||||
|
{noreply, State#state{timers = Timers}};
|
||||||
|
handle_cast(_Msg, State) ->
|
||||||
|
{noreply, State}.
|
||||||
|
|
||||||
|
handle_info({timeout, _TRef, {ping, JID}}, State) ->
|
||||||
|
IQ = #iq{type = get,
|
||||||
|
sub_el = [{xmlelement, "ping", [{"xmlns", ?NS_PING}], []}]},
|
||||||
|
Pid = self(),
|
||||||
|
F = fun(Response) ->
|
||||||
|
gen_server:cast(Pid, {iq_pong, JID, Response})
|
||||||
|
end,
|
||||||
|
From = jlib:make_jid("", State#state.host, ""),
|
||||||
|
ejabberd_local:route_iq(From, JID, IQ, F),
|
||||||
|
Timers = add_timer(JID, State#state.ping_interval, State#state.timers),
|
||||||
|
{noreply, State#state{timers = Timers}};
|
||||||
|
handle_info(_Info, State) ->
|
||||||
|
{noreply, State}.
|
||||||
|
|
||||||
|
code_change(_OldVsn, State, _Extra) ->
|
||||||
|
{ok, State}.
|
||||||
|
|
||||||
|
%%====================================================================
|
||||||
|
%% Hook callbacks
|
||||||
|
%%====================================================================
|
||||||
|
iq_ping(_From, _To, #iq{type = Type, sub_el = SubEl} = IQ) ->
|
||||||
|
case {Type, SubEl} of
|
||||||
|
{get, {xmlelement, "ping", _, _}} ->
|
||||||
|
IQ#iq{type = result, sub_el = []};
|
||||||
|
_ ->
|
||||||
|
IQ#iq{type = error, sub_el = [SubEl, ?ERR_FEATURE_NOT_IMPLEMENTED]}
|
||||||
|
end.
|
||||||
|
|
||||||
|
user_online(_SID, JID, _Info) ->
|
||||||
|
start_ping(JID#jid.lserver, JID).
|
||||||
|
|
||||||
|
user_offline(_SID, JID, _Info) ->
|
||||||
|
stop_ping(JID#jid.lserver, JID).
|
||||||
|
|
||||||
|
user_send(JID, _From, _Packet) ->
|
||||||
|
start_ping(JID#jid.lserver, JID).
|
||||||
|
|
||||||
|
%%====================================================================
|
||||||
|
%% Internal functions
|
||||||
|
%%====================================================================
|
||||||
|
add_timer(JID, Interval, Timers) ->
|
||||||
|
LJID = jlib:jid_tolower(JID),
|
||||||
|
NewTimers = case ?DICT:find(LJID, Timers) of
|
||||||
|
{ok, OldTRef} ->
|
||||||
|
cancel_timer(OldTRef),
|
||||||
|
?DICT:erase(LJID, Timers);
|
||||||
|
_ ->
|
||||||
|
Timers
|
||||||
|
end,
|
||||||
|
TRef = erlang:start_timer(Interval * 1000, self(), {ping, JID}),
|
||||||
|
?DICT:store(LJID, TRef, NewTimers).
|
||||||
|
|
||||||
|
del_timer(JID, Timers) ->
|
||||||
|
LJID = jlib:jid_tolower(JID),
|
||||||
|
case ?DICT:find(LJID, Timers) of
|
||||||
|
{ok, TRef} ->
|
||||||
|
cancel_timer(TRef),
|
||||||
|
?DICT:erase(LJID, Timers);
|
||||||
|
_ ->
|
||||||
|
Timers
|
||||||
|
end.
|
||||||
|
|
||||||
|
cancel_timer(TRef) ->
|
||||||
|
case erlang:cancel_timer(TRef) of
|
||||||
|
false ->
|
||||||
|
receive
|
||||||
|
{timeout, TRef, _} ->
|
||||||
|
ok
|
||||||
|
after 0 ->
|
||||||
|
ok
|
||||||
|
end;
|
||||||
|
_ ->
|
||||||
|
ok
|
||||||
|
end.
|
Loading…
Reference in New Issue
Block a user