Update mod_message_log to work with ejabberd 18.x
This commit is contained in:
parent
7d2a7aa418
commit
79a804ac3b
|
@ -11,11 +11,12 @@
|
||||||
-behaviour(gen_mod).
|
-behaviour(gen_mod).
|
||||||
-behaviour(gen_server).
|
-behaviour(gen_server).
|
||||||
|
|
||||||
%% gen_mod/supervisor callbacks.
|
%% gen_mod callbacks.
|
||||||
-export([start_link/1,
|
-export([start/2,
|
||||||
start/2,
|
|
||||||
stop/1,
|
stop/1,
|
||||||
mod_opt_type/1]).
|
mod_opt_type/1,
|
||||||
|
mod_options/1,
|
||||||
|
depends/2]).
|
||||||
|
|
||||||
%% gen_server callbacks.
|
%% gen_server callbacks.
|
||||||
-export([init/1,
|
-export([init/1,
|
||||||
|
@ -26,13 +27,12 @@
|
||||||
code_change/3]).
|
code_change/3]).
|
||||||
|
|
||||||
%% ejabberd_hooks callbacks.
|
%% ejabberd_hooks callbacks.
|
||||||
-export([log_packet_send/4,
|
-export([log_packet_send/1,
|
||||||
log_packet_receive/5,
|
log_packet_receive/1,
|
||||||
log_packet_offline/3,
|
log_packet_offline/1,
|
||||||
reopen_log/0]).
|
reopen_log/0]).
|
||||||
|
|
||||||
-include("ejabberd.hrl").
|
-include("xmpp.hrl").
|
||||||
-include("jlib.hrl").
|
|
||||||
|
|
||||||
-define(PROCNAME, ?MODULE).
|
-define(PROCNAME, ?MODULE).
|
||||||
-define(DEFAULT_FILENAME, <<"message.log">>).
|
-define(DEFAULT_FILENAME, <<"message.log">>).
|
||||||
|
@ -43,18 +43,13 @@
|
||||||
|
|
||||||
-type direction() :: incoming | outgoing | offline.
|
-type direction() :: incoming | outgoing | offline.
|
||||||
-type state() :: #state{}.
|
-type state() :: #state{}.
|
||||||
|
-type c2s_state() :: ejabberd_c2s:state().
|
||||||
|
-type c2s_hook_acc() :: {stanza() | drop, c2s_state()}.
|
||||||
|
|
||||||
%% -------------------------------------------------------------------
|
%% -------------------------------------------------------------------
|
||||||
%% gen_mod/supervisor callbacks.
|
%% gen_mod callbacks.
|
||||||
%% -------------------------------------------------------------------
|
%% -------------------------------------------------------------------
|
||||||
|
|
||||||
-spec start_link(gen_mod:opts()) -> {ok, pid()} | ignore | {error, _}.
|
|
||||||
|
|
||||||
start_link(Opts) ->
|
|
||||||
gen_server:start_link({local, ?PROCNAME}, ?MODULE, Opts, []).
|
|
||||||
|
|
||||||
-spec start(binary(), gen_mod:opts()) -> {ok, _} | {ok, _, _} | {error, _}.
|
-spec start(binary(), gen_mod:opts()) -> {ok, _} | {ok, _, _} | {error, _}.
|
||||||
|
|
||||||
start(Host, Opts) ->
|
start(Host, Opts) ->
|
||||||
ejabberd_hooks:add(user_send_packet, Host, ?MODULE,
|
ejabberd_hooks:add(user_send_packet, Host, ?MODULE,
|
||||||
log_packet_send, 42),
|
log_packet_send, 42),
|
||||||
|
@ -62,18 +57,16 @@ start(Host, Opts) ->
|
||||||
log_packet_receive, 42),
|
log_packet_receive, 42),
|
||||||
ejabberd_hooks:add(offline_message_hook, Host, ?MODULE,
|
ejabberd_hooks:add(offline_message_hook, Host, ?MODULE,
|
||||||
log_packet_offline, 42),
|
log_packet_offline, 42),
|
||||||
Spec = {
|
case gen_mod:start_child(?MODULE, Host, Opts) of
|
||||||
?PROCNAME,
|
{ok, Ref} ->
|
||||||
{?MODULE, start_link, [Opts]},
|
{ok, Ref};
|
||||||
permanent,
|
{error, {already_started, Ref}} ->
|
||||||
3000,
|
{ok, Ref};
|
||||||
worker,
|
{error, Reason} ->
|
||||||
[?MODULE]
|
{error, Reason}
|
||||||
},
|
end.
|
||||||
supervisor:start_child(ejabberd_sup, Spec).
|
|
||||||
|
|
||||||
-spec stop(binary()) -> ok.
|
-spec stop(binary()) -> ok.
|
||||||
|
|
||||||
stop(Host) ->
|
stop(Host) ->
|
||||||
ejabberd_hooks:delete(user_send_packet, Host, ?MODULE,
|
ejabberd_hooks:delete(user_send_packet, Host, ?MODULE,
|
||||||
log_packet_send, 42),
|
log_packet_send, 42),
|
||||||
|
@ -81,26 +74,25 @@ stop(Host) ->
|
||||||
log_packet_receive, 42),
|
log_packet_receive, 42),
|
||||||
ejabberd_hooks:delete(offline_message_hook, Host, ?MODULE,
|
ejabberd_hooks:delete(offline_message_hook, Host, ?MODULE,
|
||||||
log_packet_offline, 42),
|
log_packet_offline, 42),
|
||||||
case supervisor:terminate_child(ejabberd_sup, ?PROCNAME) of
|
gen_mod:stop_child(?MODULE, Host),
|
||||||
ok ->
|
ok.
|
||||||
ok = supervisor:delete_child(ejabberd_sup, ?PROCNAME);
|
|
||||||
{error, not_found} ->
|
|
||||||
ok % We just run one process per node.
|
|
||||||
end.
|
|
||||||
|
|
||||||
-spec mod_opt_type(atom()) -> fun((term()) -> term()) | [atom()].
|
|
||||||
|
|
||||||
|
-spec mod_opt_type(atom()) -> fun((term()) -> term()).
|
||||||
mod_opt_type(filename) ->
|
mod_opt_type(filename) ->
|
||||||
fun iolist_to_binary/1;
|
fun iolist_to_binary/1.
|
||||||
mod_opt_type(_) ->
|
|
||||||
[filename].
|
-spec mod_options(binary()) -> [{atom(), any()}].
|
||||||
|
mod_options(_Host) ->
|
||||||
|
[{filename, ?DEFAULT_FILENAME}].
|
||||||
|
|
||||||
|
-spec depends(binary(), gen_mod:opts()) -> [{module(), hard | soft}].
|
||||||
|
depends(_Host, _Opts) ->
|
||||||
|
[].
|
||||||
|
|
||||||
%% -------------------------------------------------------------------
|
%% -------------------------------------------------------------------
|
||||||
%% gen_server callbacks.
|
%% gen_server callbacks.
|
||||||
%% -------------------------------------------------------------------
|
%% -------------------------------------------------------------------
|
||||||
|
|
||||||
-spec init(gen_mod:opts()) -> {ok, state()}.
|
-spec init(gen_mod:opts()) -> {ok, state()}.
|
||||||
|
|
||||||
init(Opts) ->
|
init(Opts) ->
|
||||||
process_flag(trap_exit, true),
|
process_flag(trap_exit, true),
|
||||||
ejabberd_hooks:add(reopen_log_hook, ?MODULE, reopen_log, 42),
|
ejabberd_hooks:add(reopen_log_hook, ?MODULE, reopen_log, 42),
|
||||||
|
@ -110,12 +102,10 @@ init(Opts) ->
|
||||||
{ok, #state{filename = Filename, iodevice = IoDevice}}.
|
{ok, #state{filename = Filename, iodevice = IoDevice}}.
|
||||||
|
|
||||||
-spec handle_call(_, {pid(), _}, state()) -> {noreply, state()}.
|
-spec handle_call(_, {pid(), _}, state()) -> {noreply, state()}.
|
||||||
|
|
||||||
handle_call(_Request, _From, State) ->
|
handle_call(_Request, _From, State) ->
|
||||||
{noreply, State}.
|
{noreply, State}.
|
||||||
|
|
||||||
-spec handle_cast(_, state()) -> {noreply, state()}.
|
-spec handle_cast(_, state()) -> {noreply, state()}.
|
||||||
|
|
||||||
handle_cast({message, Direction, From, To, Type}, #state{iodevice = IoDevice} =
|
handle_cast({message, Direction, From, To, Type}, #state{iodevice = IoDevice} =
|
||||||
State) ->
|
State) ->
|
||||||
write_log(IoDevice, Direction, From, To, Type),
|
write_log(IoDevice, Direction, From, To, Type),
|
||||||
|
@ -129,106 +119,85 @@ handle_cast(_Request, State) ->
|
||||||
{noreply, State}.
|
{noreply, State}.
|
||||||
|
|
||||||
-spec handle_info(timeout | _, state()) -> {noreply, state()}.
|
-spec handle_info(timeout | _, state()) -> {noreply, state()}.
|
||||||
|
|
||||||
handle_info(_Info, State) ->
|
handle_info(_Info, State) ->
|
||||||
{noreply, State}.
|
{noreply, State}.
|
||||||
|
|
||||||
-spec terminate(normal | shutdown | {shutdown, _} | _, _) -> any().
|
-spec terminate(normal | shutdown | {shutdown, _} | _, _) -> any().
|
||||||
|
|
||||||
terminate(_Reason, State) ->
|
terminate(_Reason, State) ->
|
||||||
ejabberd_hooks:delete(reopen_log_hook, ?MODULE, reopen_log, 42),
|
ejabberd_hooks:delete(reopen_log_hook, ?MODULE, reopen_log, 42),
|
||||||
ok = file:close(State#state.iodevice).
|
ok = file:close(State#state.iodevice).
|
||||||
|
|
||||||
-spec code_change({down, _} | _, state(), _) -> {ok, state()}.
|
-spec code_change({down, _} | _, state(), _) -> {ok, state()}.
|
||||||
|
|
||||||
code_change(_OldVsn, State, _Extra) ->
|
code_change(_OldVsn, State, _Extra) ->
|
||||||
{ok, State}.
|
{ok, State}.
|
||||||
|
|
||||||
%% -------------------------------------------------------------------
|
%% -------------------------------------------------------------------
|
||||||
%% ejabberd_hooks callbacks.
|
%% ejabberd_hooks callbacks.
|
||||||
%% -------------------------------------------------------------------
|
%% -------------------------------------------------------------------
|
||||||
|
-spec log_packet_send(c2s_hook_acc()) -> c2s_hook_acc().
|
||||||
|
log_packet_send({#message{} = Msg, _C2SState} = Acc) ->
|
||||||
|
log_packet(outgoing, Msg),
|
||||||
|
Acc;
|
||||||
|
log_packet_send({_Stanza, _C2SState} = Acc) ->
|
||||||
|
Acc.
|
||||||
|
|
||||||
-spec log_packet_send(xmlel(), term(), jid(), jid()) -> xmlel().
|
-spec log_packet_receive(c2s_hook_acc()) -> c2s_hook_acc().
|
||||||
|
log_packet_receive({#message{} = Msg, _C2SState} = Acc) ->
|
||||||
|
log_packet(incoming, Msg),
|
||||||
|
Acc;
|
||||||
|
log_packet_receive({_Stanza, _C2SState} = Acc) ->
|
||||||
|
Acc.
|
||||||
|
|
||||||
log_packet_send(Packet, _C2SState, From, To) ->
|
-spec log_packet_offline(message()) -> any().
|
||||||
log_packet(outgoing, From, To, Packet),
|
log_packet_offline(Msg) ->
|
||||||
Packet.
|
log_packet(offline, Msg).
|
||||||
|
|
||||||
-spec log_packet_receive(xmlel(), term(), jid(), jid(), jid()) -> xmlel().
|
|
||||||
|
|
||||||
log_packet_receive(Packet, _C2SState, JID, From, _To) ->
|
|
||||||
log_packet(incoming, From, JID, Packet),
|
|
||||||
Packet.
|
|
||||||
|
|
||||||
-spec log_packet_offline(jid(), jid(), xmlel()) -> any().
|
|
||||||
|
|
||||||
log_packet_offline(From, To, Packet) ->
|
|
||||||
log_packet(offline, From, To, Packet).
|
|
||||||
|
|
||||||
-spec reopen_log() -> any().
|
-spec reopen_log() -> any().
|
||||||
|
|
||||||
reopen_log() ->
|
reopen_log() ->
|
||||||
gen_server:cast(?PROCNAME, reopen_log).
|
gen_server:cast(?PROCNAME, reopen_log).
|
||||||
|
|
||||||
%% -------------------------------------------------------------------
|
%% -------------------------------------------------------------------
|
||||||
%% Internal functions.
|
%% Internal functions.
|
||||||
%% -------------------------------------------------------------------
|
%% -------------------------------------------------------------------
|
||||||
|
-spec log_packet(direction(), message()) -> any().
|
||||||
|
log_packet(Direction, #message{from = From, to = To, type = Type} = Msg) ->
|
||||||
|
case should_log(Msg) of
|
||||||
|
true ->
|
||||||
|
{Type1, Direction1} = case is_carbon(Msg) of
|
||||||
|
{true, Direction0} ->
|
||||||
|
{carbon, Direction0};
|
||||||
|
false ->
|
||||||
|
{Type, Direction}
|
||||||
|
end,
|
||||||
|
gen_server:cast(?PROCNAME, {message, Direction1, From, To, Type1});
|
||||||
|
false ->
|
||||||
|
ok
|
||||||
|
end.
|
||||||
|
|
||||||
-spec log_packet(direction(), jid(), jid(), xmlel()) -> any().
|
-spec is_carbon(message()) -> {true, direction()} | false.
|
||||||
|
is_carbon(#message{meta = #{carbon_copy := true}} = Msg) ->
|
||||||
log_packet(Direction, From, To, #xmlel{name = <<"message">>} = Packet) ->
|
case xmpp:has_subtag(Msg, #carbons_sent{}) of
|
||||||
case fxml:get_subtag(Packet, <<"body">>) of
|
true ->
|
||||||
#xmlel{children = Body} when length(Body) > 0 ->
|
{true, outgoing};
|
||||||
Type = get_message_type(Packet),
|
false ->
|
||||||
gen_server:cast(?PROCNAME, {message, Direction, From, To, Type});
|
{true, incoming}
|
||||||
_ ->
|
|
||||||
case is_carbon(Packet) of
|
|
||||||
{true, OrigDirection} ->
|
|
||||||
gen_server:cast(?PROCNAME, {message, OrigDirection, From, To,
|
|
||||||
carbon});
|
|
||||||
false ->
|
|
||||||
ok
|
|
||||||
end
|
|
||||||
end;
|
end;
|
||||||
log_packet(_Direction, _From, _To, _Packet) ->
|
is_carbon(_Msg) ->
|
||||||
ok.
|
false.
|
||||||
|
|
||||||
-spec get_message_type(xmlel()) -> binary().
|
-spec should_log(message()) -> boolean().
|
||||||
|
should_log(#message{meta = #{carbon_copy := true}} = Msg) ->
|
||||||
get_message_type(#xmlel{attrs = Attrs}) ->
|
should_log(xmpp_util:unwrap_carbon(Msg));
|
||||||
case fxml:get_attr_s(<<"type">>, Attrs) of
|
should_log(#message{type = error}) ->
|
||||||
<<"">> ->
|
false;
|
||||||
<<"normal">>;
|
should_log(#message{body = Body, sub_els = SubEls}) ->
|
||||||
Type ->
|
xmpp:get_text(Body) /= <<>>
|
||||||
Type
|
orelse lists:any(fun(#xmlel{name = <<"encrypted">>}) -> true;
|
||||||
end.
|
(_) -> false
|
||||||
|
end, SubEls).
|
||||||
-spec is_carbon(xmlel()) -> {true, direction()} | false.
|
|
||||||
|
|
||||||
is_carbon(Packet) ->
|
|
||||||
{Direction, SubTag} = case {fxml:get_subtag(Packet, <<"sent">>),
|
|
||||||
fxml:get_subtag(Packet, <<"received">>)} of
|
|
||||||
{false, false} ->
|
|
||||||
{false, false};
|
|
||||||
{false, Tag} ->
|
|
||||||
{incoming, Tag};
|
|
||||||
{Tag, _} ->
|
|
||||||
{outgoing, Tag}
|
|
||||||
end,
|
|
||||||
F = fun(_, false) ->
|
|
||||||
false;
|
|
||||||
(Name, Tag) ->
|
|
||||||
fxml:get_subtag(Tag, Name)
|
|
||||||
end,
|
|
||||||
case lists:foldl(F, SubTag, [<<"forwarded">>, <<"message">>, <<"body">>]) of
|
|
||||||
#xmlel{children = Body} when length(Body) > 0 ->
|
|
||||||
{true, Direction};
|
|
||||||
_ ->
|
|
||||||
false
|
|
||||||
end.
|
|
||||||
|
|
||||||
-spec write_log(io:device(), direction(), jid(), jid(), binary()) -> ok.
|
|
||||||
|
|
||||||
|
-spec write_log(io:device(), direction(), jid(), jid(),
|
||||||
|
message_type() | offline | carbon) -> ok.
|
||||||
write_log(IoDevice, Direction, From, To, Type) ->
|
write_log(IoDevice, Direction, From, To, Type) ->
|
||||||
Date = format_date(calendar:local_time()),
|
Date = format_date(calendar:local_time()),
|
||||||
Record = io_lib:format("~s [~s, ~s] ~s -> ~s~n",
|
Record = io_lib:format("~s [~s, ~s] ~s -> ~s~n",
|
||||||
|
@ -238,7 +207,6 @@ write_log(IoDevice, Direction, From, To, Type) ->
|
||||||
ok = file:write(IoDevice, [Record]).
|
ok = file:write(IoDevice, [Record]).
|
||||||
|
|
||||||
-spec format_date(calendar:datetime()) -> io_lib:chars().
|
-spec format_date(calendar:datetime()) -> io_lib:chars().
|
||||||
|
|
||||||
format_date({{Year, Month, Day}, {Hour, Minute, Second}}) ->
|
format_date({{Year, Month, Day}, {Hour, Minute, Second}}) ->
|
||||||
Format = "~B-~2..0B-~2..0B ~2..0B:~2..0B:~2..0B",
|
Format = "~B-~2..0B-~2..0B ~2..0B:~2..0B:~2..0B",
|
||||||
io_lib:format(Format, [Year, Month, Day, Hour, Minute, Second]).
|
io_lib:format(Format, [Year, Month, Day, Hour, Minute, Second]).
|
||||||
|
|
Loading…
Reference in New Issue