欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  网络运营

脚本实现SSH登录邮件报警

程序员文章站 2022-05-12 08:48:32
登录保护是一个非常重要的环节,下面通过图文并茂的方式给大家详细讲解下: 前两天@cyy 给我发了一个图 然后我就想到ushq的ssh登录app通知功能,然后就像...

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


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

脚本实现SSH登录邮件报警

然后我就想到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 content
cat >> mail-no-base64.html <<eof
#请自行准备邮件模板,以下为可能用到的变量
#输出主机名 `hostname`
#输出登录端口 ${ssh_client##* }
#输出登录来源ip ${ssh_client%% *}
#输出ip地址归属地 {country}_${area}_${region}_${city}_${isp}
#输出登录时间 `date`
eof

#base64 encoding
base64 mail-no-base64.html > mail-base64.html

#使用sendmail
#sendmail -t >/dev/null 2>&1 <<eof
#to:example@example.com
#from:example<example@example.com>
#subject:[`hostname`]服务器登录告警

#`cat mail-no-base64.html`

#eof

#使用postfix
#cat >> mail.php <<eof
#<?php
#\$to = "example@example.com";
#\$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<example@example.com>' . "\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
<?php
require("smtp-class.php"); 
\$smtpserver = "smtp.example.com";
\$smtpserverport = 25;
\$smtpusermail = "example@example.com";
\$smtpemailto = "example@example.com";
\$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); 
?>
eof
php mail.php
yes 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

<?php
class 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;
    }
  }

}

?>
实现效果:

脚本实现SSH登录邮件报警

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