如果你想做一个网站,首先需要购买一个域名,与其说是购买不如说是租用,因为你要想使用这个域名,每年都需要缴纳一笔费用。既然是租用,所以就有期限,在到期前没有续费,则该域名会被收回,其他人可以注册并使用这个域名。

当我们名下域名非常多的时候,则很容易忘记为这些域名续费从而导致过期并回收。

本案例的需求是,写一个shell脚本来监控指定的域名是否到期,具体要求如下:

1)写一个函数,域名以参数的形式传递给这个函数

2)域名到期前的一周和到期后的一周(两周时间),每天都要发告警邮件

3)脚本每天执行一次


知识点一:whois

一个域名的信息,比如所有者邮箱、电话、地址和什么时候过期等都是公开的,可以在浏览器里访问https://www.whois.net进行查询。国内也有不少类似的网站可以查询域名信息。在Linux命令行下如何查询呢?

# whois aminglinux.com
   Domain Name: AMINGLINUX.COM
   Registry Domain ID: 1800256822_DOMAIN_COM-VRSN
   Registrar WHOIS Server: whois.55hl.com
   Registrar URL: http://www.55hl.com
   Updated Date: 2018-05-04T22:57:37Z
   Creation Date: 2013-05-10T06:02:05Z
   Registry Expiry Date: 2021-05-10T06:02:05Z
   Registrar: Jiangsu Bangning Science & technology Co. Ltd.
   Registrar IANA ID: 1469
   Registrar Abuse Contact Email: abuse@55hl.com
   Registrar Abuse Contact Phone: +86 025 86883426 1009
   Domain Status: ok https://icann.org/epp#ok
   Name Server: F1G1NS1.DNSPOD.NET
   Name Server: F1G1NS2.DNSPOD.NET
   DNSSEC: unsigned
   URL of the ICANN Whois Inaccuracy Complaint Form: https://www.icann.org/wicf/
>>> Last update of whois database: 2019-07-25T12:40:54Z <<<

系统默认没有这个命令,需要安装whois包。

本案例需要监控域名的过期时间,所以要关注的行是Registry Expiry Date。还有一个问题不得不考虑,不同的域(.com,.cn)查询到的结果有所不同,比如.cn的结果是这样的:

# whois aminglinux.cn
Domain Name: aminglinux.cn
ROID: 20160322s10001s82727381-cn
Domain Status: ok
Registrant ID: mr2272c32qw63z
Registrant: 个人用户名
Registrant Contact Email: 306798658@qq.com
Sponsoring Registrar: 北京新网数码信息技术有限公司
Name Server: ns11.xincache.com
Name Server: ns12.xincache.com
Registration Time: 2016-03-22 17:42:01
Expiration Time: 2020-03-22 17:42:01
DNSSEC: unsigned

所以.cn的过期时间需要关注的行是Expiration Time。


知识点二:cut命令

语法:cut -d '分隔字符' [-cf] n 这个n是数字

-d:后面指定分隔字符,分隔字符要用单引号括起来

-c:后面指定第几个字符

-f:后面指定第几个区块

# cat /etc/passwd |cut -d ':' -f 1 |head -n5
root
bin
daemon
adm
lp

说明:-d后面指定冒号为分隔字符,-f 1表示截取第一段,-f和1之间的空格可有可无。

# head -n2 /etc/passwd |cut -c2
o
i
# head -n2 /etc/passwd |cut -c1
r
b
# head -n2 /etc/passwd |cut -c1-10
root:x:0:0
bin:x:1:1:
# head -n2 /etc/passwd |cut -c5-10
:x:0:0
x:1:1:

说明:-c后面可以是1个数字n,也可以是一个区间n1-n2,还可以是多个数字n1,n2,n3。

# head -n2 /etc/passwd |cut -c1,3,10
ro0
bn:


知识点三:进程控制

当运行一个进程时,可以按ctrl+z使它暂停,然后使用fg命令恢复它,利用bg命令使它到后台运行,也可以按ctrl+c终止这个进程。输入jobs命令,可以看到暂停或者在后台运行的任务。

如果想把暂停的任务丢在后台跑起来,就使用bg命令。

# bg 
[1]+ vi test1.txt &
[1]+ Stopped        vi test1.txt

但是vi并不支持在后台运行,换一个其他的命令:

# sar 1 > /tmp/1.log
^Z    //这里按了ctrl+z
[2]+  Stopped            sar 1 > /tmp/1.log
# jobs
[1]-  Stopped            vi test.txt
[2]+  Stopped            sar 1 > /tmp/1.log
# bg 2
[2]+  sar 1 > /tmp/1.log &

说明:多个被暂停的任务会有编号,使用jobs命令可以看到两个任务,使用bg或者fg的时候,就需要在后面加一个编号。上例中使用bg 2把第二个被暂停的任务丢到后台跑起来了,命令行下使用命令在最后面加一个&符号,可以将这条命令直接丢入后台。

丢到后台的任务如何关闭?

如果没有退出刚才的shell,那么,先使用“fg 编号”把任务调到前台,再使用“ctrl+c”结束任务:

# fg 2
sar 1 > /tmp/1.log
^C   //这里使用了ctrl+c

另一种情况则是,关闭掉了当前的shell,再次打开一个新的shell中时,使用jobs命令并不会显示在后台运行或者被暂停的任务,要想停掉它的话,则需要先知道其pid,然后使用kill命令杀死那个进程。

# sar 1 10 > /tmp/1.log &
[1] 30218
# ps aux |grep sar
root      30218  0.0  0.0 108036   760 pts/0    S    11:22   0:00 sar 1 10
root      30221  0.0  0.0 112724   984 pts/0    S+   11:22   0:00 grep --color=auto sar

在shell脚本中,多条指令执行是有先后顺序的,就是说只有前面的指令执行完(不管成功与否),后面的指令才会执行。如果有一条指令运行时间比较久,则会阻碍后面的指令执行。若不想让这条执行慢的指令影响后面的指令,在执行该指令时,在后面加一个&,把它丢到后台去。使用&符号把任务丢到后台运行,会显示pid信息,如果忘记这个pid,我们还可以使用ps aux命令找到那个进程。想结束掉该进程,需要使用kill命令:

# kill 9433
[1]+ 已终止     sar 1 > /tmp/1.log

kill命令语法很简单,直接在后面加pid即可。


知识点四:判断一个变量是否为空

在shell脚本中,如果一个变量没有成功赋值就被引用了,则会影响到脚本的正常执行,判断一个变量的值是否为空有两种方法:

1)用-z(zero的意思,为空)

# b=       //给变量b赋值为空
# if [ -z "$b" ]; then echo "The value of b is null.";else echo "The value of b is $b";fi
The value of b is null.
# b=1
# if [ -z "$b" ]; then echo "The value of b is null.";else echo "The value of b is $b.";fi
The value of b is 1.

2)用-n(not null的意思,不为空)

# a=1
# if [ -n "$a" ]; then echo "The value of a is $a.";else echo "The value of a is null.";fi
The value of a is 1.
# a= 
# if [ -n "$a" ]; then echo "The value of a is $a.";else echo "The value of a is null.";fi
The value of a is null.


知识点五:判断某个进程是否存在

之前有用到过ps查看进程,但是需要结合grep,而且需要统计行数,其实还有一个更简单的用法:

# sleep 100 &
[1]  32744
# pgrep sleep
32744


知识点六:杀死进程

前面有用到kill命令杀死进程,但是需要知道进程的pid。再介绍一个简单的用法:

# sar 1 > /tmp/1.log &
# killall sar
[1]+  已终止     sar 1 > /tmp/sar.log

说明:killall和kill不同的地方在于killall可以直接跟进程名。killall也支持-9选项,有时候用killall杀死进程不好用,需要带上-9,但是进程杀不死时,要慎用-9。


本案例参考脚本

#!/bin/bash
#检测域名是否过期
#作者:
#日期:
#版本:v0.2

mail_u=admin@admin.com
#当前日期时间戳,用于和域名的到期时间做比较
t1=`date +%s`

#检测whois命令是否存在,不存在则安装whois包
is_install_whois()
{
    which whois >/dev/null 2>/dev/null
    if [ $? -ne 0 ]
    then
        yum install -y whois
    fi
}

notify()
{
    e_d=`whois $1 |grep 'Expiry Date' |awk '{print $4}' |cut -d 'T' -f 1`
    #上面的$1代表域名,遍历循环出来的。
    #如果e_d的值为空,则过滤关键词'Expiration Time'
    if [ -z "$e_d" ]
    then
        e_d=`whois $1|grep 'Expiration Time' |awk '{print $3}'`
    fi
    #将域名过期的日期转化为时间戳   
    e_t=`date -d "$e_d" +%s`
    #计算一周一共有多少秒
    n=`echo "86400*7" |bc`
    e_t1=$[$e_t-$n] #过期时间一周前的时间戳
    e_t2=$[$e_t+$n] #过期时间一周后的时间戳
    if [ $t1 -ge $e_t1 ] && [ $t1 -lt $e_t ]
    then
        python mail.py $mail_u "Domain $1 will to be expired." "Domain $1 expire date is $e_d."
    fi
    if [ $t1 -ge $e_t ] && [ $t1 -lt $e_t2 ]
    then
        python mail.py $mail_u "Domain $1 has been expired." "Domain $1 expire date is $e_d."
    fi    
}

#检测上次运行的whois查询进程是否存在
#若存在,需要杀死进程,以免影响本次脚本执行
if pgrep whois &>/dev/null
then
    killall -9 whois
fi
    
is_install_whois

for d in aaa.net aaa.com bbb.com aaa.cn ccc.com
do
  notify $d &
done