From 8793b32c8e8745e04571a30735b210f68b23ef92 Mon Sep 17 00:00:00 2001 From: vogelor <vogelor@ispconfig3> Date: Mon, 24 Nov 2008 17:51:48 +0000 Subject: [PATCH] Monitor Module now only shows the state of the "active" services Monitor Module now has system state --- install/sql/ispconfig3.sql | 7 +- interface/web/monitor/lib/module.conf.php | 142 +- interface/web/monitor/show_data.php | 305 +--- interface/web/monitor/show_log.php | 1 - interface/web/monitor/show_sys_state.php | 365 ++++ .../web/monitor/templates/show_sys_state.htm | 19 + interface/web/monitor/tools.inc.php | 244 +++ .../default/css/screen/content_ispc.css | 73 +- .../default/icons/x32/state_critical.png | Bin 0 -> 2409 bytes .../themes/default/icons/x32/state_error.png | Bin 0 -> 2212 bytes .../themes/default/icons/x32/state_info.png | Bin 0 -> 2184 bytes .../web/themes/default/icons/x32/state_ok.png | Bin 0 -> 1518 bytes .../default/icons/x32/state_unknown.png | Bin 0 -> 2832 bytes .../default/icons/x32/state_warning.png | Bin 0 -> 1931 bytes .../web/themes/default/icons/x64/network.png | Bin 0 -> 7481 bytes .../web/themes/default/icons/x64/server.png | Bin 0 -> 6880 bytes .../monitor_core_module.inc.php | 1552 +++++++++-------- 17 files changed, 1659 insertions(+), 1049 deletions(-) create mode 100644 interface/web/monitor/show_sys_state.php create mode 100644 interface/web/monitor/templates/show_sys_state.htm create mode 100644 interface/web/monitor/tools.inc.php create mode 100644 interface/web/themes/default/icons/x32/state_critical.png create mode 100644 interface/web/themes/default/icons/x32/state_error.png create mode 100644 interface/web/themes/default/icons/x32/state_info.png create mode 100644 interface/web/themes/default/icons/x32/state_ok.png create mode 100644 interface/web/themes/default/icons/x32/state_unknown.png create mode 100644 interface/web/themes/default/icons/x32/state_warning.png create mode 100644 interface/web/themes/default/icons/x64/network.png create mode 100644 interface/web/themes/default/icons/x64/server.png diff --git a/install/sql/ispconfig3.sql b/install/sql/ispconfig3.sql index 774b4a0b77..e7ccf33a09 100644 --- a/install/sql/ispconfig3.sql +++ b/install/sql/ispconfig3.sql @@ -1102,15 +1102,11 @@ CREATE TABLE `monitor_data` ( `type` varchar(255) NOT NULL, `created` int(11) NOT NULL, `data` mediumtext NOT NULL, - `state` enum('unknown','ok','warning','critical', 'error') NOT NULL default 'unknown', + `state` enum('no_state', 'unknown', 'ok', 'info', 'warning', 'critical', 'error') NOT NULL default 'unknown', PRIMARY KEY (`server_id`,`type`,`created`) ) ENGINE=MyISAM DEFAULT CHARSET=latin1; -SET FOREIGN_KEY_CHECKS = 1; - - - # iso_country_list.sql # @@ -1373,6 +1369,7 @@ INSERT INTO country VALUES ('ZM','ZAMBIA','Zambia','ZMB','894'); INSERT INTO country VALUES ('ZW','ZIMBABWE','Zimbabwe','ZWE','716'); +SET FOREIGN_KEY_CHECKS = 1; diff --git a/interface/web/monitor/lib/module.conf.php b/interface/web/monitor/lib/module.conf.php index d9a78a710d..2a443e3a1c 100644 --- a/interface/web/monitor/lib/module.conf.php +++ b/interface/web/monitor/lib/module.conf.php @@ -7,7 +7,17 @@ $module["name"] = "monitor"; $module["title"] = "Monitor"; $module["template"] = "module.tpl.htm"; $module["tab_width"] = ''; -$module["startpage"] = "monitor/show_data.php?type=overview"; +$module["startpage"] = "monitor/show_sys_state.php?state=system"; + +unset($items); +$items[] = array( 'title' => "Show System State", + 'target' => 'content', + 'link' => 'monitor/show_sys_state.php?state=system'); + +$module["nav"][] = array( 'title' => 'System State', + 'open' => 1, + 'items' => $items); + /* We need all the available servers on the left navigation. @@ -15,24 +25,25 @@ $module["startpage"] = "monitor/show_data.php?type=overview"; */ $servers = $app->db->queryAllRecords("SELECT server_id, server_name FROM server order by server_name"); -$dropDown = "<select id='server_id' onchange=\"loadContent('monitor/show_data.php?type=overview&server=' + document.getElementById('server_id').value);\">"; +$dropDown = "<select id='server_id' onchange=\"loadContent('monitor/show_sys_state.php?state=server&server=' + document.getElementById('server_id').value);\">"; foreach ($servers as $server) { - $dropDown .= "<option value='" . $server['server_id'] . "|" . $server['server_name'] . "'>" . $server['server_name'] . "</option>"; + $dropDown .= "<option value='" . $server['server_id'] . "|" . $server['server_name'] . "'>" . $server['server_name'] . "</option>"; } $dropDown .= "</select>"; /* Now add them as dropdown to the navigation */ +unset($items); $items[] = array( 'title' => $dropDown, - 'target' => '', // no action! - 'link' => ''); // no action! + 'target' => '', // no action! + 'link' => ''); // no action! $module["nav"][] = array( 'title' => 'Server to Monitor', - 'open' => 1, - 'items' => $items); - + 'open' => 1, + 'items' => $items); + /* The first Server at the list is the server first selected */ @@ -40,83 +51,90 @@ $_SESSION['monitor']['server_id'] = $servers[0]['server_id']; $_SESSION['monitor']['server_name'] = $servers[0]['server_name']; /* - Logmonitoring module -*/ -// aufräumen + * Logmonitoring module + */ + +/* + * Clear and set the Navigation-Items + */ unset($items); -$items[] = array( 'title' => "Server Load", - 'target' => 'content', - 'link' => 'monitor/show_data.php?type=server_load'); +$items[] = array( 'title' => "Show Server State", + 'target' => 'content', + 'link' => 'monitor/show_sys_state.php?state=server'); + +$items[] = array( 'title' => "Show Server Load", + 'target' => 'content', + 'link' => 'monitor/show_data.php?type=server_load'); -$items[] = array( 'title' => "Disk usage", - 'target' => 'content', - 'link' => 'monitor/show_data.php?type=disk_usage'); +$items[] = array( 'title' => "Show Disk usage", + 'target' => 'content', + 'link' => 'monitor/show_data.php?type=disk_usage'); -$items[] = array( 'title' => "Memory usage", - 'target' => 'content', - 'link' => 'monitor/show_data.php?type=mem_usage'); +$items[] = array( 'title' => "Show Memory usage", + 'target' => 'content', + 'link' => 'monitor/show_data.php?type=mem_usage'); -$items[] = array( 'title' => "Services", - 'target' => 'content', - 'link' => 'monitor/show_data.php?type=services'); +$items[] = array( 'title' => "Show Services", + 'target' => 'content', + 'link' => 'monitor/show_data.php?type=services'); $module["nav"][] = array( 'title' => 'Monitoring', - 'open' => 1, - 'items' => $items); + 'open' => 1, + 'items' => $items); -// aufräumen +/* + * Clear and set the Navigation-Items + */ unset($items); -$items[] = array( 'title' => "CPU", - 'target' => 'content', - 'link' => 'monitor/show_data.php?type=cpu_info'); +$items[] = array( 'title' => "Show CPU info", + 'target' => 'content', + 'link' => 'monitor/show_data.php?type=cpu_info'); $module["nav"][] = array( 'title' => 'System-Information', - 'open' => 1, - 'items' => $items); - -// aufräumen -unset($items); + 'open' => 1, + 'items' => $items); +/* + * Logmonitoring module + */ /* - Logmonitoring module -*/ + * Clear and set the Navigation-Items + */ +unset($items); -$items[] = array( 'title' => "Mail log", - 'target' => 'content', - 'link' => 'monitor/show_log.php?log=log_mail'); +$items[] = array( 'title' => "Show Mail-Log", + 'target' => 'content', + 'link' => 'monitor/show_log.php?log=log_mail'); -$items[] = array( 'title' => "Mail warn", - 'target' => 'content', - 'link' => 'monitor/show_log.php?log=log_mail_warn'); +$items[] = array( 'title' => "Show Mail warn-Log", + 'target' => 'content', + 'link' => 'monitor/show_log.php?log=log_mail_warn'); -$items[] = array( 'title' => "Mail err", - 'target' => 'content', - 'link' => 'monitor/show_log.php?log=log_mail_err'); +$items[] = array( 'title' => "Show Mail err-Log", + 'target' => 'content', + 'link' => 'monitor/show_log.php?log=log_mail_err'); -$items[] = array( 'title' => "Messages", - 'target' => 'content', - 'link' => 'monitor/show_log.php?log=log_messages'); +$items[] = array( 'title' => "Show Messages-Log", + 'target' => 'content', + 'link' => 'monitor/show_log.php?log=log_messages'); -$items[] = array( 'title' => "Freshclam", - 'target' => 'content', - 'link' => 'monitor/show_log.php?log=log_freshclam'); +$items[] = array( 'title' => "Show Freshclam-Log", + 'target' => 'content', + 'link' => 'monitor/show_log.php?log=log_freshclam'); -$items[] = array( 'title' => "Clamav", - 'target' => 'content', - 'link' => 'monitor/show_log.php?log=log_clamav'); +$items[] = array( 'title' => "Show Clamav-Log", + 'target' => 'content', + 'link' => 'monitor/show_log.php?log=log_clamav'); -$items[] = array( 'title' => "ISPConfig", - 'target' => 'content', - 'link' => 'monitor/show_log.php?log=log_ispconfig'); +$items[] = array( 'title' => "Show ISPConfig-Log", + 'target' => 'content', + 'link' => 'monitor/show_log.php?log=log_ispconfig'); $module["nav"][] = array( 'title' => 'Logfiles', - 'open' => 1, - 'items' => $items); - -// aufräumen -unset($items); + 'open' => 1, + 'items' => $items); ?> \ No newline at end of file diff --git a/interface/web/monitor/show_data.php b/interface/web/monitor/show_data.php index bee721be4a..37c77dc6c8 100644 --- a/interface/web/monitor/show_data.php +++ b/interface/web/monitor/show_data.php @@ -30,6 +30,7 @@ EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. require_once('../../lib/config.inc.php'); require_once('../../lib/app.inc.php'); +require_once('tools.inc.php'); //* Check permissions for module $app->auth->check_module_permissions('monitor'); @@ -38,58 +39,42 @@ $app->auth->check_module_permissions('monitor'); /* Get the dataType to show */ $dataType = $_GET["type"]; -/* Change the Server if needed */ -if (isset($_GET['server'])){ - $server = explode('|', $_GET['server'], 2); - $_SESSION['monitor']['server_id'] = $server[0]; - $_SESSION['monitor']['server_name'] = $server[1]; -} - - $output = ''; switch($dataType) { - case 'server_load': - $template = 'templates/show_data.htm'; - $output .= showServerLoad(); - $title = $app->lng("Server Load").' (Server: ' . $_SESSION['monitor']['server_name'] . ')'; - $description = ''; - break; - case 'disk_usage': - $template = 'templates/show_data.htm'; - $output .= showDiskUsage(); - $title = $app->lng("Disk usage").' (Server: ' . $_SESSION['monitor']['server_name'] . ')'; - $description = ''; - break; - case 'mem_usage': - $template = 'templates/show_data.htm'; - $output .= showMemUsage(); - $title = $app->lng("Memory usage").' (Server: ' . $_SESSION['monitor']['server_name'] . ')'; - $description = ''; - break; - case 'cpu_info': - $template = 'templates/show_data.htm'; - $output .= showCpuInfo(); - $title = $app->lng("CPU info").' (Server: ' . $_SESSION['monitor']['server_name'] . ')'; - $description = ''; - break; - case 'services': - $template = 'templates/show_data.htm'; - $output .= showServices(); - $title = $app->lng("Status of services").' (Server: ' . $_SESSION['monitor']['server_name'] . ')'; - $description = ''; - break; - case 'overview': - $template = 'templates/show_data.htm'; - $output .= showServerLoad(); - $output .= ' '. showDiskUsage(); - $output .= ' '.showServices(); - $title = $app->lng("System Monitor").' (Server: ' . $_SESSION['monitor']['server_name'] . ')'; - $description = ''; - break; - default: - $template = ''; - break; + case 'server_load': + $template = 'templates/show_data.htm'; + $output .= showServerLoad(); + $title = $app->lng("Server Load").' (Server: ' . $_SESSION['monitor']['server_name'] . ')'; + $description = ''; + break; + case 'disk_usage': + $template = 'templates/show_data.htm'; + $output .= showDiskUsage(); + $title = $app->lng("Disk usage").' (Server: ' . $_SESSION['monitor']['server_name'] . ')'; + $description = ''; + break; + case 'mem_usage': + $template = 'templates/show_data.htm'; + $output .= showMemUsage(); + $title = $app->lng("Memory usage").' (Server: ' . $_SESSION['monitor']['server_name'] . ')'; + $description = ''; + break; + case 'cpu_info': + $template = 'templates/show_data.htm'; + $output .= showCpuInfo(); + $title = $app->lng("CPU info").' (Server: ' . $_SESSION['monitor']['server_name'] . ')'; + $description = ''; + break; + case 'services': + $template = 'templates/show_data.htm'; + $output .= showServices(); + $title = $app->lng("Status of services").' (Server: ' . $_SESSION['monitor']['server_name'] . ')'; + $description = ''; + break; + default: + $template = ''; + break; } @@ -105,226 +90,4 @@ $app->tpl->setVar("description",$description); $app->tpl_defaults(); $app->tpl->pparse(); - - - - -function showServerLoad(){ - global $app; - - /* fetch the Data from the DB */ - $record = $app->db->queryOneRecord("SELECT data, state FROM monitor_data WHERE type = 'server_load' and server_id = " . $_SESSION['monitor']['server_id'] . " order by created desc"); - - if(isset($record['data'])) { - $data = unserialize($record['data']); - - /* - Format the data - */ - $html .= - '<table id="system_load"> - <tr> - <td>' . $app->lng("Server online since").':</td> - <td>' . $data['up_days'] . ' days, ' . $data['up_hours'] . ':' . $data['up_minutes'] . ' hours</center></td> - </tr> - <tr> - <td>' . $app->lng("Users online").':</td> - <td>' . $data['user_online'] . '</td> - </tr>' . - '<tr> - <td>' . $app->lng("System load 1 minute") . ':</td> - <td>' . $data['load_1'] . '</td> - </tr> - <tr> - <td>' . $app->lng("System load 5 minutes") . ':</td> - <td>' . $data['load_5'] . '</td> - </tr> - <tr> - <td>'.$app->lng("System load 15 minutes").':</td> - <td>' . $data['load_15'] . '</td> - </tr> - </table>'; - } else { - $html = '<p>'.$app->lng("no_data_serverload_txt").'</p>'; - } - - return $html; -} - -function showDiskUsage () { - global $app; - - /* fetch the Data from the DB */ - $record = $app->db->queryOneRecord("SELECT data, state FROM monitor_data WHERE type = 'disk_usage' and server_id = " . $_SESSION['monitor']['server_id'] . " order by created desc"); - - if(isset($record['data'])) { - $data = unserialize($record['data']); - - /* - Format the data - */ - $html .= '<table id="system_disk">'; - foreach($data as $line) { - $html .= '<tr>'; - foreach ($line as $item) { - $html .= '<td>' . $item . '</td>'; - } - $html .= '</tr>'; - } - $html .= '</table>'; - } else { - $html = '<p>'.$app->lng("no_data_diskusage_txt").'</p>'; - } - - - return $html; -} - - -function showMemUsage () -{ - global $app; - - /* fetch the Data from the DB */ - $record = $app->db->queryOneRecord("SELECT data, state FROM monitor_data WHERE type = 'mem_usage' and server_id = " . $_SESSION['monitor']['server_id'] . " order by created desc"); - - if(isset($record['data'])) { - $data = unserialize($record['data']); - - /* - Format the data - */ - $html .= '<table id="system_memusage">'; - - foreach($data as $key => $value){ - if ($key != '') { - $html .= '<tr> - <td>' . $key . ':</td> - <td>' . $value . '</td> - </tr>'; - } - } - $html .= '</table>'; - } else { - $html = '<p>'.$app->lng("no_data_memusage_txt").'</p>'; - } - - return $html; -} - -function showCpuInfo () -{ - global $app; - - /* fetch the Data from the DB */ - $record = $app->db->queryOneRecord("SELECT data, state FROM monitor_data WHERE type = 'cpu_info' and server_id = " . $_SESSION['monitor']['server_id'] . " order by created desc"); - - if(isset($record['data'])) { - $data = unserialize($record['data']); - - /* - Format the data - */ - $html .= '<table id="system_cpu">'; - foreach($data as $key => $value){ - if ($key != '') { - $html .= '<tr> - <td>' . $key . ':</td> - <td>' . $value . '</td> - </tr>'; - } - } - $html .= '</table>'; - } else { - $html = '<p>'.$app->lng("no_data_cpuinfo_txt").'</p>'; - } - - return $html; -} - -function showServices () -{ - global $app; - - /* fetch the Data from the DB */ - $record = $app->db->queryOneRecord("SELECT data, state FROM monitor_data WHERE type = 'services' and server_id = " . $_SESSION['monitor']['server_id'] . " order by created desc"); - - if(isset($record['data'])) { - $data = unserialize($record['data']); - - /* - Format the data - */ - $html .= '<table id="system_services">'; - - if($data['webserver'] == true) { - $status = '<span class="online">Online</span>'; - } else { - $status = '<span class="offline">Offline</span>'; - } - $html .= '<tr> - <td>Web-Server:</td> - <td>'.$status.'</td> - </tr>'; - - - if($data['ftpserver'] == true) { - $status = '<span class="online">Online</span>'; - } else { - $status = '<span class="offline">Offline</span>'; - } - $html .= '<tr> - <td>FTP-Server:</td> - <td>'.$status.'</td> - </tr>'; - - if($data['smtpserver'] == true) { - $status = '<span class="online">Online</span>'; - } else { - $status = '<span class="offline">Offline</span>'; - } - $html .= '<tr> - <td>SMTP-Server:</td> - <td>'.$status.'</td> - </tr>'; - - if($data['pop3server'] == true) { - $status = '<span class="online">Online</span>'; - } else { - $status = '<span class="offline">Offline</span>'; - } - $html .= '<tr> - <td>POP3-Server:</td> - <td>'.$status.'</td> - </tr>'; - - if($data['bindserver'] == true) { - $status = '<span class="online">Online</span>'; - } else { - $status = '<span class="offline">Offline</span>'; - } - $html .= '<tr> - <td>DNS-Server:</td> - <td>'.$status.'</td> - </tr>'; - - if($data['mysqlserver'] == true) { - $status = '<span class="online">Online</span>'; - } else { - $status = '<span class="offline">Offline</span>'; - } - $html .= '<tr> - <td>mySQL-Server:</td> - <td>'.$status.'</td> - </tr>'; - - - $html .= '</table></div>'; - } else { - $html = '<p>'.$app->lng("no_data_services_txt").'</p>'; - } - - - return $html; -} ?> \ No newline at end of file diff --git a/interface/web/monitor/show_log.php b/interface/web/monitor/show_log.php index 706134b4fa..3f2cabd072 100644 --- a/interface/web/monitor/show_log.php +++ b/interface/web/monitor/show_log.php @@ -43,7 +43,6 @@ $app->tpl->setInclude('content_tpl','templates/show_log.htm'); $refresh = (isset($_GET["refresh"]))?intval($_GET["refresh"]):0; $logParam = $_GET["log"]; - /* Setting the db-type and the caption */ diff --git a/interface/web/monitor/show_sys_state.php b/interface/web/monitor/show_sys_state.php new file mode 100644 index 0000000000..20db251e7e --- /dev/null +++ b/interface/web/monitor/show_sys_state.php @@ -0,0 +1,365 @@ +<?php +/* +Copyright (c) 2007-2008, Till Brehm, projektfarm Gmbh and Oliver Vogel www.muv.com +All rights reserved. + +Redistribution and use in source and binary forms, with or without modification, +are permitted provided that the following conditions are met: + + * Redistributions of source code must retain the above copyright notice, + this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. + * Neither the name of ISPConfig nor the names of its contributors + may be used to endorse or promote products derived from this software without + specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND +ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. +IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, +INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, +BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY +OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING +NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, +EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. +*/ + +require_once('../../lib/config.inc.php'); +require_once('../../lib/app.inc.php'); +require_once('tools.inc.php'); + +/* Check permissions for module */ +$app->auth->check_module_permissions('monitor'); + +/* Change the Server if needed */ +if (isset($_GET['server'])){ + $server = explode('|', $_GET['server'], 2); + $_SESSION['monitor']['server_id'] = $server[0]; + $_SESSION['monitor']['server_name'] = $server[1]; +} + +/* + * Loading the template + */ +$app->uses('tpl'); +$app->tpl->newTemplate("form.tpl.htm"); +$app->tpl->setInclude('content_tpl','templates/show_sys_state.htm'); + +/* + * setting the content + */ +if ($_GET['state'] == 'server') +{ + $output = _getServerState($_SESSION['monitor']['server_id'], $_SESSION['monitor']['server_name'], true); + $title = "Server State"; +} +else +{ + $output = _getSysState(); + $title = "System State"; +} + +$app->tpl->setVar("state_data",$output); +$app->tpl->setVar("title",$title); +$app->tpl->setVar("description",$description); + +/* + * doing the output + */ +$app->tpl_defaults(); +$app->tpl->pparse(); + + +function _getSysState(){ + global $app; + + /* + * Get all Servers and calculate the state of them + */ + $html = ''; + + $servers = $app->db->queryAllRecords("SELECT server_id, server_name FROM server order by server_name"); + foreach ($servers as $server) + { + $html .= _getServerState($server['server_id'], $server['server_name'], false); + } + + return $html; +} + +/* + * Calculates the State of ONE Server + */ +function _getServerState($serverId, $serverName, $showAll) +{ + global $app; + + /* The State of the server */ + $serverState = 'unknown'; + + /** The Number of several infos, warnings, errors, ... */ + $count = array('unknown' => 0, 'info' => 0, 'warning' => 0, 'critical' => 0, 'error' => 0); + + /** The messages */ + $messages = array(); + + /** The Result of the function */ + $res = ''; + + /* + * get all monitoring-data from the server als process then + * (count them and set the server-state) + */ + $records = $app->db->queryAllRecords("SELECT DISTINCT type FROM monitor_data WHERE server_id = " . $serverId); + foreach($records as $record){ + _processDbState($record['type'], $serverId, &$serverState, &$messages); + } + + $res .= '<div class="systemmonitor-state systemmonitor-state-' . $serverState . '">'; + $res .= '<div class="systemmonitor-serverstate">'; + $res .= '<div class="systemmonitor-state-' . $serverState . '-icon">'; + $res .= 'Server: ' . $serverName . '<br />'; + $res .= 'State: ' . $serverState . '<br />'; + // $res .= sizeof($messages['ok']) . ' ok | '; + $res .= sizeof($messages['unknown']) . ' unknown | '; + $res .= sizeof($messages['info']) . ' info | '; + $res .= sizeof($messages['warning']) . ' warning | '; + $res .= sizeof($messages['critical']) . ' critical | '; + $res .= sizeof($messages['error']) . ' error <br />'; + $res .= '<br />'; + if ($showAll){ + /* + * if we have to show all, then we do it... + */ + + /* + * Show all messages + */ + foreach($messages as $key => $state){ + /* + * There is no need, to show the "ok" - messages + */ + if ($key != 'ok') + { + $res .= $key . ':<br />'; + foreach ($state as $msg) + { + $res .= $msg . '<br />'; + } + $res .= '<br />'; + } + } + } + else + { + /* + * if not, we only show a link to the server... + */ + $res .= "<a href='#' onclick='loadContent(\"monitor/show_sys_state.php?state=server&server=" . $serverId . '|' . $serverName . "\");'> More information...</a>"; + } + $res .= '</div>'; + $res .= '</div>'; + $res .= '</div>'; + + if ($showAll){ + /* + * Show some state-info + */ + $res .= showServerLoad(); + $res .= ' '. showDiskUsage(); + $res .= ' '.showServices(); + } + + + return $res; +} + +/* + * gets the state from the db and process it + */ +function _processDbState($type, $serverId, &$serverState, &$messages) +{ + global $app; + + /* + * Always the NEWEST record of each monitoring is responsible for the + * state + */ + // get the State from the DB + $record = $app->db->queryOneRecord("SELECT state FROM monitor_data WHERE type = '" . $type . "' and server_id = " . $serverId . " order by created desc"); + // change the new state to the highest state + $serverState = _setState($serverState, $record['state']); + // count the states + $count[$record['state']]+= 1; + + /* + * The message depands on the type and the state + */ + if ($type == 'cpu_info'){ + /* this type has no state */ + } + if ($type == 'disk_usage'){ + switch ($record['state']) { + case 'ok': + $messages['ok'][] = 'The state of your Hard-Disk space is ok ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=disk_usage\");'>[more...]</a>"; + break; + case 'info': + $messages['info'][] = 'Your Hard-Disk space is going full ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=disk_usage\");'>[more...]</a>"; + break; + case 'warning': + $messages['warning'][] = 'Your Hard-Disk is nearly full ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=disk_usage\");'>[more...]</a>"; + break; + case 'critical': + $messages['critical'][] = 'Your Hard-Disk is very full '. + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=disk_usage\");'>[more...]</a>"; + break; + case 'error': + $messages['error'][] = 'Your Hard-Disk has no more space left ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=disk_usage\");'>[more...]</a>"; + break; + + default: + $messages['unknown'][] = 'Hard-Disk: ??? ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=disk_usage\");'>[more...]</a>"; + break; + } + } + if ($type == 'mem_usage'){ + /* this type has no state */ + } + if ($type == 'server_load'){ + switch ($record['state']) { + case 'ok': + $messages['ok'][] = 'Your Server load is ok ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=server_load\");'>[more...]</a>"; + break; + case 'info': + $messages['info'][] = 'Your Server in under heavy load ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=server_load\");'>[more...]</a>"; + break; + case 'warning': + $messages['warning'][] = 'Your Server in under high load ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=server_load\");'>[more...]</a>"; + break; + case 'critical': + $messages['critical'][] = 'Your Server in under higher load ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=server_load\");'>[more...]</a>"; + break; + case 'error': + $messages['error'][] = 'Your Server in under highest load ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=server_load\");'>[more...]</a>"; + break; + default: + $messages['unknown'][] = 'Server Load: ??? ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=server_load\");'>[more...]</a>"; + break; + } + } + if ($type == 'services'){ + switch ($record['state']) { + case 'ok': + $messages['error'][] = 'All needed Services are online ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=services\");'>[more...]</a>"; + + break; + case 'error': + $messages['error'][] = 'One or more needed Services are offline ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=services\");'>[more...]</a>"; + break; + default: + $messages['unknown'][] = 'services:??? ' . + "<a href='#' onclick='loadContent(\"monitor/show_data.php?type=services\");'>[more...]</a>"; + break; + } + } + if ($type == 'log_clamav'){ + /* this type has no state */ + } + if ($type == 'log_freshclam'){ + /* this type has no state */ + } + if ($type == 'log_ispconfig'){ + /* this type has no state */ + } + if ($type == 'log_mail'){ + /* this type has no state */ + } + if ($type == 'log_mail_err'){ + /* this type has no state */ + } + if ($type == 'log_mail_warn'){ + /* this type has no state */ + } + if ($type == 'log_messages'){ + /* this type has no state */ + } +} + + /* + * Set the state to the given level (or higher, but not lesser). + * * If the actual state is critical and you call the method with ok, + * then the state is critical. + * + * * If the actual state is critical and you call the method with error, + * then the state is error. + */ +function _setState($oldState, $newState) +{ + /* + * Calculate the weight of the old state + */ + switch ($oldState) { + case 'no_state': $oldInt = 0; + break; + case 'unknown': $oldInt = 1; + break; + case 'ok': $oldInt = 2; + break; + case 'info': $oldInt = 3; + break; + case 'warning': $oldInt = 4; + break; + case 'critical': $oldInt = 5; + break; + case 'error': $oldInt = 6; + break; + } + /* + * Calculate the weight of the new state + */ + switch ($newState) { + case 'no_state': $newInt = 0 ; + break; + case 'ok': $newInt = 1 ; + break; + case 'unknown': $newInt = 2 ; + break; + case 'info': $newInt = 3 ; + break; + case 'warning': $newInt = 4 ; + break; + case 'critical': $newInt = 5 ; + break; + case 'error': $newInt = 6 ; + break; + } + + /* + * Set to the higher level + */ + if ($newInt > $oldInt){ + return $newState; + } + else + { + return $oldState; + } +} + +?> \ No newline at end of file diff --git a/interface/web/monitor/templates/show_sys_state.htm b/interface/web/monitor/templates/show_sys_state.htm new file mode 100644 index 0000000000..8281f0ba8d --- /dev/null +++ b/interface/web/monitor/templates/show_sys_state.htm @@ -0,0 +1,19 @@ +<h2><tmpl_var name="title"></h2> + +<div class="panel"> + + <!--div class="pnl_toolsarea"> + <fieldset><legend>Tools</legend> + <div class="buttons"> + <select name="refreshinterval" id="refreshinterval" onChange="loadContentRefresh('monitor/show_log.php?log={tmpl_var name="log_id"}')">{tmpl_var name="refresh"}</select> + </div> + </fieldset> + </div--> + + <div class="pnl_formarea"> + <fieldset><!-- legend>Sys-State</legend --> + <div class="stateview"><tmpl_var name="state_data"></div> + </fieldset> + </div> + +</div> diff --git a/interface/web/monitor/tools.inc.php b/interface/web/monitor/tools.inc.php new file mode 100644 index 0000000000..a23867195e --- /dev/null +++ b/interface/web/monitor/tools.inc.php @@ -0,0 +1,244 @@ +<?php +function showServerLoad(){ + global $app; + + /* fetch the Data from the DB */ + $record = $app->db->queryOneRecord("SELECT data, state FROM monitor_data WHERE type = 'server_load' and server_id = " . $_SESSION['monitor']['server_id'] . " order by created desc"); + + if(isset($record['data'])) { + $data = unserialize($record['data']); + + /* + Format the data + */ + $html .= + '<table id="system_load"> + <tr> + <td>' . $app->lng("Server online since").':</td> + <td>' . $data['up_days'] . ' days, ' . $data['up_hours'] . ':' . $data['up_minutes'] . ' hours</center></td> + </tr> + <tr> + <td>' . $app->lng("Users online").':</td> + <td>' . $data['user_online'] . '</td> + </tr>' . + '<tr> + <td>' . $app->lng("System load 1 minute") . ':</td> + <td>' . $data['load_1'] . '</td> + </tr> + <tr> + <td>' . $app->lng("System load 5 minutes") . ':</td> + <td>' . $data['load_5'] . '</td> + </tr> + <tr> + <td>'.$app->lng("System load 15 minutes").':</td> + <td>' . $data['load_15'] . '</td> + </tr> + </table>'; + } else { + $html = '<p>'.$app->lng("no_data_serverload_txt").'</p>'; + } + + return $html; +} + +function showDiskUsage () { + global $app; + + /* fetch the Data from the DB */ + $record = $app->db->queryOneRecord("SELECT data, state FROM monitor_data WHERE type = 'disk_usage' and server_id = " . $_SESSION['monitor']['server_id'] . " order by created desc"); + + if(isset($record['data'])) { + $data = unserialize($record['data']); + + /* + Format the data + */ + $html .= '<table id="system_disk">'; + foreach($data as $line) { + $html .= '<tr>'; + foreach ($line as $item) { + $html .= '<td>' . $item . '</td>'; + } + $html .= '</tr>'; + } + $html .= '</table>'; + } else { + $html = '<p>'.$app->lng("no_data_diskusage_txt").'</p>'; + } + + + return $html; +} + + +function showMemUsage () +{ + global $app; + + /* fetch the Data from the DB */ + $record = $app->db->queryOneRecord("SELECT data, state FROM monitor_data WHERE type = 'mem_usage' and server_id = " . $_SESSION['monitor']['server_id'] . " order by created desc"); + + if(isset($record['data'])) { + $data = unserialize($record['data']); + + /* + Format the data + */ + $html .= '<table id="system_memusage">'; + + foreach($data as $key => $value){ + if ($key != '') { + $html .= '<tr> + <td>' . $key . ':</td> + <td>' . $value . '</td> + </tr>'; + } + } + $html .= '</table>'; + } else { + $html = '<p>'.$app->lng("no_data_memusage_txt").'</p>'; + } + + return $html; +} + +function showCpuInfo () +{ + global $app; + + /* fetch the Data from the DB */ + $record = $app->db->queryOneRecord("SELECT data, state FROM monitor_data WHERE type = 'cpu_info' and server_id = " . $_SESSION['monitor']['server_id'] . " order by created desc"); + + if(isset($record['data'])) { + $data = unserialize($record['data']); + + /* + Format the data + */ + $html .= '<table id="system_cpu">'; + foreach($data as $key => $value){ + if ($key != '') { + $html .= '<tr> + <td>' . $key . ':</td> + <td>' . $value . '</td> + </tr>'; + } + } + $html .= '</table>'; + } else { + $html = '<p>'.$app->lng("no_data_cpuinfo_txt").'</p>'; + } + + return $html; +} + +function showServices () +{ + global $app; + + /* fetch the Data from the DB */ + $record = $app->db->queryOneRecord("SELECT data, state FROM monitor_data WHERE type = 'services' and server_id = " . $_SESSION['monitor']['server_id'] . " order by created desc"); + + if(isset($record['data'])) { + $data = unserialize($record['data']); + + /* + Format the data + */ + $html .= '<table id="system_services">'; + + if($data['webserver'] != -1) { + if($data['webserver'] == 1) { + $status = '<span class="online">Online</span>'; + } else { + $status = '<span class="offline">Offline</span>'; + } + $html .= '<tr> + <td>Web-Server:</td> + <td>'.$status.'</td> + </tr>'; + } + + + if($data['ftpserver'] != -1) { + if($data['ftpserver'] == 1) { + $status = '<span class="online">Online</span>'; + } else { + $status = '<span class="offline">Offline</span>'; + } + $html .= '<tr> + <td>FTP-Server:</td> + <td>'.$status.'</td> + </tr>'; + } + + if($data['smtpserver'] != -1) { + if($data['smtpserver'] == 1) { + $status = '<span class="online">Online</span>'; + } else { + $status = '<span class="offline">Offline</span>'; + } + $html .= '<tr> + <td>SMTP-Server:</td> + <td>'.$status.'</td> + </tr>'; + } + + if($data['pop3server'] != -1) { + if($data['pop3server'] == 1) { + $status = '<span class="online">Online</span>'; + } else { + $status = '<span class="offline">Offline</span>'; + } + $html .= '<tr> + <td>POP3-Server:</td> + <td>'.$status.'</td> + </tr>'; + } + + if($data['imapserver'] != -1) { + if($data['imapserver'] == 1) { + $status = '<span class="online">Online</span>'; + } else { + $status = '<span class="offline">Offline</span>'; + } + $html .= '<tr> + <td>IMAP-Server:</td> + <td>'.$status.'</td> + </tr>'; + } + + if($data['bindserver'] != -1) { + if($data['bindserver'] == 1) { + $status = '<span class="online">Online</span>'; + } else { + $status = '<span class="offline">Offline</span>'; + } + $html .= '<tr> + <td>DNS-Server:</td> + <td>'.$status.'</td> + </tr>'; + } + + if($data['mysqlserver'] != -1) { + if($data['mysqlserver'] == 1) { + $status = '<span class="online">Online</span>'; + } else { + $status = '<span class="offline">Offline</span>'; + } + $html .= '<tr> + <td>mySQL-Server:</td> + <td>'.$status.'</td> + </tr>'; + } + + + $html .= '</table></div>'; + } else { + $html = '<p>'.$app->lng("no_data_services_txt").'</p>'; + } + + + return $html; +} +?> diff --git a/interface/web/themes/default/css/screen/content_ispc.css b/interface/web/themes/default/css/screen/content_ispc.css index 28fd643d5c..c97a364007 100644 --- a/interface/web/themes/default/css/screen/content_ispc.css +++ b/interface/web/themes/default/css/screen/content_ispc.css @@ -3,7 +3,7 @@ * "Yet Another Multicolumn Layout" - (X)HTML/CSS Framework * * (en) Uniform design of ISPConfig elements - ISPConfig 3: default theme - * (de) Einheitliche Standardformatierungen für ISPConfig-Elemente - ISPConfig 3: default theme + * (de) Einheitliche Standardformatierungen f�r ISPConfig-Elemente - ISPConfig 3: default theme * * @copyright Copyright 2005-2008, Dirk Jesse * @license CC-A 2.0 (http://creativecommons.org/licenses/by/2.0/), @@ -94,8 +94,75 @@ table.list .tbl_row_uneven { background: #f0f8ff; } table.list tr:hover { background: #fffacd; } - - .systemmonitor table { + .systemmonitor-state { + margin:20px 0; + font-family: Consolas, "Lucida Console", "Courier New", monospace; + font-size: 0.9em; + } + .systemmonitor-state-unknown { + border: 1px solid #30302e; + background-color: #cecfc5; + } + .systemmonitor-state-ok { + border: 1px solid #23fb00; + background-color: #adffa2; + } + .systemmonitor-state-info { + border: 1px solid #fdff00; + background-color: #fdffa2; + } + .systemmonitor-state-warning { + border: 1px solid #ffa800; + background-color: #ffda93; + } + .systemmonitor-state-critical { + border: 1px solid #ff0000; + background-color: #ffb9b9; + } + .systemmonitor-state-error { + border: 1px solid #ff0000; + background-color: #ff7f7f; + } + .systemmonitor-systemstate { + background-image:url("../../icons/x64/network.png"); + background-repeat: no-repeat; + } + .systemmonitor-serverstate { + background-image:url("../../icons/x64/server.png"); + background-repeat: no-repeat; + } + .systemmonitor-state-unknown-icon { + padding:2px 10px 2px 80px; + background-image:url("../../icons/x32/state_unknown.png"); + background-repeat: no-repeat; + } + .systemmonitor-state-ok-icon { + padding:2px 10px 2px 80px; + background-image:url("../../icons/x32/state_ok.png"); + background-repeat: no-repeat; + } + .systemmonitor-state-info-icon { + padding:2px 10px 2px 80px; + background-image:url("../../icons/x32/state_info.png"); + background-repeat: no-repeat; + } + .systemmonitor-state-warning-icon { + padding:2px 10px 2px 80px; + background-image:url("../../icons/x32/state_warning.png"); + background-repeat: no-repeat; + } + .systemmonitor-state-critical-icon { + padding:2px 10px 2px 80px; + background-image:url("../../icons/x32/state_critical.png"); + background-repeat: no-repeat; + } + .systemmonitor-state-error-icon { + padding:2px 10px 2px 80px; + background-image:url("../../icons/x32/state_error.png"); + background-repeat: no-repeat; + } + + .systemmonitor table { border: 1px solid #d3d3d3; width: 80%; margin-top: 10px; diff --git a/interface/web/themes/default/icons/x32/state_critical.png b/interface/web/themes/default/icons/x32/state_critical.png new file mode 100644 index 0000000000000000000000000000000000000000..501424b341f4009e1a3f4b5bb5c0dcf8a5d34229 GIT binary patch literal 2409 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 byte)YGv3SnbZ5BL00000NkvXXu0mjf<&ua5 literal 0 HcmV?d00001 diff --git a/interface/web/themes/default/icons/x32/state_error.png b/interface/web/themes/default/icons/x32/state_error.png new file mode 100644 index 0000000000000000000000000000000000000000..b9a291171e35cfac828da05a7ba72458bdb00e52 GIT binary patch literal 2212 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 literal 0 HcmV?d00001 diff --git a/interface/web/themes/default/icons/x32/state_info.png b/interface/web/themes/default/icons/x32/state_info.png new file mode 100644 index 0000000000000000000000000000000000000000..f59130d14cdb347518b4fc9c83bc258bacc2fff9 GIT binary patch literal 2184 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 KMNUMnLSTX)CrQl! literal 0 HcmV?d00001 diff --git a/interface/web/themes/default/icons/x32/state_ok.png b/interface/web/themes/default/icons/x32/state_ok.png new file mode 100644 index 0000000000000000000000000000000000000000..ab86df2f5872b0126a27b09d5e711db515d4e301 GIT binary patch literal 1518 zcmV<K1rhp*P)<h;3K|Lk000e1NJLTq001BW001Be1^@s6b9#F8000HGNkl<ZScT=+ zeQXnT9LMp`@9uiP>$Tmw2iCFe=4e6OSa}o((1hRv0i6W?5DkW?(Fkfx{6~!vgNc6; zC5G@INVhRiL?9THASxn4Kv>z6o23QT0UKjX+O6B#Zg<!F`3C_(7#qlg3E$-R-~I0W z%01+AgfWKuIoy4%KQI8>XGFYjfD|uLdQae(Yen|Zdk(Ob5=A&IRc?BFll!3yBb!Gr zcTKow09zSRgeK}MU9=%P<Zr}d)28!$SUMuVNv=WM6@aab<ntlwD_vCXsHv>QUurwf zb+kaV@d6(ef2RxnLELeG%}W%%kye&2DtA;Lt3{U)MZ$<<U*lfXU8u(pm)(>m*$>mF z$+$ZXu&pBsAEG|*;xb2dO*Oh%6bZzz|J*M)qc?-g0FcFy%_*W0<GSfALT(7i-D=}< zx2E*_>fA8?IZ?PKlh6A`nWNfYg>E5=1dZcB=RTYfnjz)@mK+I|h73HM>nfJ4d{f^Z z#)N;LD0_?99IrAq8^p-Vf&5(S{CH@fcxo9@$QhH*`&OBws<9G1v<H2x7l+z^LyOP? zIg5u`kdgdJfJlVmgMkCV?p<mU0Lazfg0ynZ9BW`({7Zv#B+iVH%kurw!+c42q2APA ze8^@|(M}NG`o*P=>a)iXOU9tGJ{%1n#<@f@BuZfxWJI<n5O3`%CL#&;e6Q=5_g?|n zR}iT;gl`TmIWniGrxO;=2D?2S?q!93Td`1b@dfQVMDBL0(ibp%YhPRMsB5i3EE$7l zs5sVn6m7ldK&4CZ)-NOIYVLAFWP!vkreA#y!We_&6WN>|9=c`a>#OFdstUq5#Pk>h zo=3Jl2am1v`0ZollGbNqR}cj|)0E!h$;vgauFL+Ttqutz29>F(?W)2B^*ku0kT?k; zZyTcR9mO1nVEnsl;{ZUc9}$yy*1EO)-3@c8D1s0UomvMpra^}WffrygS@B}|0)NK1 zl#<Y#aMOt0SxVfm`&PcZ+Tp+0fP@i;1`YK+HE7o^K%x|sGvm*dO^C)LZeBznw(<Hn z0B}r$%z<<HnvdY)h9&UF{(wm|!QhifaynwH2a#kL6|a0<6s?bcH|giX1=d53x;3w^ zbp)fwq46r1iQ$CqN4L-gd#W9FOB#-Q&md}!xJ4)FHx2;2X(}|w(9GIaU-)U4%W2)& zwGBMaBhRY9;0)+QhelK+NDR-9U6{2m^y|D&7cQ6UyK12k6-hpclX?JA(gBNXfmO1i zYE1yq)QfJZfCQuO8~w(wc?d@?&G*<k(kn8jraI5(oW^j=Bbc7?I4pt{y(|Hp=n#ki zi(x|>-vR~(ooG0z1`sjApdy9DOE|Qq7TvapTYQKG&(BY2H;Z2b0D#<g(x&NQY;r!S z3^R=)FcUK-rA$K}RiF|TI&=`8Kw~O&q9LT8L?<E;g-h}M-b$$HF}JLkg29<>1IDid z0D%1i!K1dbjblsP!?MQNp(a&0=?Dy$MuO*fXhcJj=s0aOB8m=BfkNad_EsDQ$xFJ^ zvn;_QlLLc_-w*%*_T`+X?-Z$+@Z>nBYepUtdK{^|9R=cW$efI`tO;GD6I7ruaUyn? z?T1vzyK^!!gFhA>9dx{3003l`n>=wJDkjaE;2br3IQsNn2%HErXNJo4fpQcgD0Y<o zfK<2aF38IbZp;1Ymf|-B06_k?DbqUkX`3cKHGbIS*`uLB2L~Q9WKbqyW9c?zPt0(S zaOMZUux-8N_<#Ta81-|Wr)77f!ZUY@bLR9J000xCSXWk#;;BXMqC(fHC0N#bJ8=L2 zV{E`RrLxeI_MSB~Z%+x^+~;MFmX2Ys9e=5K2m6W+X8+CBpbwncFxiv+Vdm*ctH-lv z0#6igV5|OfoG~`=Ex0_hf>VfDmZx$==qSihLd!Yt?O*n*X3&7UH4nzG+5Z~wHz#We UggAPbj{pDw07*qoM6N<$f})Gi!vFvP literal 0 HcmV?d00001 diff --git a/interface/web/themes/default/icons/x32/state_unknown.png b/interface/web/themes/default/icons/x32/state_unknown.png new file mode 100644 index 0000000000000000000000000000000000000000..47694461fc97bcfb8f5d0b8fa900369a7b120d3c GIT binary patch literal 2832 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 literal 0 HcmV?d00001 diff --git a/interface/web/themes/default/icons/x32/state_warning.png b/interface/web/themes/default/icons/x32/state_warning.png new file mode 100644 index 0000000000000000000000000000000000000000..704cf4ba5ea90bbfcd9bd07528478840d2a475f4 GIT binary patch literal 1931 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 RP-6f9002ovPDHLkV1jFGhMxcc literal 0 HcmV?d00001 diff --git a/interface/web/themes/default/icons/x64/network.png b/interface/web/themes/default/icons/x64/network.png new file mode 100644 index 0000000000000000000000000000000000000000..f4c20744e5962e6fcf8b5d408aeb37394fdd828f GIT binary patch literal 7481 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 D685nK literal 0 HcmV?d00001 diff --git a/interface/web/themes/default/icons/x64/server.png b/interface/web/themes/default/icons/x64/server.png new file mode 100644 index 0000000000000000000000000000000000000000..22a95e989a4e50224bcde7bf4421947bffabe327 GIT binary patch literal 6880 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 a4*ws@D28v~zziS&0000<MNUMnLSTZ7CpvKe literal 0 HcmV?d00001 diff --git a/server/mods-available/monitor_core_module.inc.php b/server/mods-available/monitor_core_module.inc.php index dd5443b566..452d0c5e39 100644 --- a/server/mods-available/monitor_core_module.inc.php +++ b/server/mods-available/monitor_core_module.inc.php @@ -27,714 +27,852 @@ OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ - class monitor_core_module { - - /* TODO: this should be a config - var instead of a "constant" */ - var $interval = 5; // do the monitoring every 5 minutes - - var $module_name = 'monitor_core_module'; - var $class_name = 'monitor_core_module'; - /* No actions at this time. maybe later... */ - var $actions_available = array(); - - /* - This function is called when the module is loaded - */ - - function onLoad() { - global $app; - - /* - Annonce the actions that where provided by this module, so plugins - can register on them. - */ - /* none at them moment */ - //$app->plugins->announceEvents($this->module_name,$this->actions_available); - - /* - As we want to get notified of any changes on several database tables, - we register for them. - - The following function registers the function "functionname" - to be executed when a record for the table "dbtable" is - processed in the sys_datalog. "classname" is the name of the - class that contains the function functionname. - */ - /* none at them moment */ - //$app->modules->registerTableHook('mail_access','mail_module','process'); - - /* - Do the monitor every n minutes and write the result in the db - */ - $min = date('i'); - if (($min % $this->interval) == 0) - { - $this->doMonitor(); - } - } - - /* - This function is called when a change in one of the registered tables is detected. - The function then raises the events for the plugins. - */ - function process($tablename, $action, $data) { - // global $app; - // - // switch ($tablename) { - // case 'mail_access': - // if($action == 'i') $app->plugins->raiseEvent('mail_access_insert',$data); - // if($action == 'u') $app->plugins->raiseEvent('mail_access_update',$data); - // if($action == 'd') $app->plugins->raiseEvent('mail_access_delete',$data); - // break; - // } // end switch - } // end function - - /* - This method is called every n minutes, when the module ist loaded. - The method then does a system-monitoring - */ - // TODO: what monitoring is done should be a config-var - function doMonitor() - { - /* Calls the single Monitoring steps */ - $this->monitorServer(); - $this->monitorDiskUsage(); - $this->monitorMemUsage(); - $this->monitorCpu(); - $this->monitorServices(); - $this->monitorMailLog(); - $this->monitorMailWarnLog(); - $this->monitorMailErrLog(); - $this->monitorMessagesLog(); - $this->monitorFreshClamLog(); - $this->monitorClamAvLog(); - $this->monitorIspConfigLog(); - } - - function monitorServer(){ - global $app; - global $conf; - - /* the id of the server as int */ - $server_id = intval($conf["server_id"]); - - /** The type of the data */ - $type = 'server_load'; - - /* Delete Data older than 1 day */ - $this->_delOldRecords($type, 0, 0, 1); - - /* - Fetch the data into a array - */ - $procUptime = shell_exec("cat /proc/uptime | cut -f1 -d' '"); - $data['up_days'] = floor($procUptime/86400); - $data['up_hours'] = floor(($procUptime-$data['up_days']*86400)/3600); - $data['up_minutes'] = floor(($procUptime-$data['up_days']*86400-$data['up_hours']*3600)/60); - - $data['uptime'] = shell_exec("uptime"); - - $tmp = explode(",", $data['uptime'], 3); - $tmpUser = explode(" ", trim($tmp[1])); - $data['user_online'] = intval($tmpUser[0]); - - $loadTmp = explode(":" , trim($tmp[2])); - $load = explode(",", $loadTmp[1]); - $data['load_1'] = floatval(trim($load[0])); - $data['load_5'] = floatval(trim($load[1])); - $data['load_15'] = floatval(trim($load[2])); - - // Todo: the state should be calculated. For example if the load is to heavy, the state is warning... - $state = 'ok'; - - /* - Insert the data into the database - */ - $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . - "VALUES (". - $server_id . ", " . - "'" . $app->db->quote($type) . "', " . - time() . ", " . - "'" . $app->db->quote(serialize($data)) . "', " . - "'" . $state . "'" . - ")"; - $app->db->query($sql); - } - - function monitorDiskUsage() { - global $app; - global $conf; - - /* the id of the server as int */ - $server_id = intval($conf["server_id"]); - - /** The type of the data */ - $type = 'disk_usage'; - - /* Delete Data older than 10 minutes */ - $this->_delOldRecords($type, 10); - - /* - Fetch the data into a array - */ - $dfData = shell_exec("df"); - - // split into array - $df = explode("\n", $dfData); - // ignore the first line make a array of the rest - for($i=1; $i <= sizeof($df); $i++){ - if ($df[$i] != '') - { - $s = preg_split ("/[\s]+/", $df[$i]); - $data[$i]['fs'] = $s[0]; - $data[$i]['size'] = $s[1]; - $data[$i]['used'] = $s[2]; - $data[$i]['available'] = $s[3]; - $data[$i]['percent'] = $s[4]; - $data[$i]['mounted'] = $s[5]; - } - } - - // Todo: the state should be calculated. For example if the load is to heavy, the state is warning... - $state = 'ok'; - - /* - Insert the data into the database - */ - $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . - "VALUES (". - $server_id . ", " . - "'" . $app->db->quote($type) . "', " . - time() . ", " . - "'" . $app->db->quote(serialize($data)) . "', " . - "'" . $state . "'" . - ")"; - $app->db->query($sql); - } - - - function monitorMemUsage() - { - global $app; - global $conf; - - /* the id of the server as int */ - $server_id = intval($conf["server_id"]); - - /** The type of the data */ - $type = 'mem_usage'; - - /* Delete Data older than 10 minutes */ - $this->_delOldRecords($type, 10); - - /* - Fetch the data into a array - */ - $miData = shell_exec("cat /proc/meminfo"); - - $memInfo = explode("\n", $miData); - - foreach($memInfo as $line){ - $part = split(":", $line); - $key = trim($part[0]); - $tmp = explode(" ", trim($part[1])); - $value = 0; - if ($tmp[1] == 'kB') $value = $tmp[0] * 1024; - $data[$key] = $value; - } - - // Todo: the state should be calculated. For example if the load is to heavy, the state is warning... - $state = 'ok'; - - /* - Insert the data into the database - */ - $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . - "VALUES (". - $server_id . ", " . - "'" . $app->db->quote($type) . "', " . - time() . ", " . - "'" . $app->db->quote(serialize($data)) . "', " . - "'" . $state . "'" . - ")"; - $app->db->query($sql); - } - - - function monitorCpu() - { - global $app; - global $conf; - - /* the id of the server as int */ - $server_id = intval($conf["server_id"]); - - /** The type of the data */ - $type = 'cpu_info'; - - /* There is only ONE CPU-Data, so delete the old one */ - $this->_delOldRecords($type, 0); - - /* - Fetch the data into a array - */ - $cpuData = shell_exec("cat /proc/cpuinfo"); - $cpuInfo = explode("\n", $cpuData); - - foreach($cpuInfo as $line){ - $part = split(":", $line); - $key = trim($part[0]); - $value = trim($part[1]); - $data[$key] = $value; - } - - // Todo: the state should be calculated. For example if the load is to heavy, the state is warning... - $state = 'ok'; - - /* - Insert the data into the database - */ - $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . - "VALUES (". - $server_id . ", " . - "'" . $app->db->quote($type) . "', " . - time() . ", " . - "'" . $app->db->quote(serialize($data)) . "', " . - "'" . $state . "'" . - ")"; - $app->db->query($sql); - } - - - function monitorServices() - { - global $app; - global $conf; - - /* the id of the server as int */ - $server_id = intval($conf["server_id"]); - - /** The type of the data */ - $type = 'services'; - - /* There is only ONE Service-Data, so delete the old one */ - $this->_delOldRecords($type, 0); - - /* Monitor Webserver */ - if($this->_checkTcp('localhost',80)) { - $data['webserver'] = true; - } else { - $data['webserver'] = false; - } - - /* Monitor FTP-Server */ - if($this->_checkFtp('localhost',21)) { - $data['ftpserver'] = true; - } else { - $data['ftpserver'] = false; - } - - /* Monitor SMTP-Server */ - if($this->_checkTcp('localhost',25)) { - $data['smtpserver'] = true; - } else { - $data['smtpserver'] = false; - } - - /* Monitor POP3-Server */ - if($this->_checkTcp('localhost',110)) { - $data['pop3server'] = true; - } else { - $data['pop3server'] = false; - } - - /* Monitor BIND-Server */ - if($this->_checkTcp('localhost',53)) { - $data['bindserver'] = true; - } else { - $data['bindserver'] = false; - } - - /* Monitor MYSQL-Server */ - if($this->_checkTcp('localhost',3306)) { - $data['mysqlserver'] = true; - } else { - $data['mysqlserver'] = false; - } - - // Todo: the state should be calculated. For example if the load is to heavy, the state is warning... - $state = 'ok'; - - /* - Insert the data into the database - */ - $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . - "VALUES (". - $server_id . ", " . - "'" . $app->db->quote($type) . "', " . - time() . ", " . - "'" . $app->db->quote(serialize($data)) . "', " . - "'" . $state . "'" . - ")"; - $app->db->query($sql); - - } - - function monitorMailLog() - { - global $app; - global $conf; - - /* the id of the server as int */ - $server_id = intval($conf["server_id"]); - - /** The type of the data */ - $type = 'log_mail'; - - /* There is only ONE Log-Data, so delete the old one */ - $this->_delOldRecords($type, 0); - - - /* Get the data of the log */ - $data = $this->_getLogData($type); - - // Todo: the state should be calculated. For example if the load is to heavy, the state is warning... - $state = 'ok'; - - /* - Insert the data into the database - */ - $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . - "VALUES (". - $server_id . ", " . - "'" . $app->db->quote($type) . "', " . - time() . ", " . - "'" . $app->db->quote(serialize($data)) . "', " . - "'" . $state . "'" . - ")"; - $app->db->query($sql); - } - - function monitorMailWarnLog() - { - global $app; - global $conf; - - /* the id of the server as int */ - $server_id = intval($conf["server_id"]); - - /** The type of the data */ - $type = 'log_mail_warn'; - - /* There is only ONE Log-Data, so delete the old one */ - $this->_delOldRecords($type, 0); - - - /* Get the data of the log */ - $data = $this->_getLogData($type); - - // Todo: the state should be calculated. For example if the load is to heavy, the state is warning... - $state = 'ok'; - - /* - Insert the data into the database - */ - $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . - "VALUES (". - $server_id . ", " . - "'" . $app->db->quote($type) . "', " . - time() . ", " . - "'" . $app->db->quote(serialize($data)) . "', " . - "'" . $state . "'" . - ")"; - $app->db->query($sql); - } - - function monitorMailErrLog() - { - global $app; - global $conf; - - /* the id of the server as int */ - $server_id = intval($conf["server_id"]); - - /** The type of the data */ - $type = 'log_mail_err'; - - /* There is only ONE Log-Data, so delete the old one */ - $this->_delOldRecords($type, 0); - - - /* Get the data of the log */ - $data = $this->_getLogData($type); - - // Todo: the state should be calculated. For example if the load is to heavy, the state is warning... - $state = 'ok'; - - /* - Insert the data into the database - */ - $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . - "VALUES (". - $server_id . ", " . - "'" . $app->db->quote($type) . "', " . - time() . ", " . - "'" . $app->db->quote(serialize($data)) . "', " . - "'" . $state . "'" . - ")"; - $app->db->query($sql); - } - - - function monitorMessagesLog() - { - global $app; - global $conf; - - /* the id of the server as int */ - $server_id = intval($conf["server_id"]); - - /** The type of the data */ - $type = 'log_messages'; - - /* There is only ONE Log-Data, so delete the old one */ - $this->_delOldRecords($type, 0); - - - /* Get the data of the log */ - $data = $this->_getLogData($type); - - // Todo: the state should be calculated. For example if the load is to heavy, the state is warning... - $state = 'ok'; - - /* - Insert the data into the database - */ - $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . - "VALUES (". - $server_id . ", " . - "'" . $app->db->quote($type) . "', " . - time() . ", " . - "'" . $app->db->quote(serialize($data)) . "', " . - "'" . $state . "'" . - ")"; - $app->db->query($sql); - } - - function monitorFreshClamLog() - { - global $app; - global $conf; - - /* the id of the server as int */ - $server_id = intval($conf["server_id"]); - - /** The type of the data */ - $type = 'log_freshclam'; - - /* There is only ONE Log-Data, so delete the old one */ - $this->_delOldRecords($type, 0); - - - /* Get the data of the log */ - $data = $this->_getLogData($type); - - // Todo: the state should be calculated. For example if the load is to heavy, the state is warning... - $state = 'ok'; - - /* - Insert the data into the database - */ - $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . - "VALUES (". - $server_id . ", " . - "'" . $app->db->quote($type) . "', " . - time() . ", " . - "'" . $app->db->quote(serialize($data)) . "', " . - "'" . $state . "'" . - ")"; - $app->db->query($sql); - } - - function monitorClamAvLog() - { - global $app; - global $conf; - - /* the id of the server as int */ - $server_id = intval($conf["server_id"]); - - /** The type of the data */ - $type = 'log_clamav'; - - /* There is only ONE Log-Data, so delete the old one */ - $this->_delOldRecords($type, 0); - - - /* Get the data of the log */ - $data = $this->_getLogData($type); - - // Todo: the state should be calculated. For example if the load is to heavy, the state is warning... - $state = 'ok'; - - /* - Insert the data into the database - */ - $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . - "VALUES (". - $server_id . ", " . - "'" . $app->db->quote($type) . "', " . - time() . ", " . - "'" . $app->db->quote(serialize($data)) . "', " . - "'" . $state . "'" . - ")"; - $app->db->query($sql); - } - - function monitorIspConfigLog() - { - global $app; - global $conf; - - /* the id of the server as int */ - $server_id = intval($conf["server_id"]); - - /** The type of the data */ - $type = 'log_ispconfig'; - - /* There is only ONE Log-Data, so delete the old one */ - $this->_delOldRecords($type, 0); - - - /* Get the data of the log */ - $data = $this->_getLogData($type); - - // Todo: the state should be calculated. For example if the load is to heavy, the state is warning... - $state = 'ok'; - - /* - Insert the data into the database - */ - $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . - "VALUES (". - $server_id . ", " . - "'" . $app->db->quote($type) . "', " . - time() . ", " . - "'" . $app->db->quote(serialize($data)) . "', " . - "'" . $state . "'" . - ")"; - $app->db->query($sql); - } - - - function _getLogData($log){ - switch($log) { - case 'log_mail': - $logfile = '/var/log/mail.log'; - break; - case 'log_mail_warn': - $logfile = '/var/log/mail.warn'; - break; - case 'log_mail_err': - $logfile = '/var/log/mail.err'; - break; - case 'log_messages': - $logfile = '/var/log/messages'; - break; - case 'log_freshclam': - $logfile = '/var/log/clamav/freshclam.log'; - break; - case 'log_clamav': - $logfile = '/var/log/clamav/clamav.log'; - break; - case 'log_ispconfig': - $logfile = '/var/log/ispconfig/ispconfig.log'; - break; - default: - $logfile = ''; - break; - } - - // Getting the logfile content - if($logfile != '') { - $logfile = escapeshellcmd($logfile); - if(stristr($logfile,';')) die('Logfile path error.'); - - $log = ''; - if(is_readable($logfile)) { - if($fd = popen("tail -n 30 $logfile", 'r')) { - while (!feof($fd)) { - $log .= fgets($fd, 4096); - $n++; - if($n > 1000) break; - } - fclose($fd); - } - } else { - $log = 'Unable to read '.$logfile; - } - } - - return $log; - } - - function _checkTcp ($host,$port) { - - $fp = @fsockopen ($host, $port, &$errno, &$errstr, 2); - - if ($fp) { - return true; - fclose($fp); - } else { - return false; - fclose($fp); - } - } - - function _checkUdp ($host,$port) { - - $fp = @fsockopen ('udp://'.$host, $port, &$errno, &$errstr, 2); - - if ($fp) { - return true; - fclose($fp); - } else { - return false; - fclose($fp); - } - } - - function _checkFtp ($host,$port){ - - $conn_id = @ftp_connect($host, $port); - - if($conn_id){ - @ftp_close($conn_id); - return true; - } else { - @ftp_close($conn_id); - return false; - } - } - - /* - Deletes Records older than n. - */ - function _delOldRecords($type, $min, $hour=0, $days=0) { - global $app; - - $now = time(); - $old = $now - ($min * 60) - ($hour * 60 * 60) - ($days * 24 * 60 * 60); - $sql = "DELETE FROM monitor_data " . - "WHERE " . - "type =" . "'" . $app->db->quote($type) . "' " . - "AND " . - "created < " . $old; - $app->db->query($sql); - } - - + /* TODO: this should be a config - var instead of a "constant" */ + var $interval = 5; // do the monitoring every 5 minutes + + var $module_name = 'monitor_core_module'; + var $class_name = 'monitor_core_module'; + /* No actions at this time. maybe later... */ + var $actions_available = array(); + + /* + This function is called when the module is loaded + */ + function onLoad() { + global $app; + + /* + Annonce the actions that where provided by this module, so plugins + can register on them. + */ + /* none at them moment */ + //$app->plugins->announceEvents($this->module_name,$this->actions_available); + + /* + As we want to get notified of any changes on several database tables, + we register for them. + + The following function registers the function "functionname" + to be executed when a record for the table "dbtable" is + processed in the sys_datalog. "classname" is the name of the + class that contains the function functionname. + */ + /* none at them moment */ + //$app->modules->registerTableHook('mail_access','mail_module','process'); + + /* + Do the monitor every n minutes and write the result in the db + */ + $min = date('i'); + if (($min % $this->interval) == 0) + { + $this->doMonitor(); + } + } + + /* + This function is called when a change in one of the registered tables is detected. + The function then raises the events for the plugins. + */ + function process($tablename, $action, $data) { + // global $app; + // + // switch ($tablename) { + // case 'mail_access': + // if($action == 'i') $app->plugins->raiseEvent('mail_access_insert',$data); + // if($action == 'u') $app->plugins->raiseEvent('mail_access_update',$data); + // if($action == 'd') $app->plugins->raiseEvent('mail_access_delete',$data); + // break; + // } // end switch + } // end function + + /* + This method is called every n minutes, when the module ist loaded. + The method then does a system-monitoring + */ + // TODO: what monitoring is done should be a config-var + function doMonitor() + { + /* Calls the single Monitoring steps */ + $this->monitorServer(); + $this->monitorDiskUsage(); + $this->monitorMemUsage(); + $this->monitorCpu(); + $this->monitorServices(); + $this->monitorMailLog(); + $this->monitorMailWarnLog(); + $this->monitorMailErrLog(); + $this->monitorMessagesLog(); + $this->monitorFreshClamLog(); + $this->monitorClamAvLog(); + $this->monitorIspConfigLog(); + } + + function monitorServer(){ + global $app; + global $conf; + + /* the id of the server as int */ + $server_id = intval($conf["server_id"]); + + /** The type of the data */ + $type = 'server_load'; + + /* Delete Data older than 1 day */ + $this->_delOldRecords($type, 0, 0, 1); + + /* + Fetch the data into a array + */ + $procUptime = shell_exec("cat /proc/uptime | cut -f1 -d' '"); + $data['up_days'] = floor($procUptime/86400); + $data['up_hours'] = floor(($procUptime-$data['up_days']*86400)/3600); + $data['up_minutes'] = floor(($procUptime-$data['up_days']*86400-$data['up_hours']*3600)/60); + + $data['uptime'] = shell_exec("uptime"); + + $tmp = explode(",", $data['uptime'], 3); + $tmpUser = explode(" ", trim($tmp[1])); + $data['user_online'] = intval($tmpUser[0]); + + $loadTmp = explode(":" , trim($tmp[2])); + $load = explode(",", $loadTmp[1]); + $data['load_1'] = floatval(trim($load[0])); + $data['load_5'] = floatval(trim($load[1])); + $data['load_15'] = floatval(trim($load[2])); + + /** The state of the server-load. */ + $state = 'ok'; + if ($data['load_1'] > 20 ) $state = 'info'; + if ($data['load_1'] > 50 ) $state = 'warning'; + if ($data['load_1'] > 100 ) $state = 'critical'; + if ($data['load_1'] > 150 ) $state = 'error'; + + /* + Insert the data into the database + */ + $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . + "VALUES (". + $server_id . ", " . + "'" . $app->db->quote($type) . "', " . + time() . ", " . + "'" . $app->db->quote(serialize($data)) . "', " . + "'" . $state . "'" . + ")"; + $app->db->query($sql); + } + + function monitorDiskUsage() { + global $app; + global $conf; + + /* the id of the server as int */ + $server_id = intval($conf["server_id"]); + + /** The type of the data */ + $type = 'disk_usage'; + + /* Delete Data older than 10 minutes */ + $this->_delOldRecords($type, 10); + + /** The state of the disk-usage */ + $state = 'ok'; + + /** Fetch the data into a array */ + $dfData = shell_exec("df"); + + // split into array + $df = explode("\n", $dfData); + + /* + * ignore the first line, process the rest + */ + for($i=1; $i <= sizeof($df); $i++){ + if ($df[$i] != '') + { + /* + * Make a array of the data + */ + $s = preg_split ("/[\s]+/", $df[$i]); + $data[$i]['fs'] = $s[0]; + $data[$i]['size'] = $s[1]; + $data[$i]['used'] = $s[2]; + $data[$i]['available'] = $s[3]; + $data[$i]['percent'] = $s[4]; + $data[$i]['mounted'] = $s[5]; + /* + * calculate the state + */ + $usePercent = floatval($data[$i]['percent']); + if ($usePercent > 75) $state = $this->_setState($state, 'info'); + if ($usePercent > 80) $state = $this->_setState($state, 'warning'); + if ($usePercent > 90) $state = $this->_setState($state, 'critical'); + if ($usePercent > 95) $state = $this->_setState($state, 'error'); + } + } + + + /* + Insert the data into the database + */ + $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . + "VALUES (". + $server_id . ", " . + "'" . $app->db->quote($type) . "', " . + time() . ", " . + "'" . $app->db->quote(serialize($data)) . "', " . + "'" . $state . "'" . + ")"; + $app->db->query($sql); + } + + + function monitorMemUsage() + { + global $app; + global $conf; + + /* the id of the server as int */ + $server_id = intval($conf["server_id"]); + + /** The type of the data */ + $type = 'mem_usage'; + + /* Delete Data older than 10 minutes */ + $this->_delOldRecords($type, 10); + + /* + Fetch the data into a array + */ + $miData = shell_exec("cat /proc/meminfo"); + + $memInfo = explode("\n", $miData); + + foreach($memInfo as $line){ + $part = split(":", $line); + $key = trim($part[0]); + $tmp = explode(" ", trim($part[1])); + $value = 0; + if ($tmp[1] == 'kB') $value = $tmp[0] * 1024; + $data[$key] = $value; + } + + /* + * actually this info has no state. + * maybe someone knows better...???... + */ + $state = 'no_state'; + + /* + Insert the data into the database + */ + $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . + "VALUES (". + $server_id . ", " . + "'" . $app->db->quote($type) . "', " . + time() . ", " . + "'" . $app->db->quote(serialize($data)) . "', " . + "'" . $state . "'" . + ")"; + $app->db->query($sql); + } + + + function monitorCpu() + { + global $app; + global $conf; + + /* the id of the server as int */ + $server_id = intval($conf["server_id"]); + + /** The type of the data */ + $type = 'cpu_info'; + + /* There is only ONE CPU-Data, so delete the old one */ + $this->_delOldRecords($type, 0); + + /* + Fetch the data into a array + */ + $cpuData = shell_exec("cat /proc/cpuinfo"); + $cpuInfo = explode("\n", $cpuData); + + foreach($cpuInfo as $line){ + $part = split(":", $line); + $key = trim($part[0]); + $value = trim($part[1]); + $data[$key] = $value; + } + + /* the cpu has no state. It is, what it is */ + $state = 'no_state'; + + /* + Insert the data into the database + */ + $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . + "VALUES (". + $server_id . ", " . + "'" . $app->db->quote($type) . "', " . + time() . ", " . + "'" . $app->db->quote(serialize($data)) . "', " . + "'" . $state . "'" . + ")"; + $app->db->query($sql); + } + + + function monitorServices() + { + global $app; + global $conf; + + /** the id of the server as int */ + $server_id = intval($conf["server_id"]); + + /** get the "active" Services of the server from the DB */ + $services = $app->db->queryOneRecord("SELECT * FROM server WHERE server_id = " . $server_id); + + /* The type of the Monitor-data */ + $type = 'services'; + + /* There is only ONE Service-Data, so delete the old one */ + $this->_delOldRecords($type, 0); + + /** the State of the monitoring */ + /* ok, if ALL aktive services are running, + * error, if not + * There is no other state! + */ + $state = 'ok'; + + /* Monitor Webserver */ + $data['webserver'] = -1; // unknown - not needed + if ($services['web_server'] == 1) + { + if($this->_checkTcp('localhost', 80)) { + $data['webserver'] = 1; + } else { + $data['webserver'] = 0; + $state = 'error'; // because service is down + } + } + + /* Monitor FTP-Server */ + $data['ftpserver'] = -1; // unknown - not needed + if ($services['file_server'] == 1) + { + if($this->_checkFtp('localhost', 21)) { + $data['ftpserver'] = 1; + } else { + $data['ftpserver'] = 0; + $state = 'error'; // because service is down + } + } + + /* Monitor SMTP-Server */ + $data['smtpserver'] = -1; // unknown - not needed + if ($services['mail_server'] == 1) + { + if($this->_checkTcp('localhost', 25)) { + $data['smtpserver'] = 1; + } else { + $data['smtpserver'] = 0; + $state = 'error'; // because service is down + } + } + + /* Monitor POP3-Server */ + $data['pop3server'] = -1; // unknown - not needed + if ($services['mail_server'] == 1) + { + if($this->_checkTcp('localhost', 110)) { + $data['pop3server'] = 1; + } else { + $data['pop3server'] = 0; + $state = 'error'; // because service is down + } + } + + /* Monitor IMAP-Server */ + $data['imapserver'] = -1; // unknown - not needed + if ($services['mail_server'] == 1) + { + if($this->_checkTcp('localhost', 143)) { + $data['imapserver'] = 1; + } else { + $data['imapserver'] = 0; + $state = 'error'; // because service is down + } + } + + /* Monitor BIND-Server */ + $data['bindserver'] = -1; // unknown - not needed + if ($services['dns_server'] == 1) + { + if($this->_checkTcp('localhost', 53)) { + $data['bindserver'] = 1; + } else { + $data['bindserver'] = 0; + $state = 'error'; // because service is down + } + } + + /* Monitor MYSQL-Server */ + $data['mysqlserver'] = -1; // unknown - not needed + if ($services['db_server'] == 1) + { + if($this->_checkTcp('localhost', 3306)) { + $data['mysqlserver'] = 1; + } else { + $data['mysqlserver'] = 0; + $state = 'error'; // because service is down + } + } + + + /* + Insert the data into the database + */ + $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . + "VALUES (". + $server_id . ", " . + "'" . $app->db->quote($type) . "', " . + time() . ", " . + "'" . $app->db->quote(serialize($data)) . "', " . + "'" . $state . "'" . + ")"; + $app->db->query($sql); + + } + + function monitorMailLog() + { + global $app; + global $conf; + + /* the id of the server as int */ + $server_id = intval($conf["server_id"]); + + /** The type of the data */ + $type = 'log_mail'; + + /* There is only ONE Log-Data, so delete the old one */ + $this->_delOldRecords($type, 0); + + /* Get the data of the log */ + $data = $this->_getLogData($type); + + /* + * actually this info has no state. + * maybe someone knows better...???... + */ + $state = 'no_state'; + + /* + Insert the data into the database + */ + $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . + "VALUES (". + $server_id . ", " . + "'" . $app->db->quote($type) . "', " . + time() . ", " . + "'" . $app->db->quote(serialize($data)) . "', " . + "'" . $state . "'" . + ")"; + $app->db->query($sql); + } + + function monitorMailWarnLog() + { + global $app; + global $conf; + + /* the id of the server as int */ + $server_id = intval($conf["server_id"]); + + /** The type of the data */ + $type = 'log_mail_warn'; + + /* There is only ONE Log-Data, so delete the old one */ + $this->_delOldRecords($type, 0); + + + /* Get the data of the log */ + $data = $this->_getLogData($type); + + /* + * actually this info has no state. + * maybe someone knows better...???... + */ + $state = 'no_state'; + + /* + Insert the data into the database + */ + $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . + "VALUES (". + $server_id . ", " . + "'" . $app->db->quote($type) . "', " . + time() . ", " . + "'" . $app->db->quote(serialize($data)) . "', " . + "'" . $state . "'" . + ")"; + $app->db->query($sql); + } + + function monitorMailErrLog() + { + global $app; + global $conf; + + /* the id of the server as int */ + $server_id = intval($conf["server_id"]); + + /** The type of the data */ + $type = 'log_mail_err'; + + /* There is only ONE Log-Data, so delete the old one */ + $this->_delOldRecords($type, 0); + + + /* Get the data of the log */ + $data = $this->_getLogData($type); + + /* + * actually this info has no state. + * maybe someone knows better...???... + */ + $state = 'no_state'; + + /* + Insert the data into the database + */ + $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . + "VALUES (". + $server_id . ", " . + "'" . $app->db->quote($type) . "', " . + time() . ", " . + "'" . $app->db->quote(serialize($data)) . "', " . + "'" . $state . "'" . + ")"; + $app->db->query($sql); + } + + + function monitorMessagesLog() + { + global $app; + global $conf; + + /* the id of the server as int */ + $server_id = intval($conf["server_id"]); + + /** The type of the data */ + $type = 'log_messages'; + + /* There is only ONE Log-Data, so delete the old one */ + $this->_delOldRecords($type, 0); + + /* Get the data of the log */ + $data = $this->_getLogData($type); + + /* + * actually this info has no state. + * maybe someone knows better...???... + */ + $state = 'no_state'; + + /* + Insert the data into the database + */ + $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . + "VALUES (". + $server_id . ", " . + "'" . $app->db->quote($type) . "', " . + time() . ", " . + "'" . $app->db->quote(serialize($data)) . "', " . + "'" . $state . "'" . + ")"; + $app->db->query($sql); + } + + function monitorFreshClamLog() + { + global $app; + global $conf; + + /* the id of the server as int */ + $server_id = intval($conf["server_id"]); + + /** The type of the data */ + $type = 'log_freshclam'; + + /* There is only ONE Log-Data, so delete the old one */ + $this->_delOldRecords($type, 0); + + + /* Get the data of the log */ + $data = $this->_getLogData($type); + + // Todo: the state should be calculated. + $state = 'ok'; + + /* + Insert the data into the database + */ + $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . + "VALUES (". + $server_id . ", " . + "'" . $app->db->quote($type) . "', " . + time() . ", " . + "'" . $app->db->quote(serialize($data)) . "', " . + "'" . $state . "'" . + ")"; + $app->db->query($sql); + } + + function monitorClamAvLog() + { + global $app; + global $conf; + + /* the id of the server as int */ + $server_id = intval($conf["server_id"]); + + /** The type of the data */ + $type = 'log_clamav'; + + /* There is only ONE Log-Data, so delete the old one */ + $this->_delOldRecords($type, 0); + + /* Get the data of the log */ + $data = $this->_getLogData($type); + + // Todo: the state should be calculated. + $state = 'ok'; + + /* + Insert the data into the database + */ + $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . + "VALUES (". + $server_id . ", " . + "'" . $app->db->quote($type) . "', " . + time() . ", " . + "'" . $app->db->quote(serialize($data)) . "', " . + "'" . $state . "'" . + ")"; + $app->db->query($sql); + } + + function monitorIspConfigLog() + { + global $app; + global $conf; + + /* the id of the server as int */ + $server_id = intval($conf["server_id"]); + + /** The type of the data */ + $type = 'log_ispconfig'; + + /* There is only ONE Log-Data, so delete the old one */ + $this->_delOldRecords($type, 0); + + + /* Get the data of the log */ + $data = $this->_getLogData($type); + + // Todo: the state should be calculated. + $state = 'ok'; + + /* + Insert the data into the database + */ + $sql = "INSERT INTO monitor_data (server_id, type, created, data, state) " . + "VALUES (". + $server_id . ", " . + "'" . $app->db->quote($type) . "', " . + time() . ", " . + "'" . $app->db->quote(serialize($data)) . "', " . + "'" . $state . "'" . + ")"; + $app->db->query($sql); + } + + + function _getLogData($log){ + switch($log) { + case 'log_mail': + $logfile = '/var/log/mail.log'; + break; + case 'log_mail_warn': + $logfile = '/var/log/mail.warn'; + break; + case 'log_mail_err': + $logfile = '/var/log/mail.err'; + break; + case 'log_messages': + $logfile = '/var/log/messages'; + break; + case 'log_freshclam': + $logfile = '/var/log/clamav/freshclam.log'; + break; + case 'log_clamav': + $logfile = '/var/log/clamav/clamav.log'; + break; + case 'log_ispconfig': + $logfile = '/var/log/ispconfig/ispconfig.log'; + break; + default: + $logfile = ''; + break; + } + + // Getting the logfile content + if($logfile != '') { + $logfile = escapeshellcmd($logfile); + if(stristr($logfile, ';')) { + $log = 'Logfile path error.'; + } + else + { + $log = ''; + if(is_readable($logfile)) { + if($fd = popen("tail -n 100 $logfile", 'r')) { + while (!feof($fd)) { + $log .= fgets($fd, 4096); + $n++; + if($n > 1000) break; + } + fclose($fd); + } + } else { + $log = 'Unable to read '.$logfile; + } + } + } + + return $log; + } + + function _checkTcp ($host,$port) { + + $fp = @fsockopen ($host, $port, &$errno, &$errstr, 2); + + if ($fp) { + fclose($fp); + return true; + } else { + return false; + } + } + + function _checkUdp ($host,$port) { + + $fp = @fsockopen ('udp://'.$host, $port, &$errno, &$errstr, 2); + + if ($fp) { + fclose($fp); + return true; + } else { + return false; + } + } + + function _checkFtp ($host,$port){ + + $conn_id = @ftp_connect($host, $port); + + if($conn_id){ + @ftp_close($conn_id); + return true; + } else { + return false; + } + } + + /* + Deletes Records older than n. + */ + function _delOldRecords($type, $min, $hour=0, $days=0) { + global $app; + + $now = time(); + $old = $now - ($min * 60) - ($hour * 60 * 60) - ($days * 24 * 60 * 60); + $sql = "DELETE FROM monitor_data " . + "WHERE " . + "type =" . "'" . $app->db->quote($type) . "' " . + "AND " . + "created < " . $old; + $app->db->query($sql); + } + + /* + * Set the state to the given level (or higher, but not lesser). + * * If the actual state is critical and you call the method with ok, + * then the state is critical. + * + * * If the actual state is critical and you call the method with error, + * then the state is error. + */ + function _setState($oldState, $newState) + { + /* + * Calculate the weight of the old state + */ + switch ($oldState) { + case 'no_state': $oldInt = 0; + break; + case 'ok': $oldInt = 1; + break; + case 'unknown': $oldInt = 2; + break; + case 'info': $oldInt = 3; + break; + case 'warning': $oldInt = 4; + break; + case 'critical': $oldInt = 5; + break; + case 'error': $oldInt = 6; + break; + } + /* + * Calculate the weight of the new state + */ + switch ($newState) { + case 'no_state': $newInt = 0 ; + break; + case 'unknown': $newInt = 1 ; + break; + case 'ok': $newInt = 2 ; + break; + case 'info': $newInt = 3 ; + break; + case 'warning': $newInt = 4 ; + break; + case 'critical': $newInt = 5 ; + break; + case 'error': $newInt = 6 ; + break; + } + + /* + * Set to the higher level + */ + if ($newInt > $oldInt){ + return $newState; + } + else + { + return $oldState; + } + } + + } // end class ?> \ No newline at end of file -- GitLab