侯体宗的博客
  • 首页
  • 人生(杂谈)
  • 技术
  • 关于我
  • 更多分类
    • 文件下载
    • 文字修仙
    • 中国象棋ai
    • 群聊
    • 九宫格抽奖
    • 拼图
    • 消消乐
    • 相册

脚本实现SSH登录邮件报警

技术  /  管理员 发布于 8年前   191

登录保护是一个非常重要的环节,下面通过图文并茂的方式给大家详细讲解下:


前两天@cyy 给我发了一个图

然后我就想到USHQ的ssh登录app通知功能,然后就像如果把这个部署到自用的服务器就好了。至少多一层安全系数。

首先要感谢@Legion 帮忙搞定了几个错误以及搞定了Geo2IP的JSON转换。 (P.S.此人为自动化运维大神级人物,现任职于德国一数据统计企业。)

当然,我和他相比我就是战五渣了...大家一定要多向 @Legion 学习啊~~

说下需要做的准备:

sendmail或者Postfix
php
bash
CentOS/Debian/Ubuntu
若你的生产环境中没有php sendmail Postfix等组件,请移步:

@Legion: Linux之使用shell脚本实现ssh登录报警

参考文件

首先是报警脚本文件

Shell

#!/bin/sh########################################################################## File Name: Login-alert.sh# Author: Jason# Email: master#deamwork.com# Created Time: Tue Jul 21 2015 21:23:16 PM CST ##########################################################################require jq#wget http://stedolan.github.io/jq/download/linux64/jq -O /usr/local/bin/jq#chmod +x /usr/local/bin/jq#if error, please # following one#PATH=/usr/local/nginx/sbin:/usr/local/php/bin:/usr/local/mysql/bin:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/root/bin#Geo2IP by Legion(http://www.dwhd.org/)eval `curl -s "http://ip.taobao.com/service/getIpInfo.php?ip=${SSH_CLIENT%% *}" | jq . | awk -F':|[ ]+|"' '{if($3~/^(country|area|region|city|isp)$/){print $3"="$7}}'`#html mail contentcat >> mail-no-base64.html <<EOF#请自行准备邮件模板,以下为可能用到的变量#输出主机名 `hostname`#输出登录端口 ${SSH_CLIENT##* }#输出登录来源IP ${SSH_CLIENT%% *}#输出IP地址归属地 {country}_${area}_${region}_${city}_${isp}#输出登录时间 `date`EOF#Base64 Encodingbase64 mail-no-base64.html > mail-base64.html#使用Sendmail#sendmail -t >/dev/null 2>&1 <<EOF#to:[email protected]#from:Example<[email protected]>#subject:[`hostname`]服务器登录告警#`cat mail-no-base64.html`#EOF#使用postfix#cat >> mail.php <<EOF#<?php#\$to = "[email protected]";#\$subject = "[`hostname`]服务器登录告警";#\$message = "`cat mail-base64.html`";#\$headers = "MIME-Version: 1.0" . "\r\n";#\$headers .= "Content-Type: text/html; charset=\"utf-8\"" . "\r\n";#\$headers .= "Content-Transfer-Encoding: base64" . "\r\n";#\$headers .= 'From: Example<[email protected]>' . "\r\n";#\$send = mail(\$to,\$subject,\$message,\$headers);#if(\$send){echo 'Mail Send Successful.';}else{echo 'Failed.';}#?>#EOF#使用 SMTP (require smtp-class.php)cat >> mail.php <<EOF<?phprequire("smtp-class.php"); \$smtpserver = "smtp.example.com";\$smtpserverport = 25;\$smtpusermail = "[email protected]";\$smtpemailto = "[email protected]";\$smtpuser = "example";\$smtppass = "password";\$mailsubject = "[`hostname`]服务器登录告警";\$mailbody = "`cat mail-base64.html`";\$mailtype = "HTML";\$smtp = new smtp(\$smtpserver,\$smtpserverport,true,\$smtpuser,\$smtppass);\$smtp->debug = TRUE;\$smtp->sendmail(\$smtpemailto, \$smtpusermail, \$mailsubject, \$mailbody, \$mailtype); ?>EOFphp mail.phpyes y | rm mail-no-base64.html mail-base64.html mail.php

然后是如何触发这个脚本:

Shell

复制代码 代码如下:
echo "screen -fa -d -m -S WL /etc/Login-alert.sh" >> /etc/profile

用这种方法, 新开终端或者复制终端都会触发报警

如果使用smtp方式,请保存以下文件为smtp-class.php

PHP

<?phpclass smtp{  /* Public Variables */  var $smtp_port;  var $time_out;  var $host_name;  var $log_file;  var $relay_host;  var $debug;  var $auth;  var $user;  var $pass;  /* Private Variables */   var $sock;  /* Constractor */  function smtp($relay_host = "", $smtp_port = 25,$auth = false,$user,$pass)  {    $this->debug = FALSE;    $this->smtp_port = $smtp_port;    $this->relay_host = $relay_host;    $this->time_out = 30; //is used in fsockopen()     $this->auth = $auth;//auth    $this->user = $user;    $this->pass = $pass;    $this->host_name = "localhost"; //is used in HELO command     $this->log_file = "";    $this->sock = FALSE;}  /* Main Function */  function sendmail($to, $from, $subject = "", $body = "", $mailtype, $cc = "", $bcc = "", $additional_headers = "")  {    $mail_from = $this->get_address($this->strip_comment($from));    $body = ereg_replace("(^|(\r\n))(\.)", "\1.\3", $body);    $header .= "MIME-Version:1.0\r\n";    if($mailtype=="HTML")    {      $header .= "Content-Type: text/html; charset=\"utf-8\"" . "\r\n";      $header .= "Content-Transfer-Encoding: base64" . "\r\n";    }    $header .= "To: ".$to."\r\n";    if ($cc != "")     {      $header .= "Cc: ".$cc."\r\n";    }    $header .= "From: $from<".$from.">\r\n";    $header .= "Subject: ".$subject."\r\n";    $header .= $additional_headers;    $header .= "Date: ".date("r")."\r\n";    $header .= "X-Mailer:By TianhaiTech (PHP/".phpversion().")\r\n";    list($msec, $sec) = explode(" ", microtime());    $header .= "Message-ID: <".date("YmdHis", $sec).".".($msec*1000000).".".$mail_from.">\r\n";    $TO = explode(",", $this->strip_comment($to));    if ($cc != "")     {      $TO = array_merge($TO, explode(",", $this->strip_comment($cc)));      }    if ($bcc != "")     {      $TO = array_merge($TO, explode(",", $this->strip_comment($bcc)));    }    $sent = TRUE;    foreach ($TO as $rcpt_to)     {      $rcpt_to = $this->get_address($rcpt_to);      if (!$this->smtp_sockopen($rcpt_to))       {        $this->log_write("Error: Cannot send email to ".$rcpt_to."\n");        $sent = FALSE;        continue;      }      if ($this->smtp_send($this->host_name, $mail_from, $rcpt_to, $header, $body))       {        $this->log_write("E-mail has been sent to <".$rcpt_to.">\n");      }       else       {        $this->log_write("Error: Cannot send email to <".$rcpt_to.">\n");        $sent = FALSE;      }      fclose($this->sock);      $this->log_write("Disconnected from remote host\n");    }    return $sent;  }  /* Private Functions */  function smtp_send($helo, $from, $to, $header, $body = "")  {    if (!$this->smtp_putcmd("HELO", $helo))     {      return $this->smtp_error("sending HELO command");    }    #auth    if($this->auth)    {      if (!$this->smtp_putcmd("AUTH LOGIN", base64_encode($this->user)))       {        return $this->smtp_error("sending HELO command");      }      if (!$this->smtp_putcmd("", base64_encode($this->pass)))       {        return $this->smtp_error("sending HELO command");      }    }    if (!$this->smtp_putcmd("MAIL", "FROM:<".$from.">"))     {      return $this->smtp_error("sending MAIL FROM command");    }    if (!$this->smtp_putcmd("RCPT", "TO:<".$to.">"))     {      return $this->smtp_error("sending RCPT TO command");    }    if (!$this->smtp_putcmd("DATA"))    {      return $this->smtp_error("sending DATA command");    }    if (!$this->smtp_message($header, $body))     {      return $this->smtp_error("sending message");    }    if (!$this->smtp_eom())    {      return $this->smtp_error("sending <CR><LF>.<CR><LF> [EOM]");    }    if (!$this->smtp_putcmd("QUIT"))     {      return $this->smtp_error("sending QUIT command");    }    return TRUE;  }  function smtp_sockopen($address)  {    if ($this->relay_host == "")     {      return $this->smtp_sockopen_mx($address);    }     else    {      return $this->smtp_sockopen_relay();    }  }  function smtp_sockopen_relay()  {    $this->log_write("Trying to ".$this->relay_host.":".$this->smtp_port."\n");    $this->sock = @fsockopen($this->relay_host, $this->smtp_port, $errno, $errstr, $this->time_out);    if (!($this->sock && $this->smtp_ok()))     {      $this->log_write("Error: Cannot connenct to relay host ".$this->relay_host."\n");      $this->log_write("Error: ".$errstr." (".$errno.")\n");      return FALSE;    }    $this->log_write("Connected to relay host ".$this->relay_host."\n");    return TRUE;;  }  function smtp_sockopen_mx($address)  {    $domain = ereg_replace("^.+@([^@]+)$", "\1", $address);    if (!@getmxrr($domain, $MXHOSTS))     {      $this->log_write("Error: Cannot resolve MX \"".$domain."\"\n");      return FALSE;    }    foreach ($MXHOSTS as $host)     {      $this->log_write("Trying to ".$host.":".$this->smtp_port."\n");      $this->sock = @fsockopen($host, $this->smtp_port, $errno, $errstr, $this->time_out);      if (!($this->sock && $this->smtp_ok()))       {        $this->log_write("Warning: Cannot connect to mx host ".$host."\n");        $this->log_write("Error: ".$errstr." (".$errno.")\n");        continue;      }      $this->log_write("Connected to mx host ".$host."\n");      return TRUE;    }    $this->log_write("Error: Cannot connect to any mx hosts (".implode(", ", $MXHOSTS).")\n");    return FALSE;  }  function smtp_message($header, $body)  {    fputs($this->sock, $header."\r\n".$body);    $this->smtp_debug("> ".str_replace("\r\n", "\n"."> ", $header."\n> ".$body."\n> "));    return TRUE;  }  function smtp_eom()  {    fputs($this->sock, "\r\n.\r\n");    $this->smtp_debug(". [EOM]\n");    return $this->smtp_ok();  }  function smtp_ok()  {    $response = str_replace("\r\n", "", fgets($this->sock, 512));    $this->smtp_debug($response."\n");    if (!ereg("^[23]", $response))     {      fputs($this->sock, "QUIT\r\n");      fgets($this->sock, 512);      $this->log_write("Error: Remote host returned \"".$response."\"\n");      return FALSE;    }    return TRUE;  }  function smtp_putcmd($cmd, $arg = "")  {    if ($arg != "")     {      if($cmd=="")       {        $cmd = $arg;      }      else      {        $cmd = $cmd." ".$arg;      }    }    fputs($this->sock, $cmd."\r\n");    $this->smtp_debug("> ".$cmd."\n");    return $this->smtp_ok();  }  function smtp_error($string)  {    $this->log_write("Error: Error occurred while ".$string.".\n");    return FALSE;  }  function log_write($message)  {    $this->smtp_debug($message);    if ($this->log_file == "")    {      return TRUE;    }    $message = date("M d H:i:s ").get_current_user()."[".getmypid()."]: ".$message;    if (!@file_exists($this->log_file) || !($fp = @fopen($this->log_file, "a")))     {      $this->smtp_debug("Warning: Cannot open log file \"".$this->log_file."\"\n");      return FALSE;;    }    flock($fp, LOCK_EX);    fputs($fp, $message);    fclose($fp);    return TRUE;  }  function strip_comment($address)  {    $comment = "\([^()]*\)";    while (ereg($comment, $address))     {      $address = ereg_replace($comment, "", $address);    }    return $address;  }  function get_address($address)  {    $address = ereg_replace("([ \t\r\n])+", "", $address);    $address = ereg_replace("^.*<(.+)>.*$", "\1", $address);    return $address;  }  function smtp_debug($message)  {    if ($this->debug)     {      echo $message;    }  }}

?>
实现效果:

有需要的朋友可以参考下,希望大家能够喜欢。


  • 上一条:
    shell脚本中一些特殊符号
    下一条:
    Shell中eval的用法示例
  • 昵称:

    邮箱:

    0条评论 (评论内容有缓存机制,请悉知!)
    最新最热
    • 分类目录
    • 人生(杂谈)
    • 技术
    • linux
    • Java
    • php
    • 框架(架构)
    • 前端
    • ThinkPHP
    • 数据库
    • 微信(小程序)
    • Laravel
    • Redis
    • Docker
    • Go
    • swoole
    • Windows
    • Python
    • 苹果(mac/ios)
    • 相关文章
    • 智能合约Solidity学习CryptoZombie第四课:僵尸作战系统(0个评论)
    • 智能合约Solidity学习CryptoZombie第三课:组建僵尸军队(高级Solidity理论)(0个评论)
    • 智能合约Solidity学习CryptoZombie第二课:让你的僵尸猎食(0个评论)
    • 智能合约Solidity学习CryptoZombie第一课:生成一只你的僵尸(0个评论)
    • gmail发邮件报错:534 5.7.9 Application-specific password required...解决方案(0个评论)
    • 近期文章
    • 在go语言中实现字符串可逆性压缩及解压缩功能(0个评论)
    • 使用go + gin + jwt + qrcode实现网站生成登录二维码在app中扫码登录功能(0个评论)
    • 在windows10中升级go版本至1.24后LiteIDE的Ctrl+左击无法跳转问题解决方案(0个评论)
    • 智能合约Solidity学习CryptoZombie第四课:僵尸作战系统(0个评论)
    • 智能合约Solidity学习CryptoZombie第三课:组建僵尸军队(高级Solidity理论)(0个评论)
    • 智能合约Solidity学习CryptoZombie第二课:让你的僵尸猎食(0个评论)
    • 智能合约Solidity学习CryptoZombie第一课:生成一只你的僵尸(0个评论)
    • 在go中实现一个常用的先进先出的缓存淘汰算法示例代码(0个评论)
    • 在go+gin中使用"github.com/skip2/go-qrcode"实现url转二维码功能(0个评论)
    • 在go语言中使用api.geonames.org接口实现根据国际邮政编码获取地址信息功能(1个评论)
    • 近期评论
    • 122 在

      学历:一种延缓就业设计,生活需求下的权衡之选中评论 工作几年后,报名考研了,到现在还没认真学习备考,迷茫中。作为一名北漂互联网打工人..
    • 123 在

      Clash for Windows作者删库跑路了,github已404中评论 按理说只要你在国内,所有的流量进出都在监控范围内,不管你怎么隐藏也没用,想搞你分..
    • 原梓番博客 在

      在Laravel框架中使用模型Model分表最简单的方法中评论 好久好久都没看友情链接申请了,今天刚看,已经添加。..
    • 博主 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 @1111老铁这个不行了,可以看看近期评论的其他文章..
    • 1111 在

      佛跳墙vpn软件不会用?上不了网?佛跳墙vpn常见问题以及解决办法中评论 网站不能打开,博主百忙中能否发个APP下载链接,佛跳墙或极光..
    • 2016-10
    • 2016-11
    • 2017-07
    • 2017-08
    • 2017-09
    • 2018-01
    • 2018-07
    • 2018-08
    • 2018-09
    • 2018-12
    • 2019-01
    • 2019-02
    • 2019-03
    • 2019-04
    • 2019-05
    • 2019-06
    • 2019-07
    • 2019-08
    • 2019-09
    • 2019-10
    • 2019-11
    • 2019-12
    • 2020-01
    • 2020-03
    • 2020-04
    • 2020-05
    • 2020-06
    • 2020-07
    • 2020-08
    • 2020-09
    • 2020-10
    • 2020-11
    • 2021-04
    • 2021-05
    • 2021-06
    • 2021-07
    • 2021-08
    • 2021-09
    • 2021-10
    • 2021-12
    • 2022-01
    • 2022-02
    • 2022-03
    • 2022-04
    • 2022-05
    • 2022-06
    • 2022-07
    • 2022-08
    • 2022-09
    • 2022-10
    • 2022-11
    • 2022-12
    • 2023-01
    • 2023-02
    • 2023-03
    • 2023-04
    • 2023-05
    • 2023-06
    • 2023-07
    • 2023-08
    • 2023-09
    • 2023-10
    • 2023-12
    • 2024-02
    • 2024-04
    • 2024-05
    • 2024-06
    • 2025-02
    • 2025-07
    Top

    Copyright·© 2019 侯体宗版权所有· 粤ICP备20027696号 PHP交流群

    侯体宗的博客