36.2 shell wrappers

wrapper是一个包含系统命令和工具的脚本,脚本会把一些参数传递给这些(脚本内的)命令。将一个复杂的命令封装成一个wrapper是为了调用它时比较简单好记,特别在使用sed和awk命令时会这么做。

sed或awk脚本通常在命令行下调用时是sed -e ‘命令’或者awk ‘命令’。在Bash脚本中嵌入这些命令会让它们在调用时很简单,并且能够被重用。使用这种方法可以将sed和awk的优势统一起来,比如将sed命令处理的结果通过管道传递给awk继续处理。将这些保存成为一个可执行文件,你可以重复调用它的原始版本或者修改版本,而不用在命令行里反复敲冗长的命令。

Example 36-1. shell wrapper

  1. #!/bin/bash
  2. # 这个脚本功能是去除文件中的空白行
  3. # 没有做参数检查
  4. #
  5. # 也许你想添加下面的内容:
  6. #
  7. # E_NOARGS=85
  8. # if [ -z "$1" ]
  9. # then
  10. # echo "Usage: `basename $0` target-file"
  11. # exit $E_NOARGS
  12. # fi
  13. sed -e /^$/d "$1"
  14. # 就像这个命令
  15. # sed -e '/^$/d' filename
  16. # 通过命令行调用
  17. # '-e'意思是后面为编辑命令(这个选项可省略)。
  18. # '^'代表行首,'$'代表行尾。
  19. # 这个正则表达式表示要匹配出所有行首位没有内容的行,就是空白行。
  20. # 是删除命令(译注:就是把刚才选出来的空白行删掉)
  21. # 将文件名中的特殊字符和空白进行转译
  22. # 这个脚本并不会真正的修改目标文件,如果想对目标文件真正的修改,请将输出重定向
  23. exit

Example 36-2. 稍微复杂一点的 shell wrapper

  1. #!/bin/bash
  2. # subst.sh: 在文件中进行替换字符串的脚本
  3. # 例如 "sh subst.sh Smith Jones letter.txt"
  4. # letter.txt 中的所有 Jones 都被替换为 Smith。
  5. ARGS=3 # 这个脚本需要三个参数
  6. E_BADARGS=85 # 传给脚本的参数数量不正确
  7. if [ $# -ne "$ARGS" ]
  8. then
  9. echo "Usage: `basename $0` old-pattern new-pattern filename"
  10. exit $E_BADARGS
  11. fi
  12. old_pattern=$1
  13. new_pattern=$2
  14. if [ -f "$3" ]
  15. then
  16. file_name=$3
  17. else
  18. echo "File \"$3\" does not exist."
  19. exit $E_BADARGS
  20. fi
  21. # -----------------------------------------------
  22. # 这里是最核心的部分
  23. sed -e "s/$old_pattern/$new_pattern/g" $file_name
  24. # -----------------------------------------------
  25. # 's' 是sed中的替换命令
  26. # /pattern/调用地址匹配
  27. # 'g' 表示要对文件中的所有匹配项目都进行替换操作,而不是仅对第一个这样干。
  28. # 如果需要深入了解,请阅读sed命令的相关文档。
  29. exit $? # 将这个脚本的输出重定向到一个文件即可记录真正的结果

Example 36-3. 一个通用的写日志文件的 shell wrapper

  1. #!/bin/bash
  2. # logging-wrapper.sh
  3. # 一个通用的shell wrapper,在进行操作的同时对操作进行日志记录
  4. DEFAULT_LOGFILE=logfile.txt
  5. # 设置下面两个变量的值
  6. OPERATION=
  7. # 可以是任意操作,比如一个awk脚本或者用管道连接的复杂命令
  8. LOGFILE=
  9. if [ -z "$LOGFILE" ]
  10. then # 如果没有设置日志文件,则使用默认文件名
  11. LOGFILE="$DEFAULT_LOGFILE"
  12. fi
  13. # 对于操作命令的参数(可选)
  14. OPTIONS="$@"
  15. # 日志记录
  16. echo "`date` + `whoami` + $OPERATION "$@"" >> $LOGFILE
  17. # 进行操作动作
  18. exec $OPERATION "$@"
  19. # 要在真正执行操作之前写日志
  20. # 思考下为什么要先写日志,后操作。

Example 36-4. 关于awk脚本的 shell wrapper

  1. #!/bin/bash
  2. # pr-ascii.sh: 打印ASCII码表格
  3. START=33 # 可打印的ASCII码范围(十进制)
  4. END=127 # 不会输出不可打印的ASCII码
  5. echo " Decimal Hex Character" # 表头
  6. echo " ------- --- ---------"
  7. for ((i=START; i<=END; i++))
  8. do
  9. echo $i | awk '{printf(" %3d %2x %c\n", $1, $1, $1)}'
  10. # Bash内置的printf命令无法完成下面的操作: (译注:所以这使用awk脚本来实现输出)
  11. # printf "%c" "$i"
  12. done
  13. exit 0
  14. # Decimal Hex Character
  15. # ------- --- ---------
  16. # 33 21 !
  17. # 34 22 "
  18. # 35 23 #
  19. # 36 24 $
  20. #
  21. # . . .
  22. #
  23. # 122 7a z
  24. # 123 7b {
  25. # 124 7c |
  26. # 125 7d }
  27. # 将输出重定向到文件
  28. # 或者用管道传递给"more": sh pr-asc.sh | more

Example 36-5. 另一个关于awk的 shell wrapper

  1. #!/bin/bash
  2. # 在目标文件中添加一个数字的特殊列
  3. # 十进制浮点数也可以,因为awk可以处理这样的输出。
  4. ARGS=2
  5. E_WRONGARGS=85
  6. if [ $# -ne "$ARGS" ] # Check for proper number of command-line args.
  7. then
  8. echo "Usage: `basename $0` filename column-number"
  9. exit $E_WRONGARGS
  10. fi
  11. filename=$1
  12. column_number=$2
  13. # 将shell脚本的变量传递给awk有一点难办。
  14. # 第一种方法是用引号将Bash脚本变量在awk脚本中包起来
  15. # $'$BASH_SCRIPT_VAR'
  16. # ^ ^
  17. # 下面的awk脚本就是这么干的。
  18. # 详细用法可以查阅awk文档。
  19. # 多行的awk脚本可以写成这样
  20. # awk '
  21. # ...
  22. # ...
  23. # ...
  24. # '
  25. # 开始awk脚本
  26. # -----------------------------
  27. awk '
  28. { total += $'"${column_number}"' # 译注:这就是那个bash脚本变量
  29. }
  30. END {
  31. print total
  32. }
  33. ' "$filename"
  34. # -----------------------------
  35. # 结束awk脚本
  36. # 将shell变量传递给awk脚本也许是不安全的
  37. # 所以Stephane Chazelas提出了下面的替代方案:
  38. # ---------------------------------------
  39. # awk -v column_number="$column_number" ' # 译注:将shell的值赋给一个awk变量
  40. # { total += $column_number
  41. # }
  42. # END {
  43. # print total
  44. # }' "$filename"
  45. # ---------------------------------------
  46. exit 0

能满足那些需要瑞士军刀般全能工具的脚本语言,就只有Perl了。Perl集合了sed和awk的能力,并且比C更加精简。它是模块化的并且能支持包括厨房洗碗槽在内的所有面向对象编程所能涉及的事物。短小的Perl脚本可以嵌入shell脚本中,甚至Perl可以完全替代shell脚本。(本书作者对此仍然抱有怀疑)

Example 36-6. Perl嵌入Bash脚本

  1. #!/bin/bash
  2. # shell命令先于Perl脚本执行
  3. echo "This precedes the embedded Perl script within \"$0\"."
  4. echo "==============================================================="
  5. perl -e 'print "This line prints from an embedded Perl script.\n";'
  6. # 像sed命令一样,Perl使用'-e'选项
  7. echo "==============================================================="
  8. echo "However, the script may also contain shell and system commands."
  9. exit 0

即使能将Bash脚本和Perl脚本合二为一,先执行Bash部分还是Perl部分仍然要取决于调用脚本的方式。

Example 36-7. Bash和Perl脚本合并

  1. #!/bin/bash
  2. # bashandperl.sh
  3. echo "Greetings from the Bash part of the script, $0."
  4. # 这里可以写更多的Bash命令
  5. exit
  6. # Bash脚本部分结束
  7. # =======================================================
  8. #!/usr/bin/perl
  9. # 这部分脚本要像下面这样调用
  10. # perl -x bashandperl.sh
  11. print "Greetings from the Perl part of the script, $0.\n";
  12. # Perl 看起来并不像 “echo” ...
  13. # 这里可以写更多的Perl命令
  14. # Perl命令部分结束
  1. bash$ bash bashandperl.sh
  2. Greetings from the Bash part of the script.
  3. bash$ perl -x bashandperl.sh
  4. Greetings from the Perl part of the script.

当然还可以用shell wrapper嵌入更多的“外来户”,比如Python或者其他的…

Example 36-8. Python嵌入Bash脚本

  1. #!/bin/bash
  2. # ex56py.sh
  3. # shell脚本先于Python脚本执行
  4. echo "This precedes the embedded Python script within \"$0.\""
  5. echo "==============================================================="
  6. python -c 'print "This line prints from an embedded Python script.\n";'
  7. # 并不像sed和Perl,Python使用'-c'选项
  8. python -c 'k = raw_input( "Hit a key to exit to outer script. " )'
  9. echo "==============================================================="
  10. echo "However, the script may also contain shell and system commands."
  11. exit 0

使用脚本封装mplayer或者Google翻译服务器的一些功能,你能做出给你反馈一些信息的小东西。

Example 36-9. 会讲话的脚本

  1. #!/bin/bash
  2. # 参见:
  3. # http://elinux.org/RPi_Text_to_Speech_(Speech_Synthesis)
  4. # 为了连接Google翻译服务器,这个脚本必须连接到互联网才能工作,
  5. # 而且你的计算机上必须装有mplayer。
  6. speak()
  7. {
  8. local IFS=+
  9. # 先调用mplayer,再连接Google翻译服务器。
  10. /usr/bin/mplayer -ao alsa -really-quiet -noconsolecontrols \
  11. "http://translate.google.com/translate_tts?tl=en&q="$*""
  12. # 可以说话的Google翻译
  13. }
  14. LINES=4
  15. spk=$(tail -$LINES $0) # 同样的结尾
  16. speak "$spk"
  17. exit
  18. # BRowns 很高兴与你谈话。

有个有趣的shell wrapper例子是Martin Matusiak的undvd,为复杂的mencoder工具提供了一个简单易用的命令行接口。另一个例子是Itzchak Rehberg的Ext3Undel,它为在ext3文件系统上恢复删除的文件提供了一整套工具。

Notes
[1] Linux工具事实上很多是shell wrapper,比如/usr/bin/pdf2ps,/usr/bin/batch和/usr/bin/xmkmf。