回收站功能在 Linux 中的實(shí)現(xiàn)
本文仿照 Windows 回收站的功能,運(yùn)用 Bash 腳本在 Linux 上做了實(shí)現(xiàn),創(chuàng)建 delete 腳本代替 rm 命令對(duì)文件或目錄進(jìn)行刪除操做。該腳本實(shí)現(xiàn)了以下功能:對(duì)大于 2G 的文件或目錄直接刪除,否則放入$HOME/trash 目錄下;恢復(fù) trash 目錄中的被刪除文件到原目錄下;文件存放在 trash 目錄中超過(guò)七天被自動(dòng)刪除。
概述
刪除是危險(xiǎn)系數(shù)很高的操作,一旦誤刪可能會(huì)造成難以估計(jì)的損失。在 Linux 系統(tǒng)中這種危險(xiǎn)尤為明顯,一條簡(jiǎn)單的語(yǔ)句:rm –rf /* 就會(huì)把整個(gè)系統(tǒng)全部刪除,而 Linux 并不會(huì)因?yàn)檫@條語(yǔ)句的不合理而拒絕執(zhí)行。 在 Windows 中,為了防止誤刪,系統(tǒng)提供了回收站功能。用戶(hù)在執(zhí)行刪除操作后,文件并不會(huì)直接從硬盤(pán)中刪除,而是被放到回收站中。在清空回收站前,如果發(fā)現(xiàn)有文件被誤刪,用戶(hù)可以將回收站中的文件恢復(fù)到原來(lái)的位置。而 Linux 并沒(méi)有提供類(lèi)似功能,刪除命令 rm 一旦確認(rèn)執(zhí)行,文件就會(huì)直接從系統(tǒng)中刪除,很難恢復(fù)。
回收站構(gòu)成
本文共用三個(gè)腳本實(shí)現(xiàn)了回收站的主要功能:Delete 腳本、logTrashDir 腳本和 restoreTrash 腳本。其中 Delete 腳本是核心腳本,其作用是重新封裝 rm 命令。相對(duì)于 rm 的直接刪除,該命令會(huì)先將文件或目錄移動(dòng)到$home/trash 目錄下。如果用戶(hù)想要將文件直接刪除,可以用 -f 選項(xiàng),delete 腳本會(huì)直接調(diào)用 rm –f 命令將文件從硬盤(pán)上刪除。logTrashDir 腳本用于將被刪除文件的信息記錄到 trash 目錄下的一個(gè)隱藏文件中。restoreTrash 腳本用來(lái)將放入 trash 中的文件或目錄重新恢復(fù)到原路徑下。在 Linux 系統(tǒng)中,只要將這三個(gè)腳本放到/bin/目錄下,并用 chmod +X filename 賦予可執(zhí)行權(quán)限,即可直接使用。下面將介紹每個(gè)腳本的主要部分
Delete 腳本
創(chuàng)建目錄
首先要?jiǎng)?chuàng)建目錄來(lái)存放被刪除的文件,本文在用戶(hù)根目錄$HOME 下建立 trash 目錄來(lái)存放文件。具體代碼如下:
清單 1.創(chuàng)建回收站目錄
realrm="/bin/rm"
if [ ! -d ~/trash ]
then
mkdir -v ~/trash
chmod 777 ~/trash
fi
如上所示,先判斷目錄是否已建立,如未建立,即第一次運(yùn)行該腳本,則創(chuàng)建 trash 目錄。變量 realrm 存放了 Linux 的 rm 腳本位置,用于在特定條件下調(diào)用以直接刪除文件或目錄。
輸出幫助信息
該腳本在用戶(hù)僅輸入腳本名而未輸入?yún)?shù)執(zhí)行時(shí),輸出簡(jiǎn)要幫助信息,代碼如下:
清單 2.輸出幫助信息
if [ $# -eq 0 ]
then
echo "Usage:delete file1 [file2 file3....]"
echo "If the options contain -f,then the script will exec ‘rm‘ directly"
如代碼所示,該腳本的運(yùn)用格式是 delete 后跟要?jiǎng)h除的文件或目錄的路徑,中間用空格隔開(kāi)。
直接刪除文件
有些用戶(hù)確認(rèn)失效并想直接刪除的文件,不應(yīng)放入回收站中,而應(yīng)直接從硬盤(pán)中刪除。Delete 腳本提供了-f 選項(xiàng)來(lái)執(zhí)行這項(xiàng)操作:
清單 3.直接刪除文件
while getopts "dfiPRrvW" opt
do
case $opt in
f)
exec $realrm "$@"
;;
*)
# do nothing
;;
esac
done
如果用戶(hù)在命令中加入了-f 選項(xiàng),則 delete 腳本會(huì)直接調(diào)用 rm 命令將文件或目錄直接刪除。如代碼中所示,所有的參數(shù)包括選項(xiàng)都會(huì)傳遞給 rm 命令。所以只要選項(xiàng)中包括選項(xiàng)-f 就等于調(diào)用 rm 命令,可以使用 rm 的所有功能。如:delete –rfv filename 等于 rm –rfv filename。
用戶(hù)交互
需要與用戶(hù)確認(rèn)是否將文件放入回收站。相當(dāng)于 Windows 的彈窗提示,防止用戶(hù)誤操作。
清單 4.用戶(hù)交互
echo -ne "Are you sure you want to move the files to the trash?[Y/N]:a"
read reply
if [ $reply = "y" -o $reply = "Y" ]
then #####
判斷文件類(lèi)型并直接刪除大于 2G 文件
本腳本只對(duì)普通文件和目錄做操作,其他類(lèi)型文件不做處理。先對(duì)每個(gè)參數(shù)做循環(huán),判斷他們的類(lèi)型,對(duì)于符合的類(lèi)型再判斷他們的大小是否超過(guò) 2G,如果是則直接從系統(tǒng)中刪除,避免回收站占用太大的硬盤(pán)空間。
清單 5.刪除大于 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 size is larger than 2G,will be deleted directly"
`rm –rf $file`
elif [ -d "$file" ] && [ `du –sb $file|awk ‘{print $1}‘` -gt 2147483648 ]
then
echo "The directory:$file is larger than 2G,will be deleted directly"
`rm –rf $file`
如以上代碼所示,該腳本用不同的命令分別判斷目錄和文件的大小。鑒于目錄的大小應(yīng)該是包含其中的文件以及子目錄的總大小,所以運(yùn)用了’du -sb’命令。兩種情況都使用了 awk 來(lái)獲取特定輸出字段的值來(lái)作比較。
移動(dòng)文件到回收站并做記錄
該部分是 Delete 腳本的主要部分,主要完成以下幾個(gè)功能
獲取參數(shù)的文件名。因?yàn)橛脩?hù)指定的參數(shù)中可能包含路徑,所以要從中獲取到文件名,用來(lái)生成 mv 操作的參數(shù)。該腳本中運(yùn)用了字符串正則表達(dá)式’${file##*/}’來(lái)獲取。
生成新文件名。在原文件名中加上日期時(shí)間后綴以生成新的文件名,這樣用戶(hù)在瀏覽回收站時(shí),對(duì)于每個(gè)文件的刪除日期即可一目了然。
生成被刪文件的絕對(duì)路徑。為了以后可能對(duì)被刪文件進(jìn)行的恢復(fù)操作,要從相對(duì)路徑生成絕對(duì)路徑并記錄。用戶(hù)輸入的參數(shù)可能有三種情況:只包含文件名的相對(duì)路徑,包含點(diǎn)號(hào)的相對(duì)路徑以及絕對(duì)路徑,腳本中用字符串處理對(duì)三種情況進(jìn)行判斷,并進(jìn)行相應(yīng)的處理。[!--empirenews.page--]
調(diào)用 logTrashDir 腳本,將回收站中的新文件名、原文件名、刪除時(shí)間、原文件絕對(duì)路徑記錄到隱藏文件中
將文件通過(guò) mv 命令移動(dòng)到 Trash 目錄下。詳細(xì)代碼如下所示:
清單 6.移動(dòng)文件到回收站并做記錄
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 "the full path of this file is :$fullpath"
if mv -f $file ~/trash/$newfilename
then
$(/logTrashDir "$newfilename $filename $now $fullpath")
echo "files: $file is deleted"
else
echo "the operation is failed"
fi
logTrashDir 腳本
該腳本較簡(jiǎn)單,僅是一個(gè)簡(jiǎn)單的文件寫(xiě)入操作,之所以單獨(dú)作為一個(gè)腳本,是為了以后擴(kuò)展的方便,具體代碼如下:
清單 7.logTrashDir 代碼
if [ ! -f ~/trash/.log ]
then
touch ~/trash/.log
chmod 700~/trash/.log
fi
echo $1 $2 $3 $4>> ~/trash/.log
該腳本先建立.log 隱藏文件,然后往里添加刪除文件的記錄。
restoreTrash 腳本
該腳本主要完成以下功能:
從.log 文件中找到用戶(hù)想要恢復(fù)的文件對(duì)應(yīng)的記錄。此處依然使用 awk,通過(guò)正表達(dá)式匹配找到包含被刪除文件名的一行
從記錄中找到記錄原文件名的字段,以給用戶(hù)提示
將回收站中的文件移動(dòng)到原來(lái)的位置,在這里運(yùn)用了 mv –b 移動(dòng)文件,之所以加入-b 選項(xiàng)是為了防止原位置有同名文件的情況。
將.log 文件中與被恢復(fù)文件相對(duì)應(yīng)的記錄刪除
清單 8.獲取相應(yīng)記錄
1originalPath=$(awk /$filename/‘{print $4}‘ "$HOME/trash/.log")
清單 9.查找原文件名及現(xiàn)文件名字段
filenameNow=$(awk /$filename/‘{print $1}‘ ~/trash/.log)
filenamebefore=$(awk /$filename/‘{print $2}‘ ~/trash/.log)
echo "you are about to restore $filenameNow,original name is $filenamebefore"
echo "original path is $originalPath"
清單 10.恢復(fù)文件到原來(lái)位置并刪除相應(yīng)記錄
echo "Are you sure to do that?[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 "no files restored"
fi
自動(dòng)定期清理 trash 目錄
因?yàn)?delete 操作并不是真正刪除文件,而是移動(dòng)操作,經(jīng)過(guò)一段時(shí)間的積累,trash 目錄可能會(huì)占用大量的硬盤(pán)空間,造成資源浪費(fèi),所以定期自動(dòng)清理 trash 目錄下的文件是必須得。本文的清理規(guī)則是:在回收站中存在 7 天以上的文件及目錄將會(huì)被自動(dòng)從硬盤(pán)中刪除。運(yùn)用的工具是 Linux 自帶的 crontab。
Crontab 是 Linux 用來(lái)定期執(zhí)行程序的命令。當(dāng)安裝完成操作系統(tǒng)之后,默認(rèn)便會(huì)啟動(dòng)此任務(wù)調(diào)度命令。Crontab 命令會(huì)定期檢查是否有要執(zhí)行的工作,如果有要執(zhí)行的工作便會(huì)自動(dòng)執(zhí)行該工作。而 Linux 任務(wù)調(diào)度的工作主要分為以下兩類(lèi):
1、系統(tǒng)執(zhí)行的工作:系統(tǒng)周期性所要執(zhí)行的工作,如備份系統(tǒng)數(shù)據(jù)、清理緩存
2、個(gè)人執(zhí)行的工作:某個(gè)用戶(hù)定期要做的工作,例如每隔 10 分鐘檢查郵件服務(wù)器是否有新信,這些工作可由每個(gè)用戶(hù)自行設(shè)置。
首先編寫(xiě) crontab 執(zhí)行時(shí)要調(diào)用的腳本 cleanTrashCan.如清單 10 所示,該腳本主要完成兩項(xiàng)功能:
判斷回收站中的文件存放時(shí)間是否已超過(guò) 7 天,如果超過(guò)則從回收站中刪除。
將刪除文件在.log 文件中相應(yīng)的記錄刪除,保持其中數(shù)據(jù)的有效性,提高查找效率。
清單 11.刪除存在回收站超過(guò) 7 天的文件并刪除.log 中相應(yīng)記錄
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
腳本編寫(xiě)完成后通過(guò) chmod 命令賦予其執(zhí)行權(quán)限,然后運(yùn)過(guò) crontab –e 命令添加一條新的任務(wù)調(diào)度:
110 18 * * * /bin/ cleanTrashCan
該語(yǔ)句的含義為,在每天的下午 6 點(diǎn) 10 分執(zhí)行 cleanTrashCan 腳本
通過(guò)這條任務(wù)調(diào)度,trash 的大小會(huì)得到有效的控制,不會(huì)持續(xù)增大以致影響用戶(hù)的正常操作。
實(shí)際應(yīng)用
首先要將 delete 腳本,logTrashDir 腳本,restoreTrash 腳本和 cleanTrashCan 放到/bin 目錄下,然后用 chmod +x delete restoreTrash logTrashDir cleanTrashCan 命令賦予這三個(gè)腳本可執(zhí)行權(quán)限。
運(yùn)用 delete 腳本刪除文件,例如要?jiǎng)h除在/usr 目錄下的 useless 文件。根據(jù)用戶(hù)目前所在的位置,可以用相對(duì)路徑或絕對(duì)路徑來(lái)指定參數(shù),如:delete useless,delete ./useless 或者 delete /usr/useless。執(zhí)行過(guò)程如圖 1 所示:
圖 1.delete 腳本執(zhí)行過(guò)程
執(zhí)行之后,useless 文件會(huì)從原目錄中刪除,被移動(dòng)到$HOME/trash 下,并被重命名,如圖 2.所示:[!--empirenews.page--]
圖 2.回收站目錄
生成的.log 記錄如圖 3.所示:
圖 3.log 記錄
如果用戶(hù)在七天之內(nèi)發(fā)現(xiàn)該文件還有使用價(jià)值,則可以使用 restoreTrash 命令將被刪除文件恢復(fù)到原路徑下:restoreTrash ~/trash/useless_20140923_06_28_57。具體執(zhí)行情況如圖 4 所示:
圖 4.restoreTrash 腳本執(zhí)行情況
查看/usr 目錄,可以發(fā)現(xiàn) useless 文件已經(jīng)被恢復(fù)至此。
圖 5.useless 文件被恢復(fù)
總結(jié)
本文仿照 Windows 中回收站的功能,在 Linux 中做了實(shí)現(xiàn),可以有效的防止由于誤刪而造成的損失。讀者只需要將四個(gè)腳本拷到/bin 目錄下,并配置 crontab 即可使用 Linux 版回收站。