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>
|
|
|
|
%%%
|
|
|
|
%%%
|
2009-01-12 15:44:42 +01:00
|
|
|
%%% ejabberd, Copyright (C) 2002-2009 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
|
|
|
%%%
|
2007-12-24 13:58:05 +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., 59 Temple Place, Suite 330, Boston, MA
|
|
|
|
%%% 02111-1307 USA
|
|
|
|
%%%
|
2003-01-11 21:25:11 +01:00
|
|
|
%%%----------------------------------------------------------------------
|
|
|
|
|
|
|
|
-module(mod_stats).
|
2007-12-24 13:58:05 +01:00
|
|
|
-author('alexey@process-one.net').
|
2003-01-11 21:25:11 +01:00
|
|
|
|
2003-01-24 21:18:33 +01:00
|
|
|
-behaviour(gen_mod).
|
|
|
|
|
2005-06-20 05:18:13 +02:00
|
|
|
-export([start/2,
|
|
|
|
stop/1,
|
2003-01-11 21:25:11 +01:00
|
|
|
process_local_iq/3]).
|
|
|
|
|
2003-03-09 21:46:47 +01:00
|
|
|
-include("jlib.hrl").
|
2003-01-11 21:25:11 +01:00
|
|
|
|
2005-06-20 05:18:13 +02:00
|
|
|
start(Host, Opts) ->
|
2003-01-24 21:18:33 +01:00
|
|
|
IQDisc = gen_mod:get_opt(iqdisc, Opts, one_queue),
|
2005-06-20 05:18:13 +02:00
|
|
|
gen_iq_handler:add_iq_handler(ejabberd_local, Host, ?NS_STATS,
|
2003-01-24 21:18:33 +01:00
|
|
|
?MODULE, process_local_iq, IQDisc).
|
2003-01-11 21:25:11 +01:00
|
|
|
|
2005-06-20 05:18:13 +02:00
|
|
|
stop(Host) ->
|
|
|
|
gen_iq_handler:remove_iq_handler(ejabberd_local, Host, ?NS_STATS).
|
2003-01-29 18:12:23 +01:00
|
|
|
|
2003-01-11 21:25:11 +01:00
|
|
|
|
2007-12-07 01:09:48 +01:00
|
|
|
process_local_iq(_From, To, #iq{id = _ID, type = Type,
|
|
|
|
xmlns = XMLNS, sub_el = SubEl} = IQ) ->
|
|
|
|
%%Lang = xml:get_tag_attr_s("xml:lang", SubEl),
|
2003-01-11 21:25:11 +01:00
|
|
|
case Type of
|
|
|
|
set ->
|
2003-12-17 21:13:21 +01:00
|
|
|
IQ#iq{type = error, sub_el = [SubEl, ?ERR_NOT_ALLOWED]};
|
2003-01-11 21:25:11 +01:00
|
|
|
get ->
|
2007-12-07 01:09:48 +01:00
|
|
|
{xmlelement, _, _Attrs, Els} = SubEl,
|
2003-01-11 21:25:11 +01:00
|
|
|
Node = string:tokens(xml:get_tag_attr_s("node", SubEl), "/"),
|
|
|
|
Names = get_names(Els, []),
|
2007-12-07 01:09:48 +01:00
|
|
|
|
2005-04-17 20:08:34 +02:00
|
|
|
case get_local_stats(To#jid.server, Node, Names) of
|
2003-01-18 20:42:48 +01:00
|
|
|
{result, Res} ->
|
2003-12-17 21:13:21 +01:00
|
|
|
IQ#iq{type = result,
|
|
|
|
sub_el = [{xmlelement, "query",
|
|
|
|
[{"xmlns", XMLNS}],
|
|
|
|
Res}]};
|
2003-03-09 21:46:47 +01:00
|
|
|
{error, Error} ->
|
2003-12-17 21:13:21 +01:00
|
|
|
IQ#iq{type = error, sub_el = [SubEl, Error]}
|
2003-01-11 21:25:11 +01:00
|
|
|
end
|
|
|
|
end.
|
|
|
|
|
|
|
|
|
|
|
|
get_names([], Res) ->
|
|
|
|
Res;
|
|
|
|
get_names([{xmlelement, "stat", Attrs, _} | Els], Res) ->
|
|
|
|
Name = xml:get_attr_s("name", Attrs),
|
|
|
|
case Name of
|
|
|
|
"" ->
|
|
|
|
get_names(Els, Res);
|
|
|
|
_ ->
|
|
|
|
get_names(Els, [Name | Res])
|
|
|
|
end;
|
|
|
|
get_names([_ | Els], Res) ->
|
|
|
|
get_names(Els, Res).
|
|
|
|
|
|
|
|
|
2003-01-18 20:42:48 +01:00
|
|
|
-define(STAT(Name), {xmlelement, "stat", [{"name", Name}], []}).
|
|
|
|
|
2005-04-17 20:08:34 +02:00
|
|
|
get_local_stats(_Server, [], []) ->
|
2003-01-11 21:25:11 +01:00
|
|
|
{result,
|
2003-01-18 20:42:48 +01:00
|
|
|
[?STAT("users/online"),
|
2005-04-17 20:08:34 +02:00
|
|
|
?STAT("users/total"),
|
|
|
|
?STAT("users/all-hosts/online"),
|
|
|
|
?STAT("users/all-hosts/total")
|
2003-01-18 20:42:48 +01:00
|
|
|
]};
|
|
|
|
|
2005-04-17 20:08:34 +02:00
|
|
|
get_local_stats(Server, [], Names) ->
|
|
|
|
{result, lists:map(fun(Name) ->
|
|
|
|
get_local_stat(Server, [], Name)
|
|
|
|
end, Names)};
|
2003-01-18 20:42:48 +01:00
|
|
|
|
2005-04-17 20:08:34 +02:00
|
|
|
get_local_stats(_Server, ["running nodes", _], []) ->
|
2003-01-11 21:25:11 +01:00
|
|
|
{result,
|
2003-01-18 20:42:48 +01:00
|
|
|
[?STAT("time/uptime"),
|
|
|
|
?STAT("time/cputime"),
|
2003-01-29 18:12:23 +01:00
|
|
|
?STAT("users/online"),
|
2009-01-12 19:41:46 +01:00
|
|
|
?STAT("transactions/committed"),
|
2003-01-29 18:12:23 +01:00
|
|
|
?STAT("transactions/aborted"),
|
|
|
|
?STAT("transactions/restarted"),
|
|
|
|
?STAT("transactions/logged")
|
2003-01-18 20:42:48 +01:00
|
|
|
]};
|
|
|
|
|
2005-04-17 20:08:34 +02:00
|
|
|
get_local_stats(_Server, ["running nodes", ENode], Names) ->
|
2003-01-18 20:42:48 +01:00
|
|
|
case search_running_node(ENode) of
|
|
|
|
false ->
|
2004-02-15 21:10:40 +01:00
|
|
|
{error, ?ERR_ITEM_NOT_FOUND};
|
2003-01-18 20:42:48 +01:00
|
|
|
Node ->
|
|
|
|
{result,
|
|
|
|
lists:map(fun(Name) -> get_node_stat(Node, Name) end, Names)}
|
|
|
|
end;
|
|
|
|
|
2005-04-17 20:08:34 +02:00
|
|
|
get_local_stats(_Server, _, _) ->
|
2003-03-09 21:46:47 +01:00
|
|
|
{error, ?ERR_FEATURE_NOT_IMPLEMENTED}.
|
2003-01-18 20:42:48 +01:00
|
|
|
|
2003-01-11 21:25:11 +01:00
|
|
|
|
|
|
|
|
2003-01-18 20:42:48 +01:00
|
|
|
-define(STATVAL(Val, Unit),
|
2003-01-11 21:25:11 +01:00
|
|
|
{xmlelement, "stat",
|
|
|
|
[{"name", Name},
|
|
|
|
{"units", Unit},
|
|
|
|
{"value", Val}
|
|
|
|
], []}).
|
|
|
|
|
|
|
|
-define(STATERR(Code, Desc),
|
|
|
|
{xmlelement, "stat",
|
|
|
|
[{"name", Name}],
|
|
|
|
[{xmlelement, "error",
|
|
|
|
[{"code", Code}],
|
|
|
|
[{xmlcdata, Desc}]}]}).
|
|
|
|
|
|
|
|
|
2005-04-17 20:08:34 +02:00
|
|
|
get_local_stat(Server, [], Name) when Name == "users/online" ->
|
|
|
|
case catch ejabberd_sm:get_vh_session_list(Server) of
|
2007-12-07 01:09:48 +01:00
|
|
|
{'EXIT', _Reason} ->
|
2005-04-17 20:08:34 +02:00
|
|
|
?STATERR("500", "Internal Server Error");
|
|
|
|
Users ->
|
|
|
|
?STATVAL(integer_to_list(length(Users)), "users")
|
|
|
|
end;
|
|
|
|
|
|
|
|
get_local_stat(Server, [], Name) when Name == "users/total" ->
|
2007-12-07 01:09:48 +01:00
|
|
|
%%LServer = jlib:nameprep(Server),
|
2007-05-12 20:14:21 +02:00
|
|
|
case catch ejabberd_auth:get_vh_registered_users_number(Server) of
|
2007-12-07 01:09:48 +01:00
|
|
|
{'EXIT', _Reason} ->
|
2005-04-17 20:08:34 +02:00
|
|
|
?STATERR("500", "Internal Server Error");
|
2007-05-12 20:14:21 +02:00
|
|
|
NUsers ->
|
|
|
|
?STATVAL(integer_to_list(NUsers), "users")
|
2005-04-17 20:08:34 +02:00
|
|
|
end;
|
|
|
|
|
|
|
|
get_local_stat(_Server, [], Name) when Name == "users/all-hosts/online" ->
|
2004-12-12 23:32:15 +01:00
|
|
|
case catch mnesia:table_info(session, size) of
|
2007-12-07 01:09:48 +01:00
|
|
|
{'EXIT', _Reason} ->
|
2003-01-11 21:25:11 +01:00
|
|
|
?STATERR("500", "Internal Server Error");
|
|
|
|
Users ->
|
2004-12-12 23:32:15 +01:00
|
|
|
?STATVAL(integer_to_list(Users), "users")
|
2003-01-11 21:25:11 +01:00
|
|
|
end;
|
2005-04-17 20:08:34 +02:00
|
|
|
|
|
|
|
get_local_stat(_Server, [], Name) when Name == "users/all-hosts/total" ->
|
2004-12-12 23:32:15 +01:00
|
|
|
case catch mnesia:table_info(passwd, size) of
|
2007-12-07 01:09:48 +01:00
|
|
|
{'EXIT', _Reason} ->
|
2003-01-11 21:25:11 +01:00
|
|
|
?STATERR("500", "Internal Server Error");
|
|
|
|
Users ->
|
2004-12-12 23:32:15 +01:00
|
|
|
?STATVAL(integer_to_list(Users), "users")
|
2003-01-11 21:25:11 +01:00
|
|
|
end;
|
2005-04-17 20:08:34 +02:00
|
|
|
|
|
|
|
get_local_stat(_Server, _, Name) ->
|
2003-01-11 21:25:11 +01:00
|
|
|
?STATERR("404", "Not Found").
|
2003-01-18 20:42:48 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
get_node_stat(Node, Name) when Name == "time/uptime" ->
|
|
|
|
case catch rpc:call(Node, erlang, statistics, [wall_clock]) of
|
2007-12-07 01:09:48 +01:00
|
|
|
{badrpc, _Reason} ->
|
2003-01-18 20:42:48 +01:00
|
|
|
?STATERR("500", "Internal Server Error");
|
|
|
|
CPUTime ->
|
|
|
|
?STATVAL(
|
|
|
|
io_lib:format("~.3f", [element(1, CPUTime)/1000]), "seconds")
|
|
|
|
end;
|
|
|
|
|
|
|
|
get_node_stat(Node, Name) when Name == "time/cputime" ->
|
|
|
|
case catch rpc:call(Node, erlang, statistics, [runtime]) of
|
2007-12-07 01:09:48 +01:00
|
|
|
{badrpc, _Reason} ->
|
2003-01-18 20:42:48 +01:00
|
|
|
?STATERR("500", "Internal Server Error");
|
|
|
|
RunTime ->
|
|
|
|
?STATVAL(
|
|
|
|
io_lib:format("~.3f", [element(1, RunTime)/1000]), "seconds")
|
|
|
|
end;
|
|
|
|
|
|
|
|
get_node_stat(Node, Name) when Name == "users/online" ->
|
|
|
|
case catch rpc:call(Node, ejabberd_sm, dirty_get_my_sessions_list, []) of
|
2007-12-07 01:09:48 +01:00
|
|
|
{badrpc, _Reason} ->
|
2003-01-18 20:42:48 +01:00
|
|
|
?STATERR("500", "Internal Server Error");
|
|
|
|
Users ->
|
|
|
|
?STATVAL(integer_to_list(length(Users)), "users")
|
|
|
|
end;
|
2003-01-29 18:12:23 +01:00
|
|
|
|
2009-01-12 19:41:46 +01:00
|
|
|
get_node_stat(Node, Name) when Name == "transactions/committed" ->
|
2003-01-29 18:12:23 +01:00
|
|
|
case catch rpc:call(Node, mnesia, system_info, [transaction_commits]) of
|
2007-12-07 01:09:48 +01:00
|
|
|
{badrpc, _Reason} ->
|
2003-01-29 18:12:23 +01:00
|
|
|
?STATERR("500", "Internal Server Error");
|
|
|
|
Transactions ->
|
|
|
|
?STATVAL(integer_to_list(Transactions), "transactions")
|
|
|
|
end;
|
|
|
|
|
|
|
|
get_node_stat(Node, Name) when Name == "transactions/aborted" ->
|
|
|
|
case catch rpc:call(Node, mnesia, system_info, [transaction_failures]) of
|
2007-12-07 01:09:48 +01:00
|
|
|
{badrpc, _Reason} ->
|
2003-01-29 18:12:23 +01:00
|
|
|
?STATERR("500", "Internal Server Error");
|
|
|
|
Transactions ->
|
|
|
|
?STATVAL(integer_to_list(Transactions), "transactions")
|
|
|
|
end;
|
|
|
|
|
|
|
|
get_node_stat(Node, Name) when Name == "transactions/restarted" ->
|
|
|
|
case catch rpc:call(Node, mnesia, system_info, [transaction_restarts]) of
|
2007-12-07 01:09:48 +01:00
|
|
|
{badrpc, _Reason} ->
|
2003-01-29 18:12:23 +01:00
|
|
|
?STATERR("500", "Internal Server Error");
|
|
|
|
Transactions ->
|
|
|
|
?STATVAL(integer_to_list(Transactions), "transactions")
|
|
|
|
end;
|
|
|
|
|
|
|
|
get_node_stat(Node, Name) when Name == "transactions/logged" ->
|
|
|
|
case catch rpc:call(Node, mnesia, system_info, [transaction_log_writes]) of
|
2007-12-07 01:09:48 +01:00
|
|
|
{badrpc, _Reason} ->
|
2003-01-29 18:12:23 +01:00
|
|
|
?STATERR("500", "Internal Server Error");
|
|
|
|
Transactions ->
|
|
|
|
?STATVAL(integer_to_list(Transactions), "transactions")
|
|
|
|
end;
|
|
|
|
|
2003-01-18 20:42:48 +01:00
|
|
|
get_node_stat(_, Name) ->
|
|
|
|
?STATERR("404", "Not Found").
|
|
|
|
|
|
|
|
|
|
|
|
search_running_node(SNode) ->
|
|
|
|
search_running_node(SNode, mnesia:system_info(running_db_nodes)).
|
|
|
|
|
|
|
|
search_running_node(_, []) ->
|
|
|
|
false;
|
|
|
|
search_running_node(SNode, [Node | Nodes]) ->
|
|
|
|
case atom_to_list(Node) of
|
|
|
|
SNode ->
|
|
|
|
Node;
|
|
|
|
_ ->
|
|
|
|
search_running_node(SNode, Nodes)
|
|
|
|
end.
|
|
|
|
|