2018-04-04 01:36:05 +02:00
|
|
|
%%%----------------------------------------------------------------------
|
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>
|
2018-04-04 01:36:05 +02:00
|
|
|
%%%
|
|
|
|
%%%
|
2020-01-29 11:33:55 +01:00
|
|
|
%%% ejabberd, Copyright (C) 2018-2020 ProcessOne
|
2018-04-04 01:36:05 +02:00
|
|
|
%%%
|
|
|
|
%%% This program is free software; you can redistribute it and/or
|
|
|
|
%%% modify it under the terms of the GNU General Public License as
|
|
|
|
%%% published by the Free Software Foundation; either version 2 of the
|
|
|
|
%%% License, or (at your option) any later version.
|
|
|
|
%%%
|
|
|
|
%%% This program is distributed in the hope that it will be useful,
|
|
|
|
%%% but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
%%% MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
%%% General Public License for more details.
|
|
|
|
%%%
|
|
|
|
%%% You should have received a copy of the GNU General Public License along
|
|
|
|
%%% with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
%%% 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
%%%
|
|
|
|
%%%----------------------------------------------------------------------
|
2014-05-27 00:57:06 +02:00
|
|
|
|
|
|
|
-module(mod_message_log).
|
|
|
|
-author('holger@zedat.fu-berlin.de').
|
|
|
|
|
|
|
|
-behaviour(gen_mod).
|
2015-01-04 19:22:51 +01:00
|
|
|
-behaviour(gen_server).
|
|
|
|
|
2018-04-04 01:34:53 +02:00
|
|
|
%% gen_mod callbacks.
|
|
|
|
-export([start/2,
|
2016-05-01 21:35:23 +02:00
|
|
|
stop/1,
|
2018-04-04 01:34:53 +02:00
|
|
|
mod_opt_type/1,
|
|
|
|
mod_options/1,
|
2021-07-06 20:57:36 +02:00
|
|
|
depends/2,
|
2022-07-21 15:48:53 +02:00
|
|
|
mod_status/0,
|
2021-07-06 20:57:36 +02:00
|
|
|
mod_doc/0]).
|
2015-01-04 19:22:51 +01:00
|
|
|
|
|
|
|
%% gen_server callbacks.
|
|
|
|
-export([init/1,
|
|
|
|
handle_call/3,
|
|
|
|
handle_cast/2,
|
|
|
|
handle_info/2,
|
|
|
|
terminate/2,
|
|
|
|
code_change/3]).
|
|
|
|
|
|
|
|
%% ejabberd_hooks callbacks.
|
2018-04-04 01:34:53 +02:00
|
|
|
-export([log_packet_send/1,
|
|
|
|
log_packet_receive/1,
|
|
|
|
log_packet_offline/1,
|
2014-05-27 00:57:06 +02:00
|
|
|
reopen_log/0]).
|
|
|
|
|
2021-02-08 00:15:27 +01:00
|
|
|
-include_lib("xmpp/include/xmpp.hrl").
|
2014-05-27 00:57:06 +02:00
|
|
|
|
|
|
|
-define(FILE_MODES, [append, raw]).
|
|
|
|
|
2022-07-21 11:14:17 +02:00
|
|
|
-record(state, {filename :: binary(),
|
|
|
|
iodevice :: io:device()}).
|
2015-01-04 19:22:51 +01:00
|
|
|
|
2015-01-04 19:29:54 +01:00
|
|
|
-type direction() :: incoming | outgoing | offline.
|
|
|
|
-type state() :: #state{}.
|
2018-04-04 01:34:53 +02:00
|
|
|
-type c2s_state() :: ejabberd_c2s:state().
|
|
|
|
-type c2s_hook_acc() :: {stanza() | drop, c2s_state()}.
|
2015-01-04 19:29:54 +01:00
|
|
|
|
2015-01-04 19:22:51 +01:00
|
|
|
%% -------------------------------------------------------------------
|
2018-04-04 01:34:53 +02:00
|
|
|
%% gen_mod callbacks.
|
2015-01-04 19:22:51 +01:00
|
|
|
%% -------------------------------------------------------------------
|
2022-08-12 17:57:07 +02:00
|
|
|
-spec start(binary(), gen_mod:opts()) -> ok | {ok, pid()} | {error, term()}.
|
2014-05-27 00:57:06 +02:00
|
|
|
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),
|
2022-08-12 17:57:07 +02:00
|
|
|
case gen_mod:start_child(?MODULE, <<"global">>, Opts) of
|
2018-04-04 01:34:53 +02:00
|
|
|
{ok, Ref} ->
|
|
|
|
{ok, Ref};
|
|
|
|
{error, {already_started, Ref}} ->
|
|
|
|
{ok, Ref};
|
|
|
|
{error, Reason} ->
|
|
|
|
{error, Reason}
|
|
|
|
end.
|
2014-05-27 00:57:06 +02:00
|
|
|
|
2015-01-04 19:29:54 +01:00
|
|
|
-spec stop(binary()) -> ok.
|
2014-05-27 00:57:06 +02:00
|
|
|
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),
|
2022-08-12 17:57:07 +02:00
|
|
|
gen_mod:stop_child(gen_mod:get_module_proc(global, ?MODULE)),
|
2018-04-04 01:34:53 +02:00
|
|
|
ok.
|
2016-05-01 21:35:23 +02:00
|
|
|
|
2022-08-12 17:57:07 +02:00
|
|
|
-spec mod_opt_type(atom()) -> econf:validator().
|
2016-05-01 21:35:23 +02:00
|
|
|
mod_opt_type(filename) ->
|
2022-07-21 11:14:17 +02:00
|
|
|
econf:either(auto, econf:file(write)).
|
2018-04-04 01:34:53 +02:00
|
|
|
|
|
|
|
-spec mod_options(binary()) -> [{atom(), any()}].
|
|
|
|
mod_options(_Host) ->
|
2022-07-21 11:14:17 +02:00
|
|
|
[{filename, auto}].
|
2018-04-04 01:34:53 +02:00
|
|
|
|
|
|
|
-spec depends(binary(), gen_mod:opts()) -> [{module(), hard | soft}].
|
|
|
|
depends(_Host, _Opts) ->
|
|
|
|
[].
|
2016-05-01 21:35:23 +02:00
|
|
|
|
2021-07-06 20:57:36 +02:00
|
|
|
mod_doc() -> #{}.
|
|
|
|
|
2022-07-21 15:48:53 +02:00
|
|
|
mod_status() ->
|
|
|
|
Proc = gen_mod:get_module_proc(global, ?MODULE),
|
|
|
|
{filename, Filename} = gen_server:call(Proc, get_filename, timer:seconds(15)),
|
|
|
|
io_lib:format("Logging to: ~s", [Filename]).
|
|
|
|
|
2015-01-04 19:22:51 +01:00
|
|
|
%% -------------------------------------------------------------------
|
|
|
|
%% gen_server callbacks.
|
|
|
|
%% -------------------------------------------------------------------
|
2018-04-06 00:00:11 +02:00
|
|
|
-spec init(list()) -> {ok, state()}.
|
|
|
|
init([_Host, Opts]) ->
|
2015-01-04 19:22:51 +01:00
|
|
|
process_flag(trap_exit, true),
|
|
|
|
ejabberd_hooks:add(reopen_log_hook, ?MODULE, reopen_log, 42),
|
2022-07-21 11:14:17 +02:00
|
|
|
Filename = case gen_mod:get_opt(filename, Opts) of
|
|
|
|
auto ->
|
|
|
|
filename:join(filename:dirname(ejabberd_logger:get_log_path()),
|
|
|
|
"message.log");
|
|
|
|
FN -> FN
|
|
|
|
end,
|
2015-01-04 19:22:51 +01:00
|
|
|
{ok, IoDevice} = file:open(Filename, ?FILE_MODES),
|
|
|
|
{ok, #state{filename = Filename, iodevice = IoDevice}}.
|
|
|
|
|
2015-01-04 19:29:54 +01:00
|
|
|
-spec handle_call(_, {pid(), _}, state()) -> {noreply, state()}.
|
2022-07-27 00:55:43 +02:00
|
|
|
handle_call(get_filename, _From, State) ->
|
2022-07-21 15:48:53 +02:00
|
|
|
{reply, {filename, State#state.filename}, State};
|
2015-01-04 19:22:51 +01:00
|
|
|
handle_call(_Request, _From, State) ->
|
|
|
|
{noreply, State}.
|
|
|
|
|
2015-01-04 19:29:54 +01:00
|
|
|
-spec handle_cast(_, state()) -> {noreply, state()}.
|
2015-01-04 19:22:51 +01:00
|
|
|
handle_cast({message, Direction, From, To, Type}, #state{iodevice = IoDevice} =
|
|
|
|
State) ->
|
|
|
|
write_log(IoDevice, Direction, From, To, Type),
|
|
|
|
{noreply, State};
|
|
|
|
handle_cast(reopen_log, #state{filename = Filename, iodevice = IoDevice} =
|
|
|
|
State) ->
|
|
|
|
ok = file:close(IoDevice),
|
|
|
|
{ok, NewIoDevice} = file:open(Filename, ?FILE_MODES),
|
|
|
|
{noreply, State#state{iodevice = NewIoDevice}};
|
|
|
|
handle_cast(_Request, State) ->
|
|
|
|
{noreply, State}.
|
|
|
|
|
2015-01-04 19:29:54 +01:00
|
|
|
-spec handle_info(timeout | _, state()) -> {noreply, state()}.
|
2015-01-04 19:22:51 +01:00
|
|
|
handle_info(_Info, State) ->
|
|
|
|
{noreply, State}.
|
|
|
|
|
2015-01-04 19:29:54 +01:00
|
|
|
-spec terminate(normal | shutdown | {shutdown, _} | _, _) -> any().
|
2015-01-04 19:22:51 +01:00
|
|
|
terminate(_Reason, State) ->
|
|
|
|
ejabberd_hooks:delete(reopen_log_hook, ?MODULE, reopen_log, 42),
|
|
|
|
ok = file:close(State#state.iodevice).
|
|
|
|
|
2015-01-04 19:29:54 +01:00
|
|
|
-spec code_change({down, _} | _, state(), _) -> {ok, state()}.
|
2015-01-04 19:22:51 +01:00
|
|
|
code_change(_OldVsn, State, _Extra) ->
|
|
|
|
{ok, State}.
|
|
|
|
|
|
|
|
%% -------------------------------------------------------------------
|
|
|
|
%% ejabberd_hooks callbacks.
|
|
|
|
%% -------------------------------------------------------------------
|
2018-04-04 01:34:53 +02:00
|
|
|
-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_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.
|
|
|
|
|
2018-04-04 01:44:13 +02:00
|
|
|
-spec log_packet_offline({any(), message()}) -> {any(), message()}.
|
|
|
|
log_packet_offline({_Action, Msg} = Acc) ->
|
|
|
|
log_packet(offline, Msg),
|
|
|
|
Acc.
|
2014-05-27 00:57:06 +02:00
|
|
|
|
2015-01-04 19:29:54 +01:00
|
|
|
-spec reopen_log() -> any().
|
2014-05-27 00:57:06 +02:00
|
|
|
reopen_log() ->
|
2018-04-04 01:55:35 +02:00
|
|
|
Proc = gen_mod:get_module_proc(global, ?MODULE),
|
|
|
|
gen_server:cast(Proc, reopen_log).
|
2014-05-27 00:57:06 +02:00
|
|
|
|
2015-01-04 19:22:51 +01:00
|
|
|
%% -------------------------------------------------------------------
|
2014-05-27 00:57:06 +02:00
|
|
|
%% Internal functions.
|
2015-01-04 19:22:51 +01:00
|
|
|
%% -------------------------------------------------------------------
|
2018-04-04 01:34:53 +02:00
|
|
|
-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,
|
2018-04-04 01:55:35 +02:00
|
|
|
Proc = gen_mod:get_module_proc(global, ?MODULE),
|
|
|
|
gen_server:cast(Proc, {message, Direction1, From, To, Type1});
|
2018-04-04 01:34:53 +02:00
|
|
|
false ->
|
|
|
|
ok
|
2014-05-27 00:57:06 +02:00
|
|
|
end.
|
|
|
|
|
2018-04-04 01:34:53 +02:00
|
|
|
-spec is_carbon(message()) -> {true, direction()} | false.
|
|
|
|
is_carbon(#message{meta = #{carbon_copy := true}} = Msg) ->
|
2022-08-12 17:57:07 +02:00
|
|
|
case xmpp:has_subtag(Msg, #carbons_sent{forwarded = #forwarded{}}) of
|
2018-04-04 01:34:53 +02:00
|
|
|
true ->
|
|
|
|
{true, outgoing};
|
|
|
|
false ->
|
|
|
|
{true, incoming}
|
|
|
|
end;
|
|
|
|
is_carbon(_Msg) ->
|
|
|
|
false.
|
|
|
|
|
|
|
|
-spec should_log(message()) -> boolean().
|
|
|
|
should_log(#message{meta = #{carbon_copy := true}} = Msg) ->
|
2018-12-10 12:17:19 +01:00
|
|
|
should_log(misc:unwrap_carbon(Msg));
|
2018-04-04 01:34:53 +02:00
|
|
|
should_log(#message{type = error}) ->
|
|
|
|
false;
|
|
|
|
should_log(#message{body = Body, sub_els = SubEls}) ->
|
|
|
|
xmpp:get_text(Body) /= <<>>
|
|
|
|
orelse lists:any(fun(#xmlel{name = <<"encrypted">>}) -> true;
|
|
|
|
(_) -> false
|
|
|
|
end, SubEls).
|
|
|
|
|
|
|
|
-spec write_log(io:device(), direction(), jid(), jid(),
|
|
|
|
message_type() | offline | carbon) -> ok.
|
2014-05-27 00:57:06 +02:00
|
|
|
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,
|
2018-09-10 12:07:03 +02:00
|
|
|
jid:encode(From), jid:encode(To)]),
|
2014-05-27 00:57:06 +02:00
|
|
|
ok = file:write(IoDevice, [Record]).
|
|
|
|
|
2015-01-04 19:29:54 +01:00
|
|
|
-spec format_date(calendar:datetime()) -> io_lib:chars().
|
2014-05-27 00:57:06 +02:00
|
|
|
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]).
|