• 常用 Shell 脚本


    1、检测两台服务器指定目录下的文件一致性

    #!/bin/bash
    #####################################
    #检测两台服务器指定目录下的文件一致性
    #####################################
    #通过对比两台服务器上文件的md5值,达到检测一致性的目的
    dir=/data/web
    b_ip=192.168.88.10
    #将指定目录下的文件全部遍历出来并作为md5sum命令的参数,进而得到所有文件的md5值,并写入到指定文件中
    find $dir -type f|xargs md5sum > /tmp/md5_a.txt
    ssh $b_ip "find $dir -type f|xargs md5sum > /tmp/md5_b.txt"
    scp $b_ip:/tmp/md5_b.txt /tmp
    #将文件名作为遍历对象进行一一比对
    for f in `awk '{print 2} /tmp/md5_a.txt'`
    do
        #以a机器为标准,当b机器不存在遍历对象中的文件时直接输出不存在的结果
        if grep -qw "$f" /tmp/md5_b.txt
        then
            md5_a=`grep -w "$f" /tmp/md5_a.txt|awk '{print 1}'`
            md5_b=`grep -w "$f" /tmp/md5_b.txt|awk '{print 1}'`
            #当文件存在时,如果md5值不一致则输出文件改变的结果
            if [ $md5_a != $md5_b ]
            then
                echo "$f changed."
            fi
        else
            echo "$f deleted."
        fi
    done
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28

    2、定时清空文件内容,定时记录文件大小

    #!/bin/bash
    ################################################################
    #每小时执行一次脚本(任务计划),当时间为0点或12点时,将目标目录下的所有文件内
    #容清空,但不删除文件,其他时间则只统计各个文件的大小,一个文件一行,输出到以时#间和日期命名的文件中,需要考虑目标目录下二级、三级等子目录的文件
    ################################################################
    logfile=/tmp/`date +%H-%F`.log
    n=`date +%H`
    if [ $n -eq 00 ] || [ $n -eq 12 ]
    then
        #通过for循环,以find命令作为遍历条件,将目标目录下的所有文件进行遍历并做相应操作
        for i in `find /data/log/ -type f`
        do
            true > $i
        done
    else
        for i in `find /data/log/ -type f`
        do
            du -sh $i >> $logfile
        done
    fi
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20

    3、检测网卡流量,并按规定格式记录在日志中

    #!/bin/bash
    #######################################################
    #检测网卡流量,并按规定格式记录在日志中
    #规定一分钟记录一次
    #日志格式如下所示:
    #2019-08-12 20:40
    #ens33 input: 1234bps
    #ens33 output: 1235bps
    ######################################################3
    while :
    do
        #设置语言为英文,保障输出结果是英文,否则会出现bug
        LANG=en
        logfile=/tmp/`date +%d`.log
        #将下面执行的命令结果输出重定向到logfile日志中
        exec >> $logfile
        date +"%F %H:%M"
        #sar命令统计的流量单位为kb/s,日志格式为bps,因此要*1000*8
        sar -n DEV 1 59|grep Average|grep ens33|awk '{print $2,"\t","input:","\t",$5*1000*8,"bps","\n",$2,"\t","output:","\t",$6*1000*8,"bps"}'
        echo "####################"
        #因为执行sar命令需要59秒,因此不需要sleep
    done
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22

    4、计算文档每行出现的数字个数,并计算整个文档的数字总数

    #!/bin/bash
    #########################################################
    #计算文档每行出现的数字个数,并计算整个文档的数字总数
    ########################################################
    #使用awk只输出文档行数(截取第一段)
    n=`wc -l a.txt|awk '{print $1}'`
    sum=0
    #文档中每一行可能存在空格,因此不能直接用文档内容进行遍历
    for i in `seq 1 $n`
    do
        #输出的行用变量表示时,需要用双引号
        line=`sed -n "$i"p a.txt`
        #wc -L选项,统计最长行的长度
        n_n=`echo $line|sed s'/[^0-9]//'g|wc -L`
        echo $n_n
        sum=$[$sum+$n_n]
    done
    echo "sum:$sum"
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18

    5、kill所有脚本

    
    #!/bin/bash
    if [ $# -ne 1 ]; then
        echo "Usage: $0 filename"
    fi
    dir=$(dirname $1)
    file=$(basename $1)
    ftp -n -v << EOF   # -n 自动登录
    open 192.168.1.10  # ftp服务器
    user admin password
    binary   # 设置ftp传输模式为二进制,避免MD5值不同或.tar.gz压缩包格式错误
    cd $dir
    get "$file"
    EOF
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14

    6、从 FTP 服务器下载文件

    #!/bin/bash  
    if [ $# -ne 1 ]; then  
        echo "Usage: $0 filename"  
    fi  
    dir=$(dirname $1)  
    file=$(basename $1)  
    ftp -n -v << EOF   # -n 自动登录  
    open 192.168.1.10  # ftp服务器  
    user admin password  
    binary   # 设置ftp传输模式为二进制,避免MD5值不同或.tar.gz压缩包格式错误  
    cd $dir  
    get "$file"  
    EOF
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    7、监测 Nginx 访问日志 502 情况,并做相应动作

    假设服务器环境为 lnmp,近期访问经常出现 502 现象,且 502 错误在重启 php-fpm 服务后消失,因此需要编写监控脚本,一旦出现 502,则自动重启 php-fpm 服务。

    
    #场景:
    #1.访问日志文件的路径:/data/log/access.log
    #2.脚本死循环,每10秒检测一次,10秒的日志条数为300条,出现502的比例不低于10%(30条)则需要重启php-fpm服务
    #3.重启命令为:/etc/init.d/php-fpm restart
    #!/bin/bash
    ###########################################################
    #监测Nginx访问日志502情况,并做相应动作
    ###########################################################
    log=/data/log/access.log
    N=30 #设定阈值
    while :
    do
        #查看访问日志的最新300条,并统计502的次数
        err=`tail -n 300 $log |grep -c '502" '`
        if [ $err -ge $N ]
        then
            /etc/init.d/php-fpm restart 2> /dev/null
            #设定60s延迟防止脚本bug导致无限重启php-fpm服务
            sleep 60
        fi
        sleep 10
    done
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23

    8、扫描主机端口状态

    #!/bin/bash
    HOST=$1
    PORT="22 25 80 8080"
    for PORT in $PORT; do
        if echo &>/dev/null > /dev/tcp/$HOST/$PORT; then
            echo "$PORT open"
        else
            echo "$PORT close"
        fi
    done
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    9、监控 httpd 的进程数,根据监控情况做相应处理

    
    #!/bin/bash
    ###############################################################################################################################
    #需求:
    #1.每隔10s监控httpd的进程数,若进程数大于等于500,则自动重启Apache服务,并检测服务是否重启成功
    #2.若未成功则需要再次启动,若重启5次依旧没有成功,则向管理员发送告警邮件,并退出检测
    #3.如果启动成功,则等待1分钟后再次检测httpd进程数,若进程数正常,则恢复正常检测(10s一次),否则放弃重启并向管理员发送告警邮件,并退出检测
    ###############################################################################################################################
    #计数器函数
    check_service()
    {
        j=0
        for i in `seq 1 5`
        do
            #重启Apache的命令
            /usr/local/apache2/bin/apachectl restart 2> /var/log/httpderr.log
            #判断服务是否重启成功
            if [ $? -eq 0 ]
            then
                break
            else
                j=$[$j+1]
            fi
            #判断服务是否已尝试重启5次
            if [ $j -eq 5 ]
            then
                mail.py
                exit
            fi
        done
    }
    while :
    do
        n=`pgrep -l httpd|wc -l`
        #判断httpd服务进程数是否超过500
        if [ $n -gt 500 ]
        then
            /usr/local/apache2/bin/apachectl restart
            if [ $? -ne 0 ]
            then
                check_service
            else
                sleep 60
                n2=`pgrep -l httpd|wc -l`
                #判断重启后是否依旧超过500
                if [ $n2 -gt 500 ]
                then
                    mail.py
                    exit
                fi
            fi
        fi
        #每隔10s检测一次
        sleep 10
    done
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55

    10、批量修改服务器用户密码

    Linux主机SSH连接信息:旧密码

    # cat old_pass.txt   
    192.168.18.217  root    123456     22  
    192.168.18.218  root    123456     22
    
    • 1
    • 2
    • 3

    内容格式:IP User Password Port

    SSH远程修改密码脚本:新密码随机生成

    
    #!/bin/bash  
    OLD_INFO=old_pass.txt  
    NEW_INFO=new_pass.txt  
    for IP in $(awk '/^[^#]/{print $1}' $OLD_INFO); do  
        USER=$(awk -v I=$IP 'I==$1{print $2}' $OLD_INFO)  
        PASS=$(awk -v I=$IP 'I==$1{print $3}' $OLD_INFO)  
        PORT=$(awk -v I=$IP 'I==$1{print $4}' $OLD_INFO)  
        NEW_PASS=$(mkpasswd -l 8)  # 随机密码  
        echo "$IP   $USER   $NEW_PASS   $PORT" >> $NEW_INFO  
        expect -c "  
        spawn ssh -p$PORT $USER@$IP  
        set timeout 2  
        expect {  
            \"(yes/no)\" {send \"yes\r\";exp_continue}  
            \"password:\" {send \"$PASS\r\";exp_continue}  
            \"$USER@*\" {send \"echo \'$NEW_PASS\' |passwd --stdin $USER\r exit\r\";exp_continue}  
        }"  
    done  
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

    生成新密码文件:

    # cat new_pass.txt   
    192.168.18.217  root    n8wX3mU%      22  
    192.168.18.218  root    c87;ZnnL      22
    
    • 1
    • 2
    • 3

    11、iptables 自动屏蔽访问网站频繁的IP

    场景:恶意访问,安全防范

    1)屏蔽每分钟访问超过200的IP

    方法1:根据访问日志(Nginx为例)

    #!/bin/bash
    DATE=$(date +%d/%b/%Y:%H:%M)
    ABNORMAL_IP=$(tail -n5000 access.log |grep $DATE |awk '{a[$1]++}END{for(i in a)if(a[i]>100)print i}')
    #先tail防止文件过大,读取慢,数字可调整每分钟最大的访问量。awk不能直接过滤日志,因为包含特殊字符。
    for IP in $ABNORMAL_IP; do
        if [ $(iptables -vnL |grep -c "$IP") -eq 0 ];
        then
            iptables -I INPUT -s $IP -j DROP
        fi
    done
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    方法2:通过TCP建立的连接

    
    #!/bin/bash
    ABNORMAL_IP=$(netstat -an |awk '$4~/:80$/ && $6~/ESTABLISHED/{gsub(/:[0-9]+/,"",$5);{a[$5]++}}END{for(i in a)if(a[i]>100)print i}')
    #gsub是将第五列(客户端IP)的冒号和端口去掉
    for IP in $ABNORMAL_IP; do
        if [ $(iptables -vnL |grep -c "$IP") -eq 0 ]; then
            iptables -I INPUT -s $IP -j DROP
        fi
    done
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    2)屏蔽每分钟SSH尝试登录超过10次的IP
    方法1:通过lastb获取登录状态:

    
    #!/bin/bash
    DATE=$(date +"%a %b %e %H:%M") #星期月天时分  %e单数字时显示7,而%d显示07
    ABNORMAL_IP=$(lastb |grep "$DATE" |awk '{a[$3]++}END{for(i in a)if(a[i]>10)print i}')
    for IP in $ABNORMAL_IP; do
        if [ $(iptables -vnL |grep -c "$IP") -eq 0 ];
        then
            iptables -I INPUT -s $IP -j DROP
        fi
    done
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10

    方法2:通过日志获取登录状态

    !/bin/bash  
    DATE=$(date +"%b %d %H")  
    ABNORMAL_IP="$(tail -n10000 /var/log/auth.log |grep "$DATE" |awk '/Failed/{a[$(NF-3)]++}END{for(i in a)if(a[i]>5)print i}')"  
    for IP in $ABNORMAL_IP; do  
        if [ $(iptables -vnL |grep -c "$IP") -eq 0 ]; then  
            iptables -A INPUT -s $IP -j DROP          
            echo "$(date +"%F %T") - iptables -A INPUT -s $IP -j DROP" >>~/ssh-login-limit.log      
        fi  
    done  
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9

    12、根据web访问日志,封禁请求量异常的IP,如IP在半小时后恢复正常,则解除封禁

    
    #!/bin/bash
    ####################################################################################
    #根据web访问日志,封禁请求量异常的IP,如IP在半小时后恢复正常,则解除封禁
    ####################################################################################
    logfile=/data/log/access.log
    #显示一分钟前的小时和分钟
    d1=`date -d "-1 minute" +%H%M`
    d2=`date +%M`
    ipt=/sbin/iptables
    ips=/tmp/ips.txt
    block()
    {
        #将一分钟前的日志全部过滤出来并提取IP以及统计访问次数
        grep '$d1:' $logfile|awk '{print $1}'|sort -n|uniq -c|sort -n > $ips
        #利用for循环将次数超过100的IP依次遍历出来并予以封禁
        for i in `awk '$1>100 {print $2}' $ips`
        do
            $ipt -I INPUT -p tcp --dport 80 -s $i -j REJECT
            echo "`date +%F-%T` $i" >> /tmp/badip.log
        done
    }
    unblock()
    {
        #将封禁后所产生的pkts数量小于10的IP依次遍历予以解封
        for a in `$ipt -nvL INPUT --line-numbers |grep '0.0.0.0/0'|awk '$2<10 {print $1}'|sort -nr`
        do
            $ipt -D INPUT $a
        done
        $ipt -Z
    }
    #当时间在00分以及30分时执行解封函数
    if [ $d2 -eq "00" ] || [ $d2 -eq "30" ]
    then
        #要先解再封,因为刚刚封禁时产生的pkts数量很少
        unblock
        block
    else
        block
    fi
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40

    原文:www.xampp.cc/archives/17137

  • 相关阅读:
    中国地图坐标系转换详解:从WGS-84到GCJ-02再到BD-09
    【LeetCode:2558. 从数量最多的堆取走礼物 | 大根堆】
    隐函数求导
    Python基础复习-面向过程的编程
    医疗实施-MDM主数据管理基本介绍
    跟着官方学电机,BLDC两种控制策略,学到即赚到
    计算机毕业设计JAVA辅导员职责信息管理系统mybatis+源码+调试部署+系统+数据库+lw
    TiDB数据迁移场景综述
    基于InsightFace的高精度人脸识别,可直接对标虹软
    数据结构--并查集
  • 原文地址:https://blog.csdn.net/qq_44787816/article/details/133269250