2014-05-27 00:57:06 +02:00
|
|
|
%%%-------------------------------------------------------------------
|
|
|
|
%%% File : mod_message_log.erl
|
|
|
|
%%% Author : Holger Weiss <holger@zedat.fu-berlin.de>
|
|
|
|
%%% Purpose : Log one line per message transmission
|
|
|
|
%%% Created : 27 May 2014 by Holger Weiss <holger@zedat.fu-berlin.de>
|
|
|
|
%%%-------------------------------------------------------------------
|
|
|
|
|
|
|
|
-module(mod_message_log).
|
|
|
|
-author('holger@zedat.fu-berlin.de').
|
|
|
|
|
|
|
|
-behaviour(gen_mod).
|
|
|
|
|
|
|
|
-export([start/2,
|
|
|
|
stop/1,
|
|
|
|
init/1,
|
|
|
|
perform_upgrade/1]).
|
|
|
|
|
|
|
|
-export([log_packet_send/3,
|
|
|
|
log_packet_receive/4,
|
|
|
|
log_packet_offline/3,
|
|
|
|
reopen_log/0]).
|
|
|
|
|
|
|
|
-include("ejabberd.hrl").
|
|
|
|
-include("jlib.hrl").
|
|
|
|
|
|
|
|
-define(PROCNAME, ?MODULE).
|
|
|
|
-define(DEFAULT_FILENAME, <<"message.log">>).
|
|
|
|
-define(FILE_MODES, [append, raw]).
|
|
|
|
|
|
|
|
-record(config, {filename = ?DEFAULT_FILENAME, iodevice}).
|
|
|
|
|
|
|
|
start(Host, Opts) ->
|
|
|
|
ejabberd_hooks:add(user_send_packet, Host, ?MODULE,
|
|
|
|
log_packet_send, 42),
|
|
|
|
ejabberd_hooks:add(user_receive_packet, Host, ?MODULE,
|
|
|
|
log_packet_receive, 42),
|
|
|
|
ejabberd_hooks:add(offline_message_hook, Host, ?MODULE,
|
|
|
|
log_packet_offline, 42),
|
|
|
|
case whereis(?PROCNAME) of
|
|
|
|
undefined ->
|
|
|
|
ejabberd_hooks:add(reopen_log_hook, ?MODULE, reopen_log, 42),
|
|
|
|
Filename = gen_mod:get_opt(filename, Opts, fun(V) -> V end,
|
|
|
|
?DEFAULT_FILENAME),
|
|
|
|
register(?PROCNAME, spawn(?MODULE, init,
|
|
|
|
[#config{filename = Filename}])),
|
|
|
|
ok;
|
|
|
|
_ ->
|
|
|
|
ok
|
|
|
|
end.
|
|
|
|
|
|
|
|
stop(Host) ->
|
|
|
|
ejabberd_hooks:delete(user_send_packet, Host, ?MODULE,
|
|
|
|
log_packet_send, 42),
|
|
|
|
ejabberd_hooks:delete(user_receive_packet, Host, ?MODULE,
|
|
|
|
log_packet_receive, 42),
|
|
|
|
ejabberd_hooks:delete(offline_message_hook, Host, ?MODULE,
|
|
|
|
log_packet_offline, 42),
|
|
|
|
case whereis(?PROCNAME) of
|
|
|
|
undefined ->
|
|
|
|
ok;
|
|
|
|
_ ->
|
|
|
|
ejabberd_hooks:delete(reopen_log_hook, ?MODULE, reopen_log, 42),
|
|
|
|
gen_mod:get_module_proc(Host, ?PROCNAME) ! stop,
|
|
|
|
ok
|
|
|
|
end.
|
|
|
|
|
|
|
|
init(Config) ->
|
|
|
|
{ok, IoDevice} = file:open(Config#config.filename, ?FILE_MODES),
|
|
|
|
loop(Config#config{iodevice = IoDevice}).
|
|
|
|
|
|
|
|
log_packet_send(From, To, Packet) ->
|
|
|
|
log_packet(outgoing, From, To, Packet).
|
|
|
|
|
|
|
|
log_packet_receive(JID, From, _To, Packet) ->
|
|
|
|
log_packet(incoming, From, JID, Packet).
|
|
|
|
|
|
|
|
log_packet_offline(From, To, Packet) ->
|
|
|
|
log_packet(offline, From, To, Packet).
|
|
|
|
|
|
|
|
reopen_log() ->
|
|
|
|
?PROCNAME ! reopen.
|
|
|
|
|
|
|
|
%% Internal functions.
|
|
|
|
|
|
|
|
log_packet(Direction, From, To, #xmlel{name = <<"message">>} = Packet) ->
|
|
|
|
case xml:get_subtag(Packet, <<"body">>) of
|
|
|
|
#xmlel{children = Body} when length(Body) > 0 ->
|
|
|
|
Type = get_message_type(Packet),
|
|
|
|
?PROCNAME ! {message, Direction, From, To, Type};
|
|
|
|
_ ->
|
|
|
|
case is_carbon(Packet) of
|
|
|
|
{true, OrigDirection} ->
|
|
|
|
?PROCNAME ! {message, OrigDirection, From, To, carbon};
|
|
|
|
false ->
|
|
|
|
ok
|
|
|
|
end
|
|
|
|
end;
|
|
|
|
log_packet(_Direction, _From, _To, _Packet) ->
|
|
|
|
ok.
|
|
|
|
|
|
|
|
get_message_type(#xmlel{attrs = Attrs}) ->
|
|
|
|
case xml:get_attr_s(<<"type">>, Attrs) of
|
|
|
|
<<"">> ->
|
|
|
|
<<"normal">>;
|
|
|
|
Type ->
|
|
|
|
Type
|
|
|
|
end.
|
|
|
|
|
|
|
|
is_carbon(Packet) ->
|
|
|
|
{Direction, SubTag} = case {xml:get_subtag(Packet, <<"sent">>),
|
|
|
|
xml:get_subtag(Packet, <<"received">>)} of
|
|
|
|
{false, false} ->
|
2014-11-06 00:56:55 +01:00
|
|
|
{false, false};
|
2014-05-27 00:57:06 +02:00
|
|
|
{false, Tag} ->
|
|
|
|
{incoming, Tag};
|
|
|
|
{Tag, _} ->
|
|
|
|
{outgoing, Tag}
|
|
|
|
end,
|
2014-11-06 00:56:55 +01:00
|
|
|
F = fun(_, false) ->
|
|
|
|
false;
|
|
|
|
(Name, Tag) ->
|
|
|
|
xml:get_subtag(Tag, Name)
|
|
|
|
end,
|
|
|
|
case lists:foldl(F, SubTag, [<<"forwarded">>, <<"message">>, <<"body">>]) of
|
|
|
|
#xmlel{children = Body} when length(Body) > 0 ->
|
|
|
|
{true, Direction};
|
|
|
|
_ ->
|
|
|
|
false
|
2014-05-27 00:57:06 +02:00
|
|
|
end.
|
|
|
|
|
|
|
|
loop(Config) ->
|
|
|
|
receive
|
|
|
|
{message, Direction, From, To, Type} ->
|
|
|
|
write_log(Config#config.iodevice, Direction, From, To, Type),
|
|
|
|
loop(Config);
|
|
|
|
reopen ->
|
|
|
|
file:close(Config#config.iodevice),
|
|
|
|
{ok, IoDevice} = file:open(Config#config.filename, ?FILE_MODES),
|
|
|
|
loop(Config#config{iodevice = IoDevice});
|
|
|
|
upgrade ->
|
|
|
|
mod_message_log:perform_upgrade(Config);
|
|
|
|
stop ->
|
|
|
|
exit(normal)
|
|
|
|
end.
|
|
|
|
|
|
|
|
write_log(IoDevice, Direction, From, To, Type) ->
|
|
|
|
Date = format_date(calendar:local_time()),
|
|
|
|
Record = io_lib:format("~s [~s, ~s] ~s -> ~s~n",
|
|
|
|
[Date, Direction, Type,
|
|
|
|
jlib:jid_to_string(From),
|
|
|
|
jlib:jid_to_string(To)]),
|
|
|
|
ok = file:write(IoDevice, [Record]).
|
|
|
|
|
|
|
|
format_date({{Year, Month, Day}, {Hour, Minute, Second}}) ->
|
|
|
|
Format = "~B-~2..0B-~2..0B ~2..0B:~2..0B:~2..0B",
|
|
|
|
io_lib:format(Format, [Year, Month, Day, Hour, Minute, Second]).
|
|
|
|
|
|
|
|
perform_upgrade(Config) ->
|
|
|
|
loop(Config).
|