View source with formatted comments or as raw
    1/*  Part of ClioPatria
    2
    3    Author:        Jan Wielemaker
    4    E-mail:        J.Wielemaker@vu.nl
    5    WWW:           http://www.swi-prolog.org
    6    Copyright (c)  2010-2018, University of Amsterdam,
    7                              VU University Amsterdam.
    8    All rights reserved.
    9
   10    Redistribution and use in source and binary forms, with or without
   11    modification, are permitted provided that the following conditions
   12    are met:
   13
   14    1. Redistributions of source code must retain the above copyright
   15       notice, this list of conditions and the following disclaimer.
   16
   17    2. Redistributions in binary form must reproduce the above copyright
   18       notice, this list of conditions and the following disclaimer in
   19       the documentation and/or other materials provided with the
   20       distribution.
   21
   22    THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
   23    "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
   24    LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
   25    FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
   26    COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
   27    INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
   28    BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
   29    LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
   30    CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
   31    LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
   32    ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
   33    POSSIBILITY OF SUCH DAMAGE.
   34*/
   35
   36:- module(server_statistics,
   37          [ rdf_call_statistics_table//0,
   38            http_session_table//0,
   39            http_server_statistics//0,
   40            http_server_pool_table//0
   41          ]).   42:- use_module(library(option)).   43:- use_module(library(pairs)).   44:- use_module(library(semweb/rdf_db)).   45:- use_module(library(http/http_session)).   46:- use_module(library(http/thread_httpd)).   47:- use_module(library(http/html_write)).   48:- use_module(library(http/html_head)).   49:- use_module(user(user_db)).   50:- use_module(components(basics)).   51
   52/** <module> Server statistics components
   53
   54*/
   55
   56
   57%!  rdf_call_statistics_table//
   58%
   59%   Display table with RDF-call statistics
   60
   61rdf_call_statistics_table -->
   62    { rdf_call_stats(Lookup),
   63      (   Lookup = [rdf(_,_,_)-_|_]
   64      ->  Cols = 3
   65      ;   Cols = 4
   66      )
   67    },
   68    html(table([ class(block)
   69               ],
   70               [ tr([ th(colspan(Cols), 'Indexed (SOPG)'),
   71                      th('Calls')
   72                    ]),
   73                 \lookup_statistics(Lookup, 1)
   74               ])).
   75
   76rdf_call_stats(Lookup) :-
   77    findall(Index-Count,
   78            rdf_statistics(lookup(Index, Count)),
   79            Lookup).
   80
   81lookup_statistics([], _) -->
   82    [].
   83lookup_statistics([H|T], Row) -->
   84    odd_even_row(Row, Next, \lookup_row(H)),
   85    lookup_statistics(T, Next).
   86
   87lookup_row(rdf(S,P,O)-Count) -->
   88    html([ \i(S), \i(P), \i(O), \nc(human, Count)]).
   89lookup_row(rdf(S,P,O,G)-Count) -->
   90    html([\i(S), \i(P), \i(O), \i(G), \nc(human, Count)]).
   91
   92
   93i(I) -->
   94    html(td(class(instantiated), I)).
   95
   96
   97%!  http_session_table//
   98%
   99%   HTML component that writes a table of currently logged on users.
  100
  101http_session_table -->
  102    { findall(S, session(S), Sessions0),
  103      sort(Sessions0, Sessions),
  104      Sessions \== [], !
  105    },
  106    html([ table([ class(block)
  107                 ],
  108                 [ tr([th('User'), th('Real Name'),
  109                       th('On since'), th('Idle'), th('From')])
  110                 | \sessions(Sessions, 1)
  111                 ])
  112         ]).
  113http_session_table -->
  114    html(p('No users logged in')).
  115
  116%!  session(-Session:s(Idle, User, SessionID, Peer)) is nondet.
  117%
  118%   Enumerate all current HTTP sessions.
  119
  120session(s(Idle, User, SessionID, Peer)) :-
  121    http_current_session(SessionID, peer(Peer)),
  122    http_current_session(SessionID, idle(Idle)),
  123    (   user_property(User, session(SessionID))
  124    ->  true
  125    ;   User = (-)
  126    ).
  127
  128sessions([], _) --> [].
  129sessions([H|T], Row) -->
  130    odd_even_row(Row, Next, \session(H)),
  131    sessions(T, Next).
  132
  133session(s(Idle, -, _SessionID, Peer)) -->
  134    html([td(-), td(-), td(-), td(\idle(Idle)), td(\ip(Peer))]).
  135session(s(Idle, User, _SessionID, Peer)) -->
  136    {  (   user_property(User, realname(RealName))
  137       ->  true
  138       ;   RealName = '?'
  139       ),
  140       (   user_property(User, connection(OnSince, _Idle))
  141       ->  true
  142       ;   OnSince = 0
  143       )
  144    },
  145    html([td(User), td(RealName), td(\date(OnSince)), td(\idle(Idle)), td(\ip(Peer))]).
  146
  147idle(Time) -->
  148    { Secs is round(Time),
  149      Min is Secs // 60,
  150      Sec is Secs mod 60
  151    },
  152    html('~`0t~d~2|:~`0t~d~5|'-[Min, Sec]).
  153
  154date(Date) -->
  155    { format_time(string(S), '%+', Date)
  156    },
  157    html(S).
  158
  159ip(ip(A,B,C,D)) -->
  160    !,
  161    html('~d.~d.~d.~d'-[A,B,C,D]).
  162ip(IP) -->
  163    html('~w'-[IP]).
  164
  165
  166%!  http_server_statistics//
  167%
  168%   HTML component showing statistics on the HTTP server
  169
  170http_server_statistics -->
  171    { findall(Port-ID, http_current_worker(Port, ID), Workers),
  172      group_pairs_by_key(Workers, Servers)
  173    },
  174    html([ table([ class(block)
  175                 ],
  176                 [ \servers_stats(Servers)
  177                 ])
  178         ]).
  179
  180servers_stats([]) --> [].
  181servers_stats([H|T]) -->
  182    server_stats(H), servers_stats(T).
  183
  184:- if(catch(statistics(process_cputime, _),_,fail)).  185cputime(CPU) :- statistics(process_cputime, CPU).
  186:- else.  187cputime(CPU) :- statistics(cputime, CPU).
  188:- endif.  189
  190server_stats(Port-Workers) -->
  191    { length(Workers, NWorkers),
  192      http_server_property(Port, start_time(StartTime)),
  193      format_time(string(ST), '%+', StartTime),
  194      cputime(CPU)
  195    },
  196    html([ \server_stat('Port:', Port, odd),
  197           \server_stat('Started:', ST, even),
  198           \server_stat('Total CPU usage:', [\n('~2f',CPU), ' seconds'], odd),
  199           \request_statistics,
  200           \server_stat('# worker threads:', NWorkers, odd),
  201           tr(th(colspan(6), 'Statistics by worker')),
  202           tr([ th(rowspan(2), 'Thread'),
  203                th(rowspan(2), 'CPU'),
  204                th(colspan(3), 'Stack usage')
  205              ]),
  206           tr([ th('Local'),
  207                th('Global'),
  208                th('Trail')
  209              ]),
  210           \http_workers(Workers, odd)
  211         ]).
  212
  213server_stat(Name, Value, OE) -->
  214    html(tr(class(OE),
  215            [ th([class(p_name), colspan(3)], Name),
  216              td([class(value),  colspan(3)], Value)
  217            ])).
  218
  219
  220:- if(source_exports(library(http/http_stream), cgi_statistics/1)).  221:- use_module(library(http/http_stream)).  222request_statistics -->
  223    { cgi_statistics(requests(Count)),
  224      cgi_statistics(bytes_sent(Sent))
  225    },
  226    server_stat('Requests processed:', \n(human, Count), odd),
  227    server_stat('Bytes sent:', \n(human, Sent), even).
  228:- else.  229request_statistics --> [].
  230:- endif.  231
  232
  233http_workers([], _) -->
  234    [].
  235http_workers([H|T], OE) -->
  236    { odd_even(OE, OE2) },
  237    http_worker(H, OE),
  238    http_workers(T, OE2).
  239
  240http_worker(H, OE) -->
  241    { thread_statistics(H, localused, LU),
  242      thread_statistics(H, globalused, GU),
  243      thread_statistics(H, trailused, TU),
  244      thread_statistics(H, cputime, CPU)
  245    },
  246    html([ tr(class(OE),
  247              [ td(H),
  248                \nc('~3f', CPU),
  249                \nc(human, LU),
  250                \nc(human, GU),
  251                \nc(human, TU)
  252              ])
  253         ]).
  254
  255odd_even(even, odd).
  256odd_even(odd, even).
  257
  258
  259                 /*******************************
  260                 *            POOLS             *
  261                 *******************************/
  262
  263%!  http_server_pool_table//
  264%
  265%   Display table with statistics on thread-pools.
  266
  267http_server_pool_table -->
  268    { findall(Pool, current_thread_pool(Pool), Pools),
  269      sort(Pools, Sorted)
  270    },
  271    html(table([ id('http-server-pool'),
  272                 class(block)
  273               ],
  274               [ tr([th('Name'), th('Running'), th('Size'), th('Waiting'), th('Backlog')])
  275               | \server_pools(Sorted, 1)
  276               ])).
  277
  278server_pools([], _) --> [].
  279server_pools([H|T], Row) -->
  280    odd_even_row(Row, Next, \server_pool(H)),
  281    server_pools(T, Next).
  282
  283server_pool(Pool) -->
  284    { findall(P, thread_pool_property(Pool, P), List),
  285      memberchk(size(Size), List),
  286      memberchk(running(Running), List),
  287      memberchk(backlog(Waiting), List),
  288      memberchk(options(Options), List),
  289      option(backlog(MaxBackLog), Options, infinite)
  290    },
  291    html([ th(class(p_name), Pool),
  292           \nc(human, Running),
  293           \nc(human, Size),
  294           \nc(human, Waiting),
  295           \nc(human, MaxBackLog)
  296         ])