在 Linux 终端中实现回收站功能

本文仿照 Windows 回收站的功能,运用 Bash 脚本在 Linux 上实现。创建 delete 脚本代替 rm 命令对文件或目录进行删除操作。

该脚本实现了以下功能:

  • 对大于 2G 的文件或目录直接删除,否则放入 $HOME/.trash 目录下;
  • 恢复 trash 目录中的被删除文件到原目录下;
  • 文件存放在 trash 目录中超过七天被自动删除。

删除是危险系数很高的操作,一旦误删可能会造成难以估计的损失。在 Linux 系统中这种危险尤为明显,一条简单的语句:rm –rf /* 就会把整个系统全部删除,而 Linux 并不会因为这条语句的不合理而拒绝执行。

在 Windows 中,为了防止误删,系统提供了回收站功能。用户在执行删除操作后,文件并不会直接从硬盘中删除,而是被放到回收站中。在清空回收站前,如果发现有文件被误删,用户可以将回收站中的文件恢复到原来的位置。而 Linux 并没有提供类似功能,删除命令 rm 一旦确认执行,文件就会直接从系统中删除,很难恢复。


本文共用三个脚本实现了回收站的主要功能:Delete 脚本、logTrashDir 脚本和 restoreTrash 脚本。其中 Delete 脚本是核心脚本,其作用是重新封装 rm 命令。相对于 rm 的直接删除,该命令会先将文件或目录移动到 $home/.trash 目录下。如果用户想要将文件直接删除,可以用 -f 选项,delete 脚本会直接调用 rm –f 命令将文件从硬盘上删除。logTrashDir 脚本用于将被删除文件的信息记录到 trash 目录下的一个隐藏文件中。restoreTrash 脚本用来将放入 trash 中的文件或目录重新恢复到原路径下。在 Linux 系统中,只要将这三个脚本放到 /bin/ 目录下,并用 chmod +X filename 赋予可执行权限,即可直接使用。


下面将介绍每个脚本的主要部分:

1. Delete 脚本

1.1 创建回收站目录

首先要创建目录来存放被删除的文件,本文在用户根目录 $HOME 下建立 trash 目录来存放文件。具体代码如下:

realrm="/bin/rm"
if [ ! -d ~/.trash ]
 then
      mkdir -v ~/.trash
      chmod 777 ~/.trash
 fi

如上所示,先判断目录是否已建立,如未建立,即第一次运行该脚本,则创建 trash 目录。变量 realrm 存放了 Linux 的 rm 脚本位置,用于在特定条件下调用以直接删除文件或目录。

1.2 输出帮助信息

该脚本在用户仅输入脚本名而未输入参数执行时,输出简要帮助信息,代码如下:

if [ $# -eq 0 ]
  then
      echo "用法:delete file1 [file2 file3....]"
      echo "如果使用 -f 参数,则直接调用 rm -f 命令执行删除操作。"

如代码所示,该脚本的运用格式是 delete 后跟要删除的文件或目录的路径,中间用空格隔开。

1.3 直接删除文件

有些用户确认失效并想直接删除的文件,不应放入回收站中,而应直接从硬盘中删除。Delete 脚本提供了 -f 选项来执行这项操作:

while getopts "dfiPRrvW" opt
      do
        case $opt in
            f)
               exec $realrm "$@"
                ;;
            *)
               # do nothing
                ;;
        esac
      done

如果用户在命令中加入了 -f 选项,则 delete 脚本会直接调用 rm 命令将文件或目录直接删除。如代码中所示,所有的参数包括选项都会传递给 rm 命令。所以只要选项中包括选项 -f 就等于调用 rm 命令,可以使用 rm 的所有功能。如:delete –rfv filename 等于 rm –rfv filename。

1.4 用户交互

需要与用户确认是否将文件放入回收站。相当于 Windows 的弹窗提示,防止用户误操作。

echo -ne "你确定要把文件移到回收站吗?[Y/N]:\a"
 read reply
if [ $reply = "y" -o $reply = "Y" ]
  then #####

1.5 判断文件类型并直接删除大于 2G 文件

本脚本只对普通文件和目录做操作,其他类型文件不做处理。先对每个参数做循环,判断他们的类型,对于符合的类型再判断他们的大小是否超过 2G,如果是则直接从系统中删除,避免回收站占用太大的硬盘空间。

for file in $@
 do
if [ -f "$file" –o –d "$file" ]
then
if [ -f "$file" ] && [ `ls –l $file|awk '{print $5}'` -gt 2147483648 ]
   then
      echo "$file 的大小超过 2G,将直接从硬盘删除。"
        `rm –rf $file`
elif [ -d "$file" ] && [ `du –sb $file|awk '{print $1}'` -gt 2147483648 ]
   then
      echo "文件夹 $file 大小超过 2G,将直接从硬盘删除。"
        `rm –rf $file`

如以上代码所示,该脚本用不同的命令分别判断目录和文件的大小。鉴于目录的大小应该是包含其中的文件以及子目录的总大小,所以运用了'du -sb'命令。两种情况都使用了 awk 来获取特定输出字段的值来作比较。

1.6 移动文件到回收站并做记录

该部分是 Delete 脚本的主要部分,主要完成以下几个功能:

  • 获取参数的文件名。因为用户指定的参数中可能包含路径,所以要从中获取到文件名,用来生成 mv 操作的参数。该脚本中运用了字符串正则表达式 '${file##*/}'` 来获取。
  • 生成新文件名。在原文件名中加上日期时间后缀以生成新的文件名,这样用户在浏览回收站时,对于每个文件的删除日期即可一目了然。
  • 生成被删文件的绝对路径。为了以后可能对被删文件进行的恢复操作,要从相对路径生成绝对路径并记录。用户输入的参数可能有三种情况:只包含文件名的相对路径,包含点号的相对路径以及绝对路径,脚本中用字符串处理对三种情况进行判断,并进行相应的处理。
  • 调用 logTrashDir 脚本,将回收站中的新文件名、原文件名、删除时间、原文件绝对路径记录到隐藏文件中
  • 将文件通过 mv 命令移动到 Trash 目录下。

详细代码如下所示:

now=`date +%Y%m%d_%H_%M_%S`
filename="${file##*/}"
newfilename="${file##*/}_${now}"
mark1="."
mark2="/"
if  [ "$file" = ${file/$mark2} ]
 then
  fullpath="$(pwd)/$file"
elif [ "$file" != ${file/$mark1} ]
 then
  fullpath="$(pwd)${file/$mark1}"
else
  fullpath="$file"
fi      
echo "该文件绝对路径是:$fullpath"
if mv -f $file ~/.trash/$newfilename
 then
  $(/logTrashDir "$newfilename $filename $now $fullpath")
   echo "$file 已经删除。"
 else
  echo "操作失败。"
 fi

2. logTrashDir 脚本

该脚本较简单,仅是一个简单的文件写入操作,之所以单独作为一个脚本,是为了以后扩展的方便,具体代码如下:

if [ ! -f ~/.trash/.log ]
  then
     touch ~/.trash/.log
     chmod 700~/.trash/.log
fi
   echo $1 $2 $3 $4>> ~/.trash/.log

该脚本先建立 .log 隐藏文件,然后往里添加删除文件的记录。


3. restoreTrash 脚本

该脚本主要完成以下功能:

  • 从 .log 文件中找到用户想要恢复的文件对应的记录。此处依然使用 awk,通过正表达式匹配找到包含被删除文件名的一行。
  • 从记录中找到记录原文件名的字段,以给用户提示。
  • 将回收站中的文件移动到原来的位置,在这里运用了 mv –b 移动文件,之所以加入 - b 选项是为了防止原位置有同名文件的情况。
  • 将 .log 文件中与被恢复文件相对应的记录删除。

3.1 获取相应记录

originalPath=$(awk /$filename/'{print $4}' "$HOME/.trash/.log")

3.2 查找原文件名及现文件名字段

filenameNow=$(awk /$filename/'{print $1}' ~/.trash/.log)
filenamebefore=$(awk /$filename/'{print $2}' ~/.trash/.log)
echo "你想恢复的文件为 $filenameNow,原文件名字是 $filenamebefore"
echo "原文件的路径是:$originalPath"

3.3 恢复文件到原来位置并删除相应记录

echo "你确定要执行这个动作吗?[Y/N]"
  read reply
  if [ $reply = "y" ] || [ $reply = "Y" ]
   then
$(mv -b "$HOME/.trash/$filename" "$originalPath")
$(sed -i /$filename/'d' "$HOME/.trash/.log")
  else
    echo "没有文件被恢复。"
  fi

4. 自动定期清理 trash 目录

因为 delete 操作并不是真正删除文件,而是移动操作,经过一段时间的积累,trash 目录可能会占用大量的硬盘空间,造成资源浪费,所以定期自动清理 trash 目录下的文件是必须得。本文的清理规则是:在回收站中存在 7 天以上的文件及目录将会被自动从硬盘中删除。运用的工具是 Linux 自带的 crontab。

Crontab 是 Linux 用来定期执行程序的命令。当安装完成操作系统之后,默认便会启动此任务调度命令。Crontab 命令会定期检查是否有要执行的工作,如果有要执行的工作便会自动执行该工作。而 Linux 任务调度的工作主要分为以下两类:

  1. 系统执行的工作:系统周期性所要执行的工作,如备份系统数据、清理缓存。
  2. 个人执行的工作:某个用户定期要做的工作,例如每隔 10 分钟检查邮件服务器是否有新信,这些工作可由每个用户自行设置。

首先编写 crontab 执行时要调用的脚本 cleanTrashCan. 如清单 10 所示,该脚本主要完成两项功能:

  • 判断回收站中的文件存放时间是否已超过 7 天,如果超过则从回收站中删除。
  • 将删除文件在 .log 文件中相应的记录删除,保持其中数据的有效性,提高查找效率。

删除存在回收站超过 7 天的文件并删除 .log 中相应记录

arrayA=($(find ~/.trash/* -mtime +7 | awk '{print $1}'))
   for file in ${arrayA[@]}
    do
      $(rm -rf "${file}")
      filename="${file##*/}"
      echo $filename
      $(sed -i /$filename/'d' "$HOME/.trash/.log")
    done

脚本编写完成后通过 chmod 命令赋予其执行权限,然后运过 crontab –e 命令添加一条新的任务调度:

 10 18 * * * /bin/cleanTrashCan

该语句的含义为,在每天的下午 6 点 10 分执行 cleanTrashCan 脚本。
通过这条任务调度,trash 的大小会得到有效的控制,不会持续增大以致影响用户的正常操作。

5. 实际应用

首先要将 delete、logTrashDir、restoreTrash 和 cleanTrashCan 放到 /bin 目录下,然后用 chmod +x delete restoreTrash logTrashDir cleanTrashCan 命令赋予这三个脚本可执行权限。

运用 delete 脚本删除文件,例如要删除在 /usr 目录下的 useless 文件。根据用户目前所在的位置,可以用相对路径或绝对路径来指定参数,如:

  • delete useless
  • delete ./useless
  • delete /usr/useless

执行之后,useless 文件会从原目录中删除,被移动到 $HOME/.trash 下,并被重命名。同时生成的 .log 记录。

如果用户在七天之内发现该文件还有使用价值,则可以使用 restoreTrash 命令将被删除文件恢复到原路径下:restoreTrash ~/trash/useless_20140923_06_28_57


6. 总结

本文仿照 Windows 中回收站的功能,在 Linux 中做了实现,可以有效的防止由于误删而造成的损失。读者只需要将四个脚本拷到 /bin 目录下,并配置 crontab 即可使用 Linux 版回收站。


参考资料:


整合版地址:Github Gist

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容

  • Ubuntu的发音 Ubuntu,源于非洲祖鲁人和科萨人的语言,发作 oo-boon-too 的音。了解发音是有意...
    萤火虫de梦阅读 99,126评论 9 467
  • linux资料总章2.1 1.0写的不好抱歉 但是2.0已经改了很多 但是错误还是无法避免 以后资料会慢慢更新 大...
    数据革命阅读 12,130评论 2 34
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,590评论 18 139
  • 我怕谁呀,我大哥是耶酥啊! 出来混,迟早要还的 有错就要认!挨打就立正! 黑社会怎么了?出了事我自己抗。 耽误人办...
    澜润她世界阅读 265评论 0 0
  • #好婆婆胜过好老公# 其实一直想分享一下我的感受,在我们家,婆媳关系大于夫妻关系。 凡是我的事情,老公不一定理解的...
    春雨小溪阅读 281评论 0 0