第二章:Improving On User Commands--15.将删除的文件存档

   以过往的经验来看,Unix用户最常见的一个问题是无法恢复意外删除的文件和目录。键入 rm xyz 之后,一旦你按下RETURN键,那么一切都是浮云了。这个问题的一个解决方案是秘密的、自动的存档文件和目录。想象一下,这个脚本中的基本工作可以做的让用户毫无察觉。


#!/bin/sh
 
 # newrm.sh, 一个rm命令的替代品,
 # 通过在用户的家目录下建立并利用一个新目录,提供了一个非删除的能力
 # 它可以处理目录,也可以处理独立的文件,
 # 如果用户提供了-f 选项,那么删除时就不会存档
 
 # 重要警告:如果你要自动完成删除垃圾目录的工作,那么不要使用
 # 该脚本,因为这意味着你事实上什么也没有删除,最终你的磁盘空间会爆满
 
 mydir="$HOME/.deleted-files"
 realrm="/bin/rm"
 copy="/bin/cp -R"
 
 if [ $# -eq 0 ]; then     # 让 'rm' 输出用法错误信息
     exec $realrm  #当前的shell被 /bin/rm替换掉
 fi
 
 # 解析所有选项查找'-f'
 
 flags=""
 
 while getopts "dfiPRrvW" opt; do
     case $opt in
         f)exec $realrm "$@"   ;; # exec 让我们直接退出这个脚本
         *)flags="$flags-$opt" ;; # 别的选项是给 'rm'的,不是我们
     esac
 done
 shift $(($OPTIND-1))
 
 # 确保$mydir存在
 
 if [ ! -d $mydir ]; then
     if [ ! -w $HOME ]; then
         echo "$0 failed: can't create $mydir in $HOME" >&2
         exit 1
     fi
     mkdir $mydir
     chmod 700 $mydir  # 请给点隐私
 fi
 
 for arg
 do
     newname="$mydir/$(date "+%S.%M.%H.%d.%m.").$(basename "$arg")"
     if [ -f "$arg" ]; then
         $copy "$arg" "$newname"
     elif [ -d "$arg" ]; then
         $copy "$arg" "$newname"
     fi
 done
 
 exec $realrm $flags "$@"   # 我们的shell被realrm替换掉
脚本如何工作:

这个脚本中有很多值得思考的地方。注意到错误信息大部分都是调用realrm生成的,不管是错误的选项或是文件、目录名。同样,exec命令,用新的指定的进程替换了当前进程,也是可以的。exec一调用realrm,它就实际上退出了脚本,我们确定了从进程realrm(/bin/rm)得到的返回码返回给了调用的shell,并没有丢弃。因为这个脚本私下里建立了一个目录在用户的家目录下,它需要确定那儿的文件不能因为没有设置好umask的值而意外的对于别人是可读的。所以,这个脚本使用了chmod命令。最后,有点困惑的文件命名使用了basename命令去掉所有的路径信息,然后增加了时间和日期戳来给删除掉的文件命名,具体的格式是:秒.分.时.天.月.文件名

newname="$mydir/$(date "+%S.%M.%H.%d.%m.").$(basename "$arg")"
注意2个$()的应用。这个可能有点复杂,但很有用。记住,在$(和)之间的任何东西都会被提供给一个子shell,并且命令的结果是被替换过的。为什么用一个麻烦的时间戳?因为我们删除的文件有很多,其中难免有重名的。
运行脚本:
为了导入这个脚本,简单的添加一个alias,这样你就可以在键入rm的时候,用这个脚本代替了,而不是原来的rm命令。一个Bash/Ksh的alias命令是:
alias rm=yourpath/newrm.sh
注:取消这个别名的话,就用unalias rm。
运行结果:
直接在命令行上运行,然后到你的家目录下的隐藏目录.deleted-files下查看下就行了。


你可能感兴趣的:(linux,shell,shell,bash,scripts,cool,Wicked)