對於喜愛自動化的Linux系統管理員而言,一定是用過expect這個命令行工具。Expect 是由 Don Libes 基於 Tcl 語言開發的,並被廣泛應用於交互式操作和自動化測試的場景之中,它尤其適用於需要對多台服務器執行相同操作的環境中,可以大幅度提高系統管理人員的工作效率。本文是thegeekstuff.com最近更新的一篇技術分享文章,其中詳細講述了如何通過不同的命令行選項來執行一個expect腳本,具體有什麼用,大家可以自由發揮想象力。
本文假設您對expect的基本使用方法已經有一定的了解。
如果你是expect腳本語言的新手,可以首先從我們的expect的“hello world”樣例(英文)開始。
1,使用“-c”選項,從命令行執行expect腳本
expect可以讓你使用“-c”選項,直接在命令行中執行它,如下所示:
$ expect -c 'expect "\n" {send "pressed enter\n"}
pressed enter
$
如果你執行了上面的腳本,它會等待輸入換行符(\n)。按“enter”鍵以後,它會打印出“pressed enter”這個消息,然後退出。
2,使用“-i”選項交互地執行expect腳本
使用“-i”選項,可以通過來自於標准輸入的讀命令來交互地執行expect腳本。如下所示:
$ expect -i arg1 arg2 arg3
expect1.1>set argv
arg1 arg2 arg3
expect1.2>
正常情況下,當你執行上面的expect命令的時候(沒有“-i”選項),它會把arg1當成腳本的文件名,所以“-i”選項可以讓腳本把多個參數當成一個連續的列表。
當你執行帶有“-c”選項的expect腳本的時候,這個選項是十分有用的。因為默認情況下,expect是交互地執行的。
3,當執行expect腳本的時候,輸出調試信息
當你用“-d”選項執行代碼的時候,你可以輸出診斷的信息。如下所示:
$ cat sample.exp
# !/usr/bin/expect -fexpect "\n";send "pressed enter";$ expect -d sample.expexpect version 5.43.0argv[0] = expect argv[1] = -d argv[2] = sample.expset argc 0set argv0 "sample.exp"set argv ""executing commands from command file sample.exp
expect: does "" (spawn_id exp0) match glob pattern "\n"? no
expect: does "\n" (spawn_id exp0) match glob pattern "\n"? yes
expect: set expect_out(0,string) "\n"
expect: set expect_out(spawn_id) "exp0"
expect: set expect_out(buffer) "\n"
send: sending "pressed enter" to { exp0 pressed enter}
4,使用“-D”選項啟動expect調試器
“-D”選項用於啟動調試器,它只接受一個布爾值的參數。這個參數表示提示器必須馬上啟動,還是只是初始化調試器,以後再使用它。
$ expect -D 1 script
“-D”選項左邊的選項會在調試器啟動以前被處理。然後,在調試器啟動以後,剩下的命令才會被執行。
$ expect -c 'set timeout 10' -D 1 -c 'set a 1'
1: set a 1
dbg1.0>
5,逐行地執行expect腳本
通常,expect會在執行腳本之前,把整個腳本都讀入到內存中。“-b”選項可以讓expect一次只讀取腳本中的一行。當你沒有寫完整個腳本的時候,這是十分有用的,expect可以開始執行這個不完整的腳本,並且,它可以避免把腳本寫入到臨時文件中。
$ expect -b
6,讓expect不解釋命令行參數
你可以使用標識符讓expect不解釋命令行參數。
你可以像下面這樣的讀入命令行參數:
$ cat print_cmdline_args.exp
#!/usr/bin/expect
puts 'argv0 : [lindex $argv 0]';
puts 'argv1 : [lindex $argv 1]';
當執行上面的腳本的時候,會跳過命令行選項,它們會被當成參數(而不是expect選項),如下所示:
$ expect print_cmdline_args.exp -d -c
argv0 : -d
argv1 : -c