第二章:Improving On User Commands--16.使用已刪除的文檔

   既然有了一個被刪除文件的目錄,並且這個目錄是在用戶的家目錄下的隱藏目錄,那麼一個讓用戶檢索這些已刪除文件的腳本的程序就會很是有用了。無論怎樣,想要代表全部可能發生的狀況是很難的,由於它包括了沒有匹配、僅有一個匹配以及多個匹配三種狀況。在多個匹配的情形中,好比,你是想要挑出最新的文件,而後將它還原?仍是指明有多少種狀況匹配成功,而後就退出?又或是展現下不一樣狀況的數據而後供用戶挑選?下面,讓咱們來看看咱們到底都能作些什麼... git

#!/bin/sh
 
 # unrm.sh -- 查找已刪除文檔中的給定文件或是目錄
 # 若是有多個匹配,那麼給出一個按時間戳排序的結果列表,
 # 而後,讓用戶指定還原哪一個
 
 mydir="$HOME/.deleted-files"
 realrm="/bin/rm"
 move="/bin/mv"
 
 dest=$(pwd)
 
 if [ ! -d $mydir ]; then
     echo "`basename $0`: No deleted files directory: nothing to unrm" >&2
     exit 1
 fi
 
 cd $mydir
 
 if [ $# -eq 0 ]; then
     echo "Contents of your deleted files archive(sorted by data):"    
     # ls中的-F是給列出來的項增長指示器,好比,文件不加後綴,目錄加斜槓/,可執行文件加星號*
     # ls中的-C是按照列顯示
     ls -FC | sed -e 's/\([[:digit:]][[:digit:]]\.\)\{5\}//g' \
     -e 's/^/ /'    # 這條替換的目的是給行頭加空格
     exit 0
 fi
 
 # 不然,咱們必須使用一個用戶指定的模式。
 # 讓咱們來看看該模式在文檔中是否匹配多個文件或是目錄
 
 matches="$(ls *"$1" 2> /dev/null | wc -l)"
 
 if [ $matches -eq 0 ]; then
     echo "No match for \"$1\" in the deleted file archive." >&2
     exit 1
 fi
 
 if [ $matches -gt 1 ]; then
     echo "More than one file or directory match in the archive:"
     index=1
     # ls中的-t是按照最近修改時間顯示
     # ls中的-d是隻顯示目錄的名字,而不是顯示目錄中的內容
     for name in $(ls -td *"$1")
     do
         datetime="$(echo $name | cut -c1-14 | \
             awk -F. '{print $5"/"$4" at "$3":"$2":"$1}')"  # 使用awk格式化輸出
         if [ -d $name ]; then
             size="$(ls $name | wc -l | sed 's/[^[:digit:]]//g')"    # 算出$name這個目錄中文件的數目
             echo "$index) $1 (contents = ${size} itmes, deleted = $datetime)"
         else
             size="$(ls -sdk1 $name | awk '{print $1}')"    # 計算文件的大小
             echo "$index) $1 (size = ${size}Kb, deleted = $datetime)"
         fi
         index=$(($index+1))
     done
     echo ""
     echo -n "Which version of $1 do you want to restore ('0' to quit)? [1]: "
     read desired
     if [ ${desired:=1} -ge $index ]; then
         echo "$(basename $0): Restore canceled by user: index value too big." >&2
         exit 1
     fi
 
     if [ $desired -lt 1 ]; then
         echo "$(basename $0): canceled by user." >& 2
         exit 1
     fi
 
     restore="$(ls -td1 *"$1" | sed -n "${desired}p")"
 
     if [ -e "$dest/$1" ]; then
         echo "\"$1\" already exists in this directory. Cannot overwrite." >&2
         exit 1
     fi
 
     echo -n "Restoring file \"$1\"..."
     $move "$restore" "$dest/$1"
     echo "done."
 
     echo -n "Delete the additional copies of this file? [y]: "
     read answer
 
     if [ ${answer:=y} = "y" ]; then
         $realrm -rf *"$1"
         echo "deleted"
     else
         echo "additional copies retained."
     fi
 else
     if [ -e "$dest/$1" ]; then
         echo "\"$1\" already exists in this directory. Cannot Overwrite." >&2
         exit 1
     fi
 
     restore="$(ls -d *"$1")"
 
     echo -n "Restoring file \"$1\"..."
     $move "$restore" "$dest/$1"
     echo "done."
 fi
 
 exit 0

腳本如何工做:
第一個大段的代碼,if [ $# -eq 0 ]條件語塊,會按此執行:若是沒有參數給定,那麼就列出刪除文檔中的全部內容。但這裏有個地方隱瞞了。咱們並不能展現真實的文件名,由於咱們並不想用戶看到時間戳,這些時間戳只是用來在內部保護文件名之間並不會相互衝突用的。爲了用一種更美觀方式展現文件名,sed表達式刪除了開始的5個數字段。若是給定了一個參數,它就是要恢復的文件名或是目錄了。下一步就是要查明有多少個能匹配給定的名稱。下面的語句完成了這個功能:
shell

matches="$(ls *"$1" 2> /dev/null | wc -l)"
在ls的參數中有對不經常使用的引號,它們是用來保證該模式會匹配到已嵌入空白的文件名,而通配符'*'則被shell適當的擴展了。而2> /dev/null保住了命令中產生的錯誤信息會被拋棄掉,不會讓它們顯示給用戶看到。丟棄的信息絕大部分有多是No such file or directory,通常都是因爲沒找到給定的文件名引發的。若是對給定的文件或是目錄名有多個匹配,最複雜的腳本部分,就是if [ $matches -gt 1 ]語塊。這個語塊,展現了全部的結果。在for循環中的ls命令中使用-t選項,會將文檔按照重新到舊排序顯示。而後的awk語句將文件名中的時間戳給分割了開來。下面的ls中內含的-k選項是用來計算文件大小時使用千字節(kb)做爲單位,而不是平時的字節。

size="$(ls -sdk1 $name | awk '{print $1}')"
腳本會在每一個匹配的目錄中顯示文件的數目,而不是毫無心義的只是顯示匹配文件項的大小。一個目錄中項的數目事實上很容易計算,使用wc命令:

size="$(ls $name | wc -l | sed 's/[^[:digit:]]//g')"
一旦用戶給定一種可能的匹配文件或是目錄,對應的擴展名就會被下面的語句定義好:

restore="$(ls -td1 *"$1" | sed -n "${desired}p")"
這個句子包含了一點sed的另類用法。使用-n選項,而後是一個跟着打印命令p的數字(${desired}),這種方法能夠很快的從輸入流中提取給定行號的行。
運行腳本:
有兩種方法調用腳本。第一種,無參數,它會顯示刪除文檔中全部的文件和目錄。第二種,有一個要求的文件或是目錄名做爲參數,腳本要麼會恢復該文件或是目錄(若是隻有一個匹配),要麼會顯示能夠恢復的全部的候選名單,這樣用戶就能夠自行選定一個。
運行結果:
無參數時,腳本會顯示全部的刪除文檔:
unrm.sh
Contents of your deleted files archive(sorted by data):
.a.txt  .a.o*  .adir/
有一個文件名作參數:

unrm.sh a.txt
 More than one file or directory match in the archive:
 1) a.txt (size = 4Kb, deleted = 11/18 at 17:15:45)
 2) a.txt (size = 4Kb, deleted = 11/18 at 17:15:10)
 
 Which version of a.txt do you want to restore ('0' to quit)? [1] 2
 Restoring file "a.txt"...done.
 Delete the additional copies of this file? [y] y
 deleted
分析腳本: 若是你執行這個腳本,那麼就有一個潛在的危險須要注意。沒有任何控制或是限制的話,在刪除文檔中的文件或是目錄會無限制增長。爲了不這點,能夠添加一個cronjob來減小文檔。保留14天內的文檔應當時比較合理的了。
相關文章
相關標籤/搜索