`
san_yun
  • 浏览: 2653499 次
  • 来自: 杭州
文章分类
社区版块
存档分类
最新评论

写出健壮的Bash脚本

 
阅读更多

许多人用shell脚本完成一些简单任务,而且变成了他们生命的一部分。不幸的是,shell脚本在运行异常时会受到非常大的影响。在写脚本时将这类问题最小化是十分必要的。本文中我将介绍一些让bash脚本变得健壮的技术。

使用set -u

你因为没有对变量初始化而使脚本崩溃过多少次?对于我来说,很多次。

chroot=$1
...
rm -rf $chroot/usr/share/doc

如果上面的代码你没有给参数就运行,你不会仅仅删除掉chroot中的文档,而是将系统的所有文档都删除。那你应该做些什么呢?好在bash提供了set -u ,当你使用未初始化的变量时,让bash自动退出。你也可以使用可读性更强一点的set -o nounset

david% bash /tmp/shrink-chroot.sh

$chroot=

david% bash -u /tmp/shrink-chroot.sh

/tmp/shrink-chroot.sh: line 3: $1: unbound variable

david%

使用set -e

你写的每一个脚本的开始都应该包含set -e 。这告诉bash一但有任何一个语句返回非真的值,则退出bash。使用-e的好处是避免错误滚雪球般的变成严重错误,能尽早的捕获错误。更加可读的版本:set -o errexit

使用-e把你从检查错误中解放出来。如果你忘记了检查,bash会替你做这件事。不过你也没有办法使用$? 来获取命令执行状态了,因为bash无法获得任何非0的返回值。你可以使用另一种结构:

command

if [ "$?"-ne 0]; then echo "command failed"; exit 1; fi

可以替换成:

command || { echo "command failed"; exit 1; }

或者使用:

if ! command; then echo "command failed"; exit 1; fi

如果你必须使用返回非0值的命令,或者你对返回值并不感兴趣呢?你可以使用 command || true ,或者你有一段很长的代码,你可以暂时关闭错误检查功能,不过我建议你谨慎使用。

set +e

command1

command2

set -e

相关文档指出,bash默认返回管道中最后一个命令的值,也许是你不想要的那个。比如执行 false | true 将会被认为命令成功执行。如果你想让这样的命令被认为是执行失败,可以使用 set -o pipefail

程序防御 - 考虑意料之外的事

你的脚本也许会被放到“意外”的账户下运行,像缺少文件或者目录没有被创建等情况。你可以做一些预防这些错误事情。比如,当你创建一个目录后,如果父目录不存在,mkdir 命令会返回一个错误。如果你创建目录时给mkdir 命令加上-p选项,它会在创建需要的目录前,把需要的父目录创建出来。另一个例子是 rm 命令。如果你要删除一个不存在的文件,它会“吐槽”并且你的脚本会停止工作。(因为你使用了-e选项,对吧?)你可以使用-f选项来解决这个问题,在文件不存在的时候让脚本继续工作。

准备好处理文件名中的空格

有些人从在文件名或者命令行参数中使用空格,你需要在编写脚本时时刻记得这件事。你需要时刻记得用引号包围变量。

if [ $filename = "foo" ];

$filename 变量包含空格时就会挂掉。可以这样解决:

if [ "$filename" = "foo" ];

使用$@ 变量时,你也需要使用引号,因为空格隔开的两个参数会被解释成两个独立的部分。

david% foo() { for i in $@; do echo $i; done }; foo bar "baz quux"

bar

baz

quux

david% foo() { for i in "$@"; do echo $i; done }; foo bar "baz quux"

bar

baz quux

我没有想到任何不能使用"$@" 的时候,所以当你有疑问的时候,使用引号就没有错误。

如果你同时使用find和xargs,你应该使用 -print0 来让字符分割文件名,而不是换行符分割。

david% touch "foo bar"

david% find | xargs ls

ls: ./foo: No such file or directory

ls: bar: No such file or directory

david% find -print0 | xargs -0 ls

./foo bar

设置的陷阱

当你编写的脚本挂掉后,文件系统处于未知状态。比如锁文件状态、临时文件状态或者更新了一个文件后在更新下一个文件前挂掉。如果你能解决这些问题,无论是 删除锁文件,又或者在脚本遇到问题时回滚到已知状态,你都是非常棒的。幸运的是,bash提供了一种方法,当bash接收到一个UNIX信号时,运行一个 命令或者一个函数。可以使用trap 命令。

trap command signal [signal ...]

你可以链接多个信号(列表可以使用kill -l获得),但是为了清理残局,我们只使用其中的三个:INTTERMEXIT 。你可以使用-as来让traps恢复到初始状态。

信号描述

 

INT

Interrupt - 当有人使用Ctrl-C终止脚本时被触发

TERM

Terminate - 当有人使用kill杀死脚本进程时被触发

EXIT

Exit - 这是一个伪信号,当脚本正常退出或者set -e后因为出错而退出时被触发

 

 

 

 

当你使用锁文件时,可以这样写:

if [ ! -e $lockfile ]; then

touch $lockfile

critical-section

rm $lockfile

else

echo "critical-section is already running"

fi

当最重要的部分(critical-section)正在运行时,如果杀死了脚本进程,会发生什么呢?锁文件会被扔在那,而且你的脚本在它被删除以前再也不会运行了。解决方法:

if [ ! -e $lockfile ]; then

trap " rm -f $lockfile; exit" INT TERM EXIT

touch $lockfile

critical-section

rm $lockfile

trap - INT TERM EXIT

else

echo "critical-section is already running"

fi

现在当你杀死进程时,锁文件一同被删除。注意在trap命令中明确地退出了脚本,否则脚本会继续执行trap后面的命令。

竟态条件 (wikipedia )

在上面锁文件的例子中,有一个竟态条件是不得不指出的,它存在于判断锁文件和创建锁文件之间。一个可行的解决方法是使用IO重定向和bash的noclobber(wikipedia )模式,重定向到不存在的文件。我们可以这么做:

if ( set -o noclobber; echo "$$" > "$lockfile") 2> /dev/null;

then

trap 'rm -f "$lockfile"; exit $?' INT TERM EXIT

critical-section

rm -f "$lockfile"

trap - INT TERM EXIT

else

echo "Failed to acquire lockfile: $lockfile"

echo "held by $(cat $lockfile)"

fi

更复杂一点儿的问题是你要更新一大堆文件,当它们更新过程中出现问题时,你是否能让脚本挂得更加优雅一些。你想确认那些正确更新了,哪些根本没有变化。比如你需要一个添加用户的脚本。

add_to_passwd $user

cp -a /etc/skel /home/$user

chown $user /home/$user -R

当磁盘空间不足或者进程中途被杀死,这个脚本就会出现问题。在这种情况下,你也许希望用户账户不存在,而且他的文件也应该被删除。

rollback() {

del_from_passwd $user

if [ -e /home/$user ]; then

rm -rf /home/$user

fi

exit

}

 

trap rollback INT TERM EXIT

add_to_passwd $user

 

cp -a /etc/skel /home/$user

chown $user /home/$user -R

 

trap - INT TERM EXIT

在脚本最后需要使用trap关闭rollback调用,否则当脚本正常退出的时候rollback将会被调用,那么脚本等于什么都没做。

保持原子化

又是你需要一次更新目录中的一大堆文件,比如你需要将URL重写到另一个网站的域名。你也许会写:

for file in $(find /var/www -type f -name "*.html"); do

perl -pi -e 's/www.example.net/www.example.com/' $file

done

如果修改到一半是脚本出现问题,一部分使用www.example.com,而另一部分使用www.example.net。你可以使用备份和trap解决,但在升级过程中你的网站URL是不一致的。

解决方法是将这个改变做成一个原子操作。先对数据做一个副本,在副本中更新URL,再用副本替换掉现在工作的版本。你需要确认副本和工作版本目录在同一个磁盘分区上,这样你就可以利用Linux系统的优势,它移动目录仅仅是更新目录指向的inode节点。

cp -a /var/www /var/www-tmp

for file in $(find /var/www-tmp -type -f -name "*.html"); do

perl -pi -e 's/www.example.net/www.example.com/' $file

done

mv /var/www /var/www-old

mv /var/www-tmp /var/www

这意味着如果更新过程出问题,线上系统不会受影响。线上系统受影响的时间降低为两次mv操作的时间,这个时间非常短,因为文件系统仅更新inode而不用真正的复制所有的数据。

这种技术的缺点是你需要两倍的磁盘空间,而且那些长时间打开文件的进程需要比较长的时间才能升级到新文件版本,建议更新完成后重新启动这些进程。对于 apache服务器来说这不是问题,因为它每次都重新打开文件。你可以使用lsof命令查看当前正打开的文件。优势是你有了一个先前的备份,当你需要还原 时,它就派上用场了。

进阶阅读

 

  • Classic Shell Scripting
  • Learning the Bash Shell
  • Bash website
  • Bash Manual
  • Advanced Bash-Scripting Guide
  • 分享到:
    评论

    相关推荐

      写出健壮Bash Shell脚本的一些技巧总结

      以下是一些提高Bash脚本健壮性的关键技巧: 1. **使用`set -u`**: `set -u` 或 `set -o nounset` 是Bash的一个选项,它强制脚本在使用未定义的变量时立即退出。这有助于避免因未初始化的变量导致的潜在危险,例如...

      Linux中高效编写Bash脚本的9个技巧

      ### Linux中高效编写Bash脚本的9个技巧 ...总之,以上这些技巧可以帮助开发者编写出更加高效、可靠和易于维护的Bash脚本。无论是初学者还是经验丰富的系统管理员,掌握这些基础知识都是非常有益的。

      高级Bash脚本编程指南: 一本深入学习shell脚本艺术的书籍

      通过阅读《高级Bash脚本编程指南》,你可以深入理解Bash脚本的内在原理,并能够编写出高效、可靠的自动化脚本来简化日常的Linux系统管理工作。书中实例丰富,注释详细,非常适合自学和参考。同时,配合HTML格式的...

      bash_book.rar

      最后,本书可能还会讨论bash的调试技巧和最佳实践,帮助开发者定位问题并写出更健壮的脚本。例如,使用set命令启用脚本的调试模式,或者通过set -o nounset防止未定义变量的使用,可以避免很多潜在的错误。 总的来...

      shop:商店的简单bash脚本

      在IT行业中,Shell脚本是一种极其实用的工具,特别是在系统管理、自动化任务执行以及日常工作效率提升方面。...不过,通过学习和应用这些Bash脚本的基础知识,你可以创建出适应各种业务需求的自动化工具。

      Linux 命令与Shell 脚本编程pdf书籍

      错误处理和调试技巧,帮助你写出更健壮的脚本;以及如何利用脚本与系统交互,如处理进程、系统资源和网络通信。对于那些希望深入理解Linux系统工作原理和提升系统管理技能的人来说,这部分内容尤为重要。 通过阅读...

      高级Bash编程-by Mendel Cooper

      这份“高级Bash编程”教程,尽管年代稍显久远,但其内容仍然具有很高的实用价值,尤其对于那些希望通过掌握Bash脚本提高工作效率或进行自动化任务处理的人来说,是一份不可或缺的学习资料。 一、Bash基础 1. **...

      高级Shell脚本编程指南

      5. **错误处理**:使用`set -e`来使脚本在遇到错误时立即退出,确保脚本的健壮性。 通过深入学习和实践这些概念,你将能够编写出高效、灵活的Bash脚本来自动化日常任务,提高工作效率。在实际应用中,不断探索Bash...

      脚本撰写指南.pdf

      文档列出了Shell脚本中的常用命令,例如grep、nohup、tar、unzip和cp等,这些命令是Shell脚本编写中常用的工具。grep命令用于在文件中搜索特定的文本模式,并可以进行复杂的文本匹配操作,比如忽略大小写或显示匹配...

      ABS Guide中文版(非扫描版)

      - **O¶**:这里可能介绍了一些最佳实践,帮助用户写出可维护性和可读性更好的脚本。 - **€²**:这部分可能探讨了如何处理脚本中的异常情况,确保脚本即使在出现问题时也能优雅地退出。 - **?§º‚**:这里可能...

      BashSample:我的 Bash 基础练习

      BashSample项目是针对初学者设计的基础练习,旨在帮助用户熟悉Bash脚本编写和命令行操作。 1. **Bash基础概念** - **Shell**:Shell是操作系统提供的一种用户界面,允许用户与内核进行交互。 - **Bash Shell**:...

      用Python加持Linux Shell脚本编写CSV文件即可完美解决脚本中的返回

      通过这种方式,可以创建出更健壮、可维护的自动化解决方案,尤其适用于那些需要复杂数据处理和逻辑判断的场景。 在实际项目中,我们可能还需要考虑错误处理、日志记录、性能优化等细节。例如,使用try-except捕获...

      我的脚本

      2. 脚本文件:如`script1.py`、`script2.sh`等,分别代表Python脚本和Bash脚本。 3. 测试文件:用于验证脚本功能的正确性,如`test_script1.py`。 4. 配置文件:可能包含环境变量、数据库连接信息等,如`.env`。 5. ...

      oracle日志alter.log每天切割脚本

      2. **提升脚本健壮性**:对脚本中的变量进行更严格的校验,比如检查`ORACLE_SID`和`db_unique_name`等变量的有效性。 3. **优化性能**:对于非常大的日志文件,可以考虑采用更高效的读写策略,比如使用`dd`命令替代`...

      bash函数:有用的bash函数

      要定义一个Bash函数,首先需要在shell会话或脚本中写一个函数名,后跟一对圆括号,里面包含执行的命令。例如,创建一个名为`greet`的函数,它打印一条问候消息: ```bash greet() { echo "Hello, World!" } ``...

      LINUX与UNIX SHELL编程指南

      这不仅能帮助我们写出健壮的脚本,还能提升代码的可读性和维护性。 总的来说,"LINUX与UNIX SHELL编程指南"将引导读者全面理解Shell编程,通过学习,你可以编写出自动化任务、管理文件系统、集成系统服务,甚至构建...

      shellcheck

      - **陷阱和常见错误**:它能识别出可能导致问题的常见模式,如不安全的文件操作、未捕获的错误条件等,帮助你写出更健壮的脚本。 要在Windows上运行Shellcheck,首先解压文件,然后双击`shellcheck-latest.exe`。你...

    Global site tag (gtag.js) - Google Analytics