M22 程序员偷懒战法

前段时间有个外国的程序猿走红网络,这个哥们可以说是懒到了极点,上班请假给领导发短信写脚本,下班晚回家给老婆发短信写脚本,甚至于接个咖啡也要写脚本。这个哥们离职之后,他的同事在他的办公电脑上发现了这些脚本,并公布到了网上,引起众程序猿纷纷膜拜。最近我刚好学到shell脚本部分,就让我分析其中的一个跟领导请假的脚本吧。

#!/bin/sh -e

# Exit early if any session with my username is found
if who | grep -wq $USER; then
  exit
fi

# Phone numbers
MY_NUMBER='+xxx'
NUMBER_OF_BOSS='+xxx'

EXCUSES=(
  'Locked out'
  'Pipes broke'
  'Food poisoning'
  'Not feeling well'
)
rand=$[ $RANDOM % ${#EXCUSES[@]} ]

RANDOM_EXCUSE=${EXCUSES[$rand]}
MESSAGE="Gonna work from home. "$RANDOM_EXCUSE

# Send a text message
RESPONSE=`curl -fSs -u "$TWILIO_ACCOUNT_SID:$TWILIO_AUTH_TOKEN" \
  -d "From=$MY_NUMBER" -d "To=$NUMBER_OF_BOSS" -d "Body=$MESSAGE" \
  "https://api.twilio.com/2010-04-01/Accounts/$TWILIO_ACCOUNT_SID/Messages"`

# Log errors
if [ $? -gt 0 ]; then
  echo "Failed to send SMS: $RESPONSE"
  exit 1
fi

一 shebang声明

#!/bin/sh -e

脚本的shebang声明是在脚本直接执行,而不使用sh或者bash间接执行的时候,告诉解释程序此脚本的类型,如果脚本中没有此声明,那么脚本在执行的时候会使用默认的shell程序执行它。在此例中,脚本使用了/bin/sh作为解释程序。
在ubuntu系列linux里,为了加快系统启动速度,/bin/sh被设置成了指向dash的链接,而非bash

#ll /bin/sh
lrwxrwxrwx 1 root root 4 2月  16 10:23 /bin/sh -> dash*

在Centos里,/bin/sh链接到了bash

#ll /bin/sh
lrwxrwxrwx. 1 root root 4 2月  14 16:55 /bin/sh -> bash

所以在ubuntu里如果要使用bash要写成#!/bin/bash才行。
-e选项表示脚本中有任何错误都会终止执行脚本。

二 执行条件过滤

# Exit early if any session with my username is found
if who | grep -wq $USER; then
  exit
fi

脚本在主要代码执行前,首先要判断用户传入的参数、执行者的权限、文件目录是否存在等是否符合脚本执行的条件,所以会进行相应的判断,如果没达到符合的条件就会提示用户退出执行。在本例中,判断了脚本的所属者是否登录了系统,如果已经登录了就表明他已经来上班了,下面的请假过程就不用执行了。

三 声明变量

# Phone numbers
MY_NUMBER='+xxx'
NUMBER_OF_BOSS='+xxx'

定义字符串变量,分别存入本人和老板的电话号码。

EXCUSES=(
  'Locked out'
  'Pipes broke'
  'Food poisoning'
  'Not feeling well'
)

定义请假理由的数组

rand=$[ $RANDOM % ${#EXCUSES[@]} ]

随机抽取请假编号。${#EXCUSES[@]}代表数值元素的个数,然后通过与随机数的取余运算获得随机的编号。$[  ]为算数运算符号,与let、$(())等价

RANDOM_EXCUSE=${EXCUSES[$rand]}

取得数组中的某一项。

MESSAGE="Gonna work from home. "$RANDOM_EXCUSE

将消息头与随机项合并组成消息。

三 执行主要任务

# Send a text message
RESPONSE=`curl -fSs -u "$TWILIO_ACCOUNT_SID:$TWILIO_AUTH_TOKEN" \
  -d "From=$MY_NUMBER" -d "To=$NUMBER_OF_BOSS" -d "Body=$MESSAGE" \
  "https://api.twilio.com/2010-04-01/Accounts/$TWILIO_ACCOUNT_SID/Messages"`

其中的$TWILIO_ACCOUNT_SID和$TWILIO_AUTH_TOKEN为环境变量,通过curl命令执行了登录网站发送短信的过程,并把执行的结果返回给RESPONSE变量。

四 判断执行的结果

# Log errors
if [ $? -gt 0 ]; then
  echo "Failed to send SMS: $RESPONSE"
  exit 1
fi

当脚本的主要代码执行完毕后,通过检测执行后的状态来判断脚本的执行结果,如果执行结果大于0,说明执行过程中出现了错误,后台执行的脚本会以发邮件的方式提示用户。

五 设置环境变量和周期执行

TWILIO_ACCOUNT_SID=ACxxxxxxxxxxxxxxxxxxxxxxxxxxxxx
TWILIO_AUTH_TOKEN=yyyyyyyyyyyyyyyyyyyyyyyyyyyyyyy

此环境变量在用户未登录的情况下也要执行,所以不能设置在~/.bashrc ~/.bash_profil /etc/bashrc /etc/profile.d里,最好写在/etc/rc.local里

# Runs `hangover.sh` monday to friday at 8:45 am.
45 8 * * 1-5 /path/to/scripts/hangover.sh >> /path/to/hangover.log 2>&1

周一至周五的8点45分执行脚本,并输出日志。此任务没有用户名,应该是使用命令crontab -e编辑的。

六 总结

这位仁兄的代码简洁,注释清晰,内容包含了shell编程的主要步骤,值得我等新人研究学习。

原创文章,作者:realmaster,如若转载,请注明出处:http://www.178linux.com/72647

(0)
realmasterrealmaster
上一篇 2017-04-06 15:09
下一篇 2017-04-06 19:20

相关推荐

  • DNS and BIND 进阶

    主DNS服务器的ip地址:192.168.1.109 主DNS服务器主机名:bogon 正向区域名:sh.com 反向区域名:1.168.192.in-addr.arpa 正向区域文件名称:/var/named/sh.com.hosts 反向区域文件名称:/var/named/192.168.1.rev 配置主文件/etc/named.conf过程 1、编辑…

    Linux干货 2016-12-06
  • 第六周作业 bash编程学得好痛苦啊,啊啊啊~~需要挺住!

    请详细总结vim编辑器的使用并完成以下练习题 VIM很强大,但是万变不离其宗,最核心的三模式以下图表示: 1、复制/etc/rc.d/rc.sysinit文件至/tmp目录,将/tmp/rc.sysinit文件中的以至少一个空白字符开头的行的行首加#; :%s/^[[:blank:]]\+.*/\0#/g 2、复制/boot/grub/grub.conf至/…

    Linux干货 2016-12-19
  • linux磁盘管理

    硬盘接口类型:         IDE:并口,133M/s;100个IO/s        SCSI:并口,UltraSCSI320,320M/s UltraSCSI640 640M/s 150-200IO/s&nbsp…

    Linux干货 2016-08-29
  • linux程序包管理rpm,yum和编译安装以及冒泡排序练习

    linux程序包管理: API:Application Programming Interface POSIX:Portable OS 程序源代码–> 预处理–> 编译–> 汇编–> 链接 静态编译: 共享编译:.so ABI:Application Binary Interface W…

    Linux干货 2016-08-24
  • N26——第三周作业

    一、列出当前系统上所有已经登录的用户的用户名,注意:同一个用户登录多次,则只显示一次即可 [root@localhost ~]# who | cut -d ' ' -f1 | uniq root zhaoyujia 二、取出最后登录到当前系…

    Linux干货 2017-01-14
  • rsyslog+mysql+loganalyzer 搭建日志服务器及监控

    rsyslog 日志:历史事件; 历史事件:时间、地点、事件; syslog: klogd:kernel yslogd:system(application) 事件记录格式:日期时间 主机 进程[pid]:事件内容; C/S架构;通tcp或udp协议的服务完成日志记录的传送; rsyslog: rsyslog的特性: – 多线程; –…

    Linux干货 2017-01-10