Chinaunix首页 | 论坛 | 博客
  • 博客访问: 432707
  • 博文数量: 75
  • 博客积分: 2780
  • 博客等级: 少校
  • 技术积分: 789
  • 用 户 组: 普通用户
  • 注册时间: 2008-12-23 20:38
文章分类

全部博文(75)

文章存档

2012年(1)

2011年(11)

2010年(12)

2009年(39)

2008年(12)

我的朋友

分类: LINUX

2008-12-30 07:41:36


  我管理服务器太多了,不可能每时每刻都登录到上面查看服务器状况,于是写了一个“服务器监控系统 ServMon V1.0”。其原理是用shell脚本定时抓取被监控服务器上信息,再用curl POST到监控机的PHP接口程序上,由PHP接口程序对信息进行分析,如果存在异常,则发送报警邮件给管理员。

  下午,ServMon V1.0 已经在sina播客服务器上部署,以下为ServMon的删节版内容,只监控磁盘分区使用率和Swap交换空间两项:

  

  1、被监控端服务器的shell脚本(Linux)
  (1)、发送磁盘分区使用率给监控端接口的脚本
引用
/usr/local/servmon/bin/diskstat_send
#!/bin/sh
LANG=zh_cn
disk=$(/bin/df -kP | grep / | awk -F ' ' ' {print $6":"$5}' | awk -F '%' ' {print $1}' | tr -s '\n' ';')
today=$(date -d "today" +"%Y-%m-%d_%H:%M:%S")
ip=$(/sbin/ifconfig | grep "inet addr" | grep -v "127.0.0.1" | awk '{print $2;}' | awk -F':' '{print $2;}' | tr -s '\n' ';')
/usr/bin/curl -d menu=diskstat -d password=$2 -d date=$today -d ip=$ip -d data=$disk $1

  磁盘分区使用率监控报警邮件示例图:
  


  (2)、发送Swap交换空间使用率给监控端接口的脚本
  /usr/local/servmon/bin/swapstat_send
引用
#!/bin/sh
LANG=zh_cn
data=$(/usr/bin/free -m | grep Swap | awk '{print $2":"$3":"$4}')
today=$(date -d "today" +"%Y-%m-%d_%H:%M:%S")
ip=$(/sbin/ifconfig | grep "inet addr" | grep -v "127.0.0.1" | awk '{print $2;}' | awk -F':' '{print $2;}' | tr -s '\n' ';')
/usr/bin/curl -d menu=swapstat -d password=$2 -d date=$today -d ip=$ip -d data=$data $1

  Swap交换空间使用率监控报警邮件示例图:
  

  (3)、计划任务:每30分钟将磁盘分区使用率发送给监控端接口,每10分钟将Swap交换空间使用率发送给监控端接口。
  /etc/cron.d/servmon
引用
0-30/30 * * * * root /bin/sh /usr/local/servmon/bin/diskstat_send  yourpassword
0-50/10 * * * * root /bin/sh /usr/local/servmon/bin/swapstat_send  yourpassword





  2、监控端服务器配置(Linux)
  (1)、启动sendmail
  /usr/sbin/sendmail -bd -q30m
  -b:指定Sendmail在后台运行,并且监听端口25的请求。
  -d:指定Sendmail以Daemon方式运行(守护进程)。
  -q:当Sendmail无法将邮件成功地发送到目的地时,它会将邮件保存在队列里。该参数指定邮件在队列里保存的时间。其中的30m表示保留30分钟。

  (2)、修改php.ini的[mail function]部分为以下内容,然后重启Apache,使PHP能够使用mail()函数发送电子邮件
引用
[mail function]
; For Win32 only.
;SMTP = localhost
;smtp_port = 25

; For Win32 only.
;sendmail_from = me@example.com

; For Unix only.  You may supply arguments as well (default: "sendmail -t -i").
sendmail_path = /usr/sbin/sendmail -t -i


  (4)、在MySQL上创建一个数据库servmon,然后在该数据库中执行以下SQL语句创建表:
引用
CREATE TABLE `tb_diskstat` (
 `ip` varchar(128) NOT NULL default '',
 `date` int(10) NOT NULL default '0',
 UNIQUE KEY `ip` (`ip`)
) TYPE=MyISAM;

  备注:这张表是PHP接口程序用来记录发送磁盘分区报警邮件的时间,避免在一段时间内重复发送大量报警邮件的,其实完全可以用Session、文本文件来代替它。这里之所以用MySQL数据库,是因为PHP接口中还将增加一些需要记录信息到数据库的监控模块,所以就统一用数据库了。)

  (3)、PHP接口程序
  
  1. /** 
  2. ********************************************************************* 
  3. * Copyright (c) 2007, 回忆未来[张宴] 
  4. * All rights reserved. 
  5. * 
  6. * 文件名称: interface.php 
  7. * 摘    要: 服务器监控系统ServMon接口程序 
  8. * 描    述: 
  9. * 作    者: 张宴  新浪UC:103500  博客:blog.s135.com 
  10. * 版    本: 1.0 
  11. * 时    间: 2007-08-03 
  12. ********************************************************************* 
  13. */  
  14.   
  15. //------------------配置信息------------------  
  16. $password = "yourpassword";//设置数据传输验证密码  
  17. $mysql_server = "127.0.0.1:3306";//MySQL服务器地址及端口,例如localhost:3306  
  18. $mysql_db = "servmon";//MySQL数据库名  
  19. $mysql_username = "root";//MySQL帐号  
  20. $mysql_password = "";//MySQL密码  
  21. $disk_max = "90";//磁盘分区使用百分比超过此值,则报警。默认为90%  
  22. $disk_sendmail_interval = "43200";//如果同一台服务器连续报警,设置每隔多少秒发送一次报警邮件。默认为43200秒,即12小时。  
  23. $swap_max = "50";//Swap交换空间使用百分比超过此值,则报警。默认为50%  
  24. $email = "servmon@sina.com";//管理员邮箱  
  25. //--------------------------------------------  
  26.   
  27. //密码验证  
  28. if (htmlspecialchars($_POST["password"]) != $password)  
  29. {  
  30.     echo "数据传输验证密码错误!\n";  
  31.     exit();  
  32. }  
  33.   
  34. //磁盘分区报警(分区使用率超过90%后会发邮件报警)  
  35. if (htmlspecialchars($_POST["menu"]) == "diskstat")  
  36. {  
  37.     $is_send_mail = "0";//是否发邮件,0为不发邮件  
  38.     $date = htmlspecialchars($_POST["date"]);  
  39.     $ip = htmlspecialchars($_POST["ip"]);  
  40.     $data = htmlspecialchars($_POST["data"]);  
  41.     $subject = "[系统监控]_磁盘分区空间不足_" . $ip;  
  42.     $message = "服务器监控系统 ServMon V1.0\r\n";  
  43.     $message = $message . "-------------------------------------------------------\r\n";  
  44.     $message = $message . "报警服务器:" . $ip . "\r\n";  
  45.     $message = $message . "-------------------------------------------------------\r\n";  
  46.     $message = $message . "报警时间:" . $date . "\r\n";  
  47.     $message = $message . "-------------------------------------------------------\r\n";  
  48.     $message = $message . "报警内容:\r\n";  
  49.     $headers = 'From: ServMon ' . "\r\n" .  
  50.         'Reply-To: ServMon ' . "\r\n" . 'X-Mailer: PHP/' . phpversion();  
  51.     $arr1 = explode(";"$data);  
  52.     foreach ($arr1 as $key1 => $value1)  
  53.     {  
  54.         $arr2 = explode(":"$value1);  
  55.         if ($arr2[0] != "" && $arr2[1] != "" && $arr2[1] >= $disk_max)  
  56.         {  
  57.             $message = $message . "●磁盘分区 " . $arr2[0] . " 空间不足(空间使用率达到" . $arr2[1] . "%)\r\n";  
  58.             $is_send_mail = "1";//是否发邮件,1为发送邮件  
  59.         }  
  60.     }  
  61.     if ($is_send_mail == "1")  
  62.     {  
  63.         $connect = mysql_connect($mysql_server$mysql_username$mysql_password);  
  64.         $select = mysql_select_db($mysql_db$connect);  
  65.         $select_result = mysql_query("SELECT date FROM tb_diskstat WHERE ip='" . $ip .  
  66.             "' LIMIT 1");  
  67.         $rs = mysql_fetch_array($select_result);  
  68.         $interval = date("U") - $rs["date"];  
  69.         if ($interval >= $disk_sendmail_interval)  
  70.         {  
  71.             if (mail($email$subject$message$headers))  
  72.             {  
  73.                 echo "已成功发送报警邮件!\n";  
  74.             }  
  75.             else  
  76.             {  
  77.                 echo "发送报警邮件失败!\n";  
  78.             }  
  79.             $insert_result = mysql_query("REPLACE INTO tb_diskstat (ip, date) VALUES ('" . $ip .  
  80.                 "', '" . date("U") . "');");  
  81.         }  
  82.         else  
  83.         {  
  84.             if ($disk_sendmail_interval >= 3600)  
  85.             {  
  86.                 $disk_sendmail_interval_text = $disk_sendmail_interval / 3600 . "小时";  
  87.             }  
  88.             else  
  89.             {  
  90.                 $disk_sendmail_interval_text = $disk_sendmail_interval . "秒";  
  91.             }  
  92.             echo "已发送过报警邮件," . $disk_sendmail_interval_text . "内不再重复发送!\n";  
  93.         }  
  94.   
  95.     }  
  96.     else  
  97.     {  
  98.         echo "磁盘分区空间足够,无须发送报警邮件!\n";  
  99.     }  
  100.     exit();  
  101. }  
  102.   
  103. //Swap交换空间报警(Swap交换空间使用率超过60%后会发邮件报警)  
  104. if (htmlspecialchars($_POST["menu"]) == "swapstat")  
  105. {  
  106.     $is_send_mail = "0";//是否发邮件,0为不发邮件  
  107.     $date = htmlspecialchars($_POST["date"]);  
  108.     $ip = htmlspecialchars($_POST["ip"]);  
  109.     list($total$used$free) = split(":", htmlspecialchars($_POST["data"]));  
  110.     $percentage = round($used / $total * 100, 0);  
  111.     $subject = "[系统监控]_Swap交换空间使用率高达" . $percentage . "%_" . $ip;  
  112.     $message = "服务器监控系统 ServMon V1.0\r\n";  
  113.     $message = $message . "-------------------------------------------------------\r\n";  
  114.     $message = $message . "报警服务器:" . $ip . "\r\n";  
  115.     $message = $message . "-------------------------------------------------------\r\n";  
  116.     $message = $message . "报警时间:" . $date . "\r\n";  
  117.     $message = $message . "-------------------------------------------------------\r\n";  
  118.     $message = $message . "报警内容:\r\n";  
  119.     $headers = 'From: ServMon ' . "\r\n" .  
  120.         'Reply-To: ServMon ' . "\r\n" . 'X-Mailer: PHP/' . phpversion();  
  121.     $arr1 = explode(";"$data);  
  122.     foreach ($arr1 as $key1 => $value1)  
  123.     {  
  124.         $arr2 = explode(":"$value1);  
  125.         if ($arr2[0] != "" && $arr2[1] != "" && $arr2[1] >= $disk_max)  
  126.         {  
  127.             $message = $message . "●磁盘分区 " . $arr2[0] . " 空间不足(空间使用率达到" . $arr2[1] . "%)\r\n";  
  128.             $is_send_mail = "1";//是否发邮件,1为发送邮件  
  129.         }  
  130.     }  
  131.     if ($percentage >= $swap_max)  
  132.     {  
  133.         $message = $message . "●Swap交换空间使用率高达" . $percentage . "%,若继续增加到100%可能会导致服务器死机。\r\n";  
  134.         $message = $message . "●Swap交换空间总量:" . $total . " MB\r\n";  
  135.         $message = $message . "●Swap交换空间使用:" . $used . " MB\r\n";  
  136.         $message = $message . "●Swap交换空间剩余:" . $free . " MB\r\n";  
  137.         $is_send_mail = "1";//是否发邮件,1为发送邮件  
  138.     }  
  139.     if ($is_send_mail == "1")  
  140.     {  
  141.         if (mail($email$subject$message$headers))  
  142.         {  
  143.             echo "已成功发送报警邮件!\n";  
  144.         }  
  145.         else  
  146.         {  
  147.             echo "发送报警邮件失败!\n";  
  148.         }  
  149.     }  
  150.     else  
  151.     {  
  152.         echo "Swap交换空间使用率不高,无须发送报警邮件!\n";  
  153.     }  
  154.     exit();  
  155. }  
  156. ?>  
阅读(1388) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~