在 Linux 系統中,當你輸入一個命令,再按兩次 TAB 鍵,就會列出所有以你輸入字符開頭的可用命令。這並不新鮮,可能你已經知道了。這個功能被稱作命令行補全bash completion。默認情況下,bash 命令行可以自動補全文件或目錄名稱。不過,我們可以增強 bash 命令補全功能,通過 complete 命令讓它達到新的高度。
這個教程說明了我們是怎樣使用可編程的命令行補全功能(programmable completion)把自動補全功能應用於選項或者命令行參數。
例如:在輸入 write 命令之後,如果你按兩次 TAB 按鍵,自動補全功能會提供可供執行 write 操作的用戶列表。
$ write [TAB][TAB]
bala raj
jason randy
john ritu
mayla thomas
nisha www-data
在下面的例子中,可以為 telnet 命令顯示可用的主機名:
$ telnet [TAB][TAB]
localhost dev-db fileserver
要讓可編程命令補全功能在你的終端起作用 ,你只需要如下執行/etc/bash_completion即可:
# . /etc/bash_completion
你也可以取消/etc/bash.bashrc(來自 Ubuntu Linux 13.04 系統)中如下的注釋,這樣,你就可以不需要執行上面的命令了:
### enable bash completion in interactive shells
if ! shopt -oq posix; then
if [ -f /usr/share/bash-completion/bash_completion ]; then
. /usr/share/bash-completion/bash_completion
elif [ -f /etc/bash_completion ]; then
. /etc/bash_completion
fi
fi
如果你沒有發現這些代碼,也沒有找到/etc/bash_completion文件,那麼你只需要通過使用apt-get命令來安裝bash_completion 包即可。
1. 查看已有的命令行補全
在啟用可編程的命令行補全功能後,就已經有了一些定義好的命令補全功能。complete 命令用於定義命令行補全。
要查看已有的命令行補全,如下使用 complete 命令:
complete -p | less
上面例子中的 -p 選項是可選的。
2. 列出 bash 中標准補全功能
程序員高效開發利器:編程水杯
程序員高效開發利器:編程水杯
默認情況下,Bash 為 Linux 用戶提供了下列標准補全功能。
1.變量補全
2.用戶名補全
3.主機名補全
4.路徑補全
5.文件名補全
我們在之前的 bash 標准補全中討論過這些。
3. 定義一個命令名補全
通過 -c 選項可以將所有的可用命令作為一個命令的補全參數。在下面的例子裡面,為 which 命令定義了一個補全(LCTT譯注:在按兩下 TAB 時,可以列出所有命令名作為可補全的參數)。
$ complete -c which
$ which [TAB][TAB]
Display all 2116 possibilities? (y or n)
如上,如果按下 ‘y’,就會列出所有的命令名。
4. 定義一個目錄補全
通過選項 -d,可以定義一個僅包含目錄名的補全參數。在下面的例子中,為 ls 命令定義了補全。
$ ls
countfiles.sh dir1/ dir2/ dir3/
$ complete -d ls
$ ls [TAB][TAB]
dir1/ dir2/ dir3/
如上,連按下 TAB 僅會顯示目錄名。
5. 定義一個後台任務名補全
補全功能也能夠以任務名作為補全參數。選項 -j 可以定義任務名作為傳遞給命令的參數,如下:
$ jobs
[1]- Stopped cat
[2]+ Stopped sed 'p'
$ complete -j ./list_job_attrib.sh
$ ./list_job_attrib.sh [TAB][TAB]
cat sed
關於後台任務,你可以參考 Linux 後台任務中的例子了解如何管理後台任務。
6. 帶有前綴和後綴的補全
補全功能可以為實際的補全內容定義前綴和後綴。在下面的例子中,為 list_job_attrib.sh 定義了補全內容的前綴和後綴。
$ jobs
[1]+ Stopped cat
$ complete -P '">' -S '<"' ./list_job_attrib.sh
$ ./list_job_attrib.sh [TAB][TAB]
$ ./list_job_attrib.sh ">cat<"
7. 帶有排除的文件名和目錄名補全
假如腳本運行完成後,輸出目錄如下:
$ cd output/
$ ls
all_calls.txt incoming_calls.txt outgoing_calls.txt missed_calls.txt
parser_mod.tmp extract.o
如上,如果你想要 ls 命令的補全忽略 .tmp 和 .o 文件:
$ export FIGNORE='.tmp:.o'
$ complete -f -d ls
$ cd output
$ ls [TAB][TAB]
all_calls.txt incoming_calls.txt outgoing_calls.txt missed_calls.txt
FIGNORE 是一個環境變量,它包含了自動補全所需要排除的文件名後綴。
8. 通過 IFS 變量分割字符串得到補全值
可以通過 -W 選項定義補全值列表,然後通過 IFS 環境變量進行切分。切分結果會展開變量並作為補全顯示。
$ export IFS=" "
$ complete -W "bubble quick" ./sort_numbers.sh
$ ./sort_numbers.sh [TAB][TAB]
bubble quick
如上所述,字符串通過 IFS 分隔符進行切分後,內嵌的變量會被展開為變量值,所以可以如下使用變量:
$ echo $SORT_TYPE1
bubble
$ echo $SORT_TYPE2
quick
$ complete -W "$SORT_TYPE1 $SORT_TYPE2" ./sort_numbers.sh
$ ./sort_numbers.sh [TAB][TAB]
bubble quick
9. 寫個函數來生成補全
你可以引入一個函數來定義補全。使用 -F 選項將函數名傳給 complete 命令,執行函數生成補全內容。例如,函數如下:
_parser_options()
{
local curr_arg;
curr_arg=${COMP_WORDS[COMP_CWORD]}
COMPREPLY=( $(compgen -W '-i --incoming -o --outgoing -m --missed' -- $curr_arg ) );
}
在上述函數中:
1.COMPREPLY : 該數組控制連按下 TAB 後顯示的結果
2.COMP_WORDS : 該數組包含命令行輸入的單詞
3.COMP_CWORD : COMP_WORDS 數組的索引,使用它來區分命令行可以訪問的單詞位置
4.compgen : -W 基於 $current_arg 提供可能的補全及其參數
該函數放在 parser_option 文件中,並通過 source 命令引入:
$ source parser_option
將該函數和你的 parser.pl 腳本關聯起來:
$ complete -F _parser_options ./parser.pl
$ ./parser.pl [TAB][TAB]
-i --incoming -o --outgoing -m --missed
如上,parser.pl 的選項是由函數 _parser_options() 生成的。
提示: 查看/etc/bash_completion 來了解更多的可編程補全函數。
10. 當第一個規則沒有生成結果時,就使用第二個
如果定義的補全規則沒有生成匹配時,可以使用 -o 選項生成補全。
$ complete -F _count_files -o dirnames ./countfiles.sh
如上,為 ./countfiles.sh 定義了 _count_files 補全函數。 如果 the _count_files() 函數沒有生成任何匹配的話,就會觸發目錄補全。
$ ls
countfiles.sh dir1/ dir2/ dir3/
$./countfiles.sh [TAB][TAB]
dir1 dir2 dir3