2003-01-11 21:25:11 +01:00
|
|
|
%%%----------------------------------------------------------------------
|
|
|
|
%%% File : mod_stats.erl
|
2007-12-24 13:58:05 +01:00
|
|
|
%%% Author : Alexey Shchepin <alexey@process-one.net>
|
|
|
|
%%% Purpose : Basic statistics.
|
|
|
|
%%% Created : 11 Jan 2003 by Alexey Shchepin <alexey@process-one.net>
|
|
|
|
%%%
|
|
|
|
%%%
|
2024-01-22 16:40:01 +01:00
|
|
|
%%% ejabberd, Copyright (C) 2002-2024 ProcessOne
|
2007-12-24 13:58:05 +01: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.
|
2009-01-12 15:44:42 +01:00
|
|
|
%%%
|
2014-02-22 11:27:40 +01:00
|
|
|
%%% 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.
|
2007-12-24 13:58:05 +01:00
|
|
|
%%%
|
2003-01-11 21:25:11 +01:00
|
|
|
%%%----------------------------------------------------------------------
|
|
|
|
|
|
|
|
-module(mod_stats).
|
2013-03-14 10:33:02 +01:00
|
|
|
|
2007-12-24 13:58:05 +01:00
|
|
|
-author('alexey@process-one.net').
|
2003-01-11 21:25:11 +01:00
|
|
|
|
2015-05-21 17:02:36 +02:00
|
|
|
-protocol({xep, 39, '0.6.0'}).
|
|
|
|
|
2003-01-24 21:18:33 +01:00
|
|
|
-behaviour(gen_mod).
|
|
|
|
|
2018-02-11 10:54:15 +01:00
|
|
|
-export([start/2, stop/1, reload/3, process_iq/1,
|
2020-01-08 10:24:51 +01:00
|
|
|
mod_options/1, depends/2, mod_doc/0]).
|
2003-01-11 21:25:11 +01:00
|
|
|
|
2013-04-08 11:12:54 +02:00
|
|
|
-include("logger.hrl").
|
2020-09-03 13:45:57 +02:00
|
|
|
-include_lib("xmpp/include/xmpp.hrl").
|
2019-06-22 16:08:45 +02:00
|
|
|
-include("translate.hrl").
|
2003-01-11 21:25:11 +01:00
|
|
|
|
2023-08-04 17:53:50 +02:00
|
|
|
start(_Host, _Opts) ->
|
|
|
|
{ok, [{iq_handler, ejabberd_local, ?NS_STATS, process_iq}]}.
|
2003-01-11 21:25:11 +01:00
|
|
|
|
2023-08-04 17:53:50 +02:00
|
|
|
stop(_Host) ->
|
|
|
|
ok.
|
2003-01-11 21:25:11 +01:00
|
|
|
|
2023-08-04 19:54:02 +02:00
|
|
|
reload(_Host, _NewOpts, _OldOpts) ->
|
2023-08-04 17:53:50 +02:00
|
|
|
ok.
|
2017-02-22 17:46:47 +01:00
|
|
|
|
2016-07-06 13:58:48 +02:00
|
|
|
depends(_Host, _Opts) ->
|
|
|
|
[].
|
|
|
|
|
2016-07-31 13:51:16 +02:00
|
|
|
process_iq(#iq{type = set, lang = Lang} = IQ) ->
|
2019-06-22 16:08:45 +02:00
|
|
|
Txt = ?T("Value 'set' of 'type' attribute is not allowed"),
|
2016-07-31 13:51:16 +02:00
|
|
|
xmpp:make_error(IQ, xmpp:err_not_allowed(Txt, Lang));
|
|
|
|
process_iq(#iq{type = get, to = To, lang = Lang,
|
|
|
|
sub_els = [#stats{} = Stats]} = IQ) ->
|
|
|
|
Node = str:tokens(Stats#stats.node, <<"/">>),
|
|
|
|
Names = [Name || #stat{name = Name} <- Stats#stats.list],
|
|
|
|
case get_local_stats(To#jid.server, Node, Names, Lang) of
|
|
|
|
{result, List} ->
|
|
|
|
xmpp:make_iq_result(IQ, Stats#stats{list = List});
|
|
|
|
{error, Error} ->
|
|
|
|
xmpp:make_error(IQ, Error)
|
2003-01-11 21:25:11 +01:00
|
|
|
end.
|
|
|
|
|
2016-07-31 13:51:16 +02:00
|
|
|
-define(STAT(Name), #stat{name = Name}).
|
2003-01-18 20:42:48 +01:00
|
|
|
|
2016-03-31 10:00:29 +02:00
|
|
|
get_local_stats(_Server, [], [], _Lang) ->
|
2003-01-11 21:25:11 +01:00
|
|
|
{result,
|
2013-03-14 10:33:02 +01:00
|
|
|
[?STAT(<<"users/online">>), ?STAT(<<"users/total">>),
|
|
|
|
?STAT(<<"users/all-hosts/online">>),
|
|
|
|
?STAT(<<"users/all-hosts/total">>)]};
|
2016-03-31 10:00:29 +02:00
|
|
|
get_local_stats(Server, [], Names, _Lang) ->
|
2003-01-11 21:25:11 +01:00
|
|
|
{result,
|
2013-03-14 10:33:02 +01:00
|
|
|
lists:map(fun (Name) -> get_local_stat(Server, [], Name)
|
|
|
|
end,
|
|
|
|
Names)};
|
|
|
|
get_local_stats(_Server, [<<"running nodes">>, _],
|
2016-03-31 10:00:29 +02:00
|
|
|
[], _Lang) ->
|
2013-03-14 10:33:02 +01:00
|
|
|
{result,
|
|
|
|
[?STAT(<<"time/uptime">>), ?STAT(<<"time/cputime">>),
|
|
|
|
?STAT(<<"users/online">>),
|
|
|
|
?STAT(<<"transactions/committed">>),
|
|
|
|
?STAT(<<"transactions/aborted">>),
|
|
|
|
?STAT(<<"transactions/restarted">>),
|
|
|
|
?STAT(<<"transactions/logged">>)]};
|
|
|
|
get_local_stats(_Server, [<<"running nodes">>, ENode],
|
2016-03-31 10:00:29 +02:00
|
|
|
Names, Lang) ->
|
2003-01-18 20:42:48 +01:00
|
|
|
case search_running_node(ENode) of
|
2016-03-31 10:00:29 +02:00
|
|
|
false ->
|
2019-06-22 16:08:45 +02:00
|
|
|
Txt = ?T("No running node found"),
|
2016-07-31 13:51:16 +02:00
|
|
|
{error, xmpp:err_item_not_found(Txt, Lang)};
|
2013-03-14 10:33:02 +01:00
|
|
|
Node ->
|
|
|
|
{result,
|
|
|
|
lists:map(fun (Name) -> get_node_stat(Node, Name) end,
|
|
|
|
Names)}
|
2003-01-18 20:42:48 +01:00
|
|
|
end;
|
2016-03-31 10:00:29 +02:00
|
|
|
get_local_stats(_Server, _, _, Lang) ->
|
2019-06-22 16:08:45 +02:00
|
|
|
Txt = ?T("No statistics found for this item"),
|
2016-07-31 13:51:16 +02:00
|
|
|
{error, xmpp:err_feature_not_implemented(Txt, Lang)}.
|
2003-01-18 20:42:48 +01:00
|
|
|
|
2016-07-31 13:51:16 +02:00
|
|
|
-define(STATVAL(Val, Unit), #stat{name = Name, units = Unit, value = Val}).
|
2003-01-11 21:25:11 +01:00
|
|
|
|
|
|
|
-define(STATERR(Code, Desc),
|
2016-07-31 13:51:16 +02:00
|
|
|
#stat{name = Name,
|
|
|
|
error = #stat_error{code = Code, reason = Desc}}).
|
2013-03-14 10:33:02 +01:00
|
|
|
|
|
|
|
get_local_stat(Server, [], Name)
|
|
|
|
when Name == <<"users/online">> ->
|
2005-04-17 20:08:34 +02:00
|
|
|
case catch ejabberd_sm:get_vh_session_list(Server) of
|
2013-03-14 10:33:02 +01:00
|
|
|
{'EXIT', _Reason} ->
|
2016-07-31 13:51:16 +02:00
|
|
|
?STATERR(500, <<"Internal Server Error">>);
|
2013-03-14 10:33:02 +01:00
|
|
|
Users ->
|
2016-11-24 13:06:06 +01:00
|
|
|
?STATVAL((integer_to_binary(length(Users))),
|
2013-03-14 10:33:02 +01:00
|
|
|
<<"users">>)
|
2005-04-17 20:08:34 +02:00
|
|
|
end;
|
2013-03-14 10:33:02 +01:00
|
|
|
get_local_stat(Server, [], Name)
|
|
|
|
when Name == <<"users/total">> ->
|
|
|
|
case catch
|
2017-05-11 13:37:21 +02:00
|
|
|
ejabberd_auth:count_users(Server)
|
2013-03-14 10:33:02 +01:00
|
|
|
of
|
|
|
|
{'EXIT', _Reason} ->
|
2016-07-31 13:51:16 +02:00
|
|
|
?STATERR(500, <<"Internal Server Error">>);
|
2013-03-14 10:33:02 +01:00
|
|
|
NUsers ->
|
2016-11-24 13:06:06 +01:00
|
|
|
?STATVAL((integer_to_binary(NUsers)),
|
2013-03-14 10:33:02 +01:00
|
|
|
<<"users">>)
|
2005-04-17 20:08:34 +02:00
|
|
|
end;
|
2013-03-14 10:33:02 +01:00
|
|
|
get_local_stat(_Server, [], Name)
|
|
|
|
when Name == <<"users/all-hosts/online">> ->
|
2016-10-11 22:20:22 +02:00
|
|
|
Users = ejabberd_sm:connected_users_number(),
|
2016-11-24 13:06:06 +01:00
|
|
|
?STATVAL((integer_to_binary(Users)), <<"users">>);
|
2013-03-14 10:33:02 +01:00
|
|
|
get_local_stat(_Server, [], Name)
|
|
|
|
when Name == <<"users/all-hosts/total">> ->
|
|
|
|
NumUsers = lists:foldl(fun (Host, Total) ->
|
2017-05-11 13:37:21 +02:00
|
|
|
ejabberd_auth:count_users(Host)
|
2013-03-14 10:33:02 +01:00
|
|
|
+ Total
|
|
|
|
end,
|
2019-06-14 11:33:26 +02:00
|
|
|
0, ejabberd_option:hosts()),
|
2016-11-24 13:06:06 +01:00
|
|
|
?STATVAL((integer_to_binary(NumUsers)),
|
2013-03-14 10:33:02 +01:00
|
|
|
<<"users">>);
|
2005-04-17 20:08:34 +02:00
|
|
|
get_local_stat(_Server, _, Name) ->
|
2016-07-31 13:51:16 +02:00
|
|
|
?STATERR(404, <<"Not Found">>).
|
2013-03-14 10:33:02 +01:00
|
|
|
|
|
|
|
get_node_stat(Node, Name)
|
|
|
|
when Name == <<"time/uptime">> ->
|
2015-10-07 00:06:58 +02:00
|
|
|
case catch ejabberd_cluster:call(Node, erlang, statistics,
|
2013-03-14 10:33:02 +01:00
|
|
|
[wall_clock])
|
|
|
|
of
|
|
|
|
{badrpc, _Reason} ->
|
2016-07-31 13:51:16 +02:00
|
|
|
?STATERR(500, <<"Internal Server Error">>);
|
2013-03-14 10:33:02 +01:00
|
|
|
CPUTime ->
|
2016-11-24 13:06:06 +01:00
|
|
|
?STATVAL(str:format("~.3f", [element(1, CPUTime) / 1000]),
|
2013-03-14 10:33:02 +01:00
|
|
|
<<"seconds">>)
|
2003-01-18 20:42:48 +01:00
|
|
|
end;
|
2013-03-14 10:33:02 +01:00
|
|
|
get_node_stat(Node, Name)
|
|
|
|
when Name == <<"time/cputime">> ->
|
2015-10-07 00:06:58 +02:00
|
|
|
case catch ejabberd_cluster:call(Node, erlang, statistics, [runtime])
|
2013-03-14 10:33:02 +01:00
|
|
|
of
|
|
|
|
{badrpc, _Reason} ->
|
2016-07-31 13:51:16 +02:00
|
|
|
?STATERR(500, <<"Internal Server Error">>);
|
2013-03-14 10:33:02 +01:00
|
|
|
RunTime ->
|
2016-11-24 13:06:06 +01:00
|
|
|
?STATVAL(str:format("~.3f", [element(1, RunTime) / 1000]),
|
2013-03-14 10:33:02 +01:00
|
|
|
<<"seconds">>)
|
2003-01-18 20:42:48 +01:00
|
|
|
end;
|
2013-03-14 10:33:02 +01:00
|
|
|
get_node_stat(Node, Name)
|
|
|
|
when Name == <<"users/online">> ->
|
2015-10-07 00:06:58 +02:00
|
|
|
case catch ejabberd_cluster:call(Node, ejabberd_sm,
|
2013-03-14 10:33:02 +01:00
|
|
|
dirty_get_my_sessions_list, [])
|
|
|
|
of
|
|
|
|
{badrpc, _Reason} ->
|
2016-07-31 13:51:16 +02:00
|
|
|
?STATERR(500, <<"Internal Server Error">>);
|
2013-03-14 10:33:02 +01:00
|
|
|
Users ->
|
2016-11-24 13:06:06 +01:00
|
|
|
?STATVAL((integer_to_binary(length(Users))),
|
2013-03-14 10:33:02 +01:00
|
|
|
<<"users">>)
|
2003-01-18 20:42:48 +01:00
|
|
|
end;
|
2013-03-14 10:33:02 +01:00
|
|
|
get_node_stat(Node, Name)
|
|
|
|
when Name == <<"transactions/committed">> ->
|
2015-10-07 00:06:58 +02:00
|
|
|
case catch ejabberd_cluster:call(Node, mnesia, system_info,
|
2013-03-14 10:33:02 +01:00
|
|
|
[transaction_commits])
|
|
|
|
of
|
|
|
|
{badrpc, _Reason} ->
|
2016-07-31 13:51:16 +02:00
|
|
|
?STATERR(500, <<"Internal Server Error">>);
|
2013-03-14 10:33:02 +01:00
|
|
|
Transactions ->
|
2016-11-24 13:06:06 +01:00
|
|
|
?STATVAL((integer_to_binary(Transactions)),
|
2013-03-14 10:33:02 +01:00
|
|
|
<<"transactions">>)
|
2003-01-29 18:12:23 +01:00
|
|
|
end;
|
2013-03-14 10:33:02 +01:00
|
|
|
get_node_stat(Node, Name)
|
|
|
|
when Name == <<"transactions/aborted">> ->
|
2015-10-07 00:06:58 +02:00
|
|
|
case catch ejabberd_cluster:call(Node, mnesia, system_info,
|
2013-03-14 10:33:02 +01:00
|
|
|
[transaction_failures])
|
|
|
|
of
|
|
|
|
{badrpc, _Reason} ->
|
2016-07-31 13:51:16 +02:00
|
|
|
?STATERR(500, <<"Internal Server Error">>);
|
2013-03-14 10:33:02 +01:00
|
|
|
Transactions ->
|
2016-11-24 13:06:06 +01:00
|
|
|
?STATVAL((integer_to_binary(Transactions)),
|
2013-03-14 10:33:02 +01:00
|
|
|
<<"transactions">>)
|
2003-01-29 18:12:23 +01:00
|
|
|
end;
|
2013-03-14 10:33:02 +01:00
|
|
|
get_node_stat(Node, Name)
|
|
|
|
when Name == <<"transactions/restarted">> ->
|
2015-10-07 00:06:58 +02:00
|
|
|
case catch ejabberd_cluster:call(Node, mnesia, system_info,
|
2013-03-14 10:33:02 +01:00
|
|
|
[transaction_restarts])
|
|
|
|
of
|
|
|
|
{badrpc, _Reason} ->
|
2016-07-31 13:51:16 +02:00
|
|
|
?STATERR(500, <<"Internal Server Error">>);
|
2013-03-14 10:33:02 +01:00
|
|
|
Transactions ->
|
2016-11-24 13:06:06 +01:00
|
|
|
?STATVAL((integer_to_binary(Transactions)),
|
2013-03-14 10:33:02 +01:00
|
|
|
<<"transactions">>)
|
2003-01-29 18:12:23 +01:00
|
|
|
end;
|
2013-03-14 10:33:02 +01:00
|
|
|
get_node_stat(Node, Name)
|
|
|
|
when Name == <<"transactions/logged">> ->
|
2015-10-07 00:06:58 +02:00
|
|
|
case catch ejabberd_cluster:call(Node, mnesia, system_info,
|
2013-03-14 10:33:02 +01:00
|
|
|
[transaction_log_writes])
|
|
|
|
of
|
|
|
|
{badrpc, _Reason} ->
|
2016-07-31 13:51:16 +02:00
|
|
|
?STATERR(500, <<"Internal Server Error">>);
|
2013-03-14 10:33:02 +01:00
|
|
|
Transactions ->
|
2016-11-24 13:06:06 +01:00
|
|
|
?STATVAL((integer_to_binary(Transactions)),
|
2013-03-14 10:33:02 +01:00
|
|
|
<<"transactions">>)
|
2003-01-29 18:12:23 +01:00
|
|
|
end;
|
2003-01-18 20:42:48 +01:00
|
|
|
get_node_stat(_, Name) ->
|
2016-07-31 13:51:16 +02:00
|
|
|
?STATERR(404, <<"Not Found">>).
|
2003-01-18 20:42:48 +01:00
|
|
|
|
|
|
|
search_running_node(SNode) ->
|
2013-03-14 10:33:02 +01:00
|
|
|
search_running_node(SNode,
|
|
|
|
mnesia:system_info(running_db_nodes)).
|
2003-01-18 20:42:48 +01:00
|
|
|
|
2013-03-14 10:33:02 +01:00
|
|
|
search_running_node(_, []) -> false;
|
2003-01-18 20:42:48 +01:00
|
|
|
search_running_node(SNode, [Node | Nodes]) ->
|
2013-03-14 10:33:02 +01:00
|
|
|
case iolist_to_binary(atom_to_list(Node)) of
|
|
|
|
SNode -> Node;
|
|
|
|
_ -> search_running_node(SNode, Nodes)
|
2003-01-18 20:42:48 +01:00
|
|
|
end.
|
2015-06-01 14:38:27 +02:00
|
|
|
|
2018-02-11 10:54:15 +01:00
|
|
|
mod_options(_Host) ->
|
|
|
|
[].
|
2020-01-08 10:24:51 +01:00
|
|
|
|
|
|
|
mod_doc() ->
|
|
|
|
#{desc =>
|
|
|
|
[?T("This module adds support for "
|
|
|
|
"https://xmpp.org/extensions/xep-0039.html"
|
|
|
|
"[XEP-0039: Statistics Gathering]. This protocol "
|
|
|
|
"allows you to retrieve the following statistics "
|
|
|
|
"from your ejabberd server:"), "",
|
|
|
|
?T("- Total number of registered users on the current "
|
|
|
|
"virtual host (users/total)."), "",
|
|
|
|
?T("- Total number of registered users on all virtual "
|
|
|
|
"hosts (users/all-hosts/total)."), "",
|
|
|
|
?T("- Total number of online users on the current "
|
|
|
|
"virtual host (users/online)."), "",
|
|
|
|
?T("- Total number of online users on all virtual "
|
|
|
|
"hosts (users/all-hosts/online)."), "",
|
|
|
|
?T("NOTE: The protocol extension is deferred and seems "
|
|
|
|
"like even a few clients that were supporting it "
|
|
|
|
"are now abandoned. So using this module makes "
|
|
|
|
"very little sense.")]}.
|