需求背景
Mac下沒有XShell這樣好用的工具,所以需要經常性操作Linux伺服器的我需要有一款趁手的工具。日常操作的需求基本是這幾個
- 在iTerm2環境下使用(Mac上這是最好用的終端吧)
- 支援使用rz和sz命令上傳下載(為啥不是ftp?因為跳板機環境下不能使用)
- 可以管理使用者名稱和密碼(密碼太複雜記不住)
- 支援使用pem格式的金鑰檔案(登入跳板機都是用密碼+金鑰的)
谷歌了很多方案,主流的是採用expect進行響應式操作。這個方法挺好的,非常靈活並且可以在跳板機環境下多次響應後直接登入。不過我在使用的過程中發現不能配合sz和rz使用,不知道是不是我的RPWT,反正我使用expect登入伺服器之後,再使用sz/rz就進入假死狀態。所以我最後選了另外一個叫autossh的工具,這個工具可以實現Session管理(使用者,密碼,金鑰),但是不能響應式操作。
優點
- 可以免輸密碼登入伺服器
- 配合rz/sz毫無違和感
缺點
- 如果是跳板機這種需要繼續選擇機器和二次輸入密碼的場景,就只能登入到跳板機介面後手工操作
在iTerm下配置rz/sz
Mac上iTerm原生不支援rz/sz命令,也就是不支援Zmodem來進行檔案傳輸,不過只要通過簡單的配置就可以實現。網上的教程一大把,這裡就簡單的記錄一下過程。
安裝lrzsz
首先安裝Homebrew(這裡不寫這個過程),然後通過它先給Mac安裝lrzsz。在終端下輸入brew install lrzsz,靜等一會即可安裝完畢。
brew install lrzsz
複製程式碼
下載iTerm2輔助檔案
iTerm不能直接使用lrzsz,不過網上有大神提供了兩個輔助指令碼。我們只需要把檔案下載到 /usr/local/bin/目錄下並賦予可執行許可權即可。
cd /usr/local/bin
wget https://raw.githubusercontent.com/mmastrac/iterm2-zmodem/master/iterm2-send-zmodem.sh
wget https://raw.githubusercontent.com/mmastrac/iterm2-zmodem/master/iterm2-recv-zmodem.sh
chmod +x iterm2-recv-zmodem.sh iterm2-send-zmodem.sh
複製程式碼
這兩個指令碼實際是使用AppleScript來彈出檔案選擇視窗,然後把選中的檔名稱傳遞給rzsz命令。我們開啟其中一個看下程式碼。如果這一部分看不懂沒關係,直接跳過即可,對後續的配置使用沒有任何不良影響 :-D
#!/bin/bash
# Author: Matt Mastracci (matthew@mastracci.com)
# AppleScript from http://stackoverflow.com/questions/4309087/cancel-button-on-osascript-in-a-bash-script
# licensed under cc-wiki with attribution required
# Remainder of script public domain
osascript -e 'tell application "iTerm2" to version' > /dev/null 2>&1 && NAME=iTerm2 || NAME=iTerm
if [[ $NAME = "iTerm" ]]; then
FILE=`osascript -e 'tell application "iTerm" to activate' -e 'tell application "iTerm" to set thefile to choose folder with prompt "Choose a folder to place received files in"' -e "do shell script (\"echo \"&(quoted form of POSIX path of thefile as Unicode text)&\"\")"`
else
FILE=`osascript -e 'tell application "iTerm2" to activate' -e 'tell application "iTerm2" to set thefile to choose folder with prompt "Choose a folder to place received files in"' -e "do shell script (\"echo \"&(quoted form of POSIX path of thefile as Unicode text)&\"\")"`
fi
if [[ $FILE = "" ]]; then
echo Cancelled.
# Send ZModem cancel
echo -e \\x18\\x18\\x18\\x18\\x18
sleep 1
echo
echo \# Cancelled transfer
else
cd "$FILE"
/usr/local/bin/rz -E -e -b
sleep 1
echo
echo
echo \# Sent \-\> $FILE
fi
複製程式碼
配置iTerm2觸發器
這一步最關鍵,是在iTerm裡面配置觸發器,當監控到特定字串的時候執行剛才下載的兩個檔案。為了使用方便,我專門建立了一個Profile配置,名字是Remote,並且配合後面的autossh使用。
開啟iTerm2 -> Preferences -> Profiles 選擇 Advanced 設定 Triggers ,點選 Edit
在彈出視窗中進行如下配置,最後的Instant一定要勾選上。
配置的具體內容在這裡
Regular expression: rz waiting to receive.\*\*B0100
Action: Run Silent Coprocess
Parameters: /usr/local/bin/iterm2-send-zmodem.sh
Regular expression: \*\*B00000000000000
Action: Run Silent Coprocess
Parameters: /usr/local/bin/iterm2-recv-zmodem.sh
複製程式碼
重新啟動iTerm之後,rz/sz就應該可以正常使用了。
autossh
如前文所說,當使用expect方案的時候rz/sz不能彈出檔案選擇視窗。我猜可能是expect啟動的session上下文中不能正常執行AppleScript造成的,因為對這兩個玩意兒不是很熟,所以也就沒有辦法深入研究和折騰。各種谷歌後,發現這個好東西 https://github.com/islenbo/autossh ,優秀的產品總是很簡單的,就一個可執行檔案,加上一個json配置檔案。
讓我們先來看看幫助資訊
auto_ssh $ ./autossh -h
go寫的一個ssh遠端客戶端。可一鍵登入遠端伺服器,主要用來彌補Mac/Linux Terminal ssh無法儲存密碼的不足。
基本用法:
直接輸入autossh不帶任何引數,列出所有伺服器,輸入對應編號登入。
引數:
-v, --version 顯示 autossh 的版本資訊。
-h, --help 顯示幫助資訊。
操作:
list 顯示所有server。
add <name> 新增一個 server。如:autossh add vagrant。
edit <name> 編輯一個 server。如:autossh edit vagrant。
remove <name> 刪除一個 server。如:autossh remove vagrant。
複製程式碼
伺服器資訊配置
配置檔案servers.json也很容易編輯,看看官方提供的例子
[
{
"name": "vagrant", // 顯示名稱
"ip": "192.168.33.10", // 伺服器IP或域名
"port": 22, // 埠
"user": "root", // 使用者名稱
"password": "vagrant", // 密碼
"method": "password" // 認證方式,目前支援password和pem
},
{
"name": "ssh-pem",
"ip": "192.168.33.11",
"port": 22,
"user": "root",
"password": "your pem file password or empty", // pem金鑰密碼,若無密碼則留空
"method": "pem", // pem金鑰認證
"key": "your pem file path" // pem金鑰檔案絕對路徑
}
// ...可配置多個
]
複製程式碼
使用方法
儲存servers.json,執行autossh,即可看到相應伺服器資訊,輸入對應序號,即可自動登入到伺服器。下面的動圖是借用官方的,原諒一下自己的懶惰 :-D
用iTerm自動呼叫
前文說過,我在iTerm裡面建立了一個叫Remote的Profile,目的就是為了在使用CMD+O開啟一個新視窗的時候自動執行autossh,那麼我們還需要在iTerm裡面新增一點配置,開啟視窗時自動執行一段shell。如下圖
大功告成,打完收工