以过往的经验来看,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下查看下就行了。
来源:oschina
链接:https://my.oschina.net/u/578519/blog/107848