亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > Golang > 正文

用Go寫一個輕量級的ssh批量操作工具的方法

2020-04-01 18:59:04
字體:
來源:轉載
供稿:網友

前言

這是一個輪子。

大家都知道 Ansible 是功能超級強大的自動化運維工具,十分的高大上。太高大上了以至于在低端運維有點水土不服,在于三點:

  1. Ansible 是基于 Python 的,而 Python 下的安裝是有一堆依賴的。。。不要笑!對于很多使用 Win 的用戶而言,光是裝 Python, 裝 pip 就夠喝一壺的了。
  2. Ansible 的 paybook 所使用的 yaml 語法當然非常強大了。然而對于新人而言,剛入手是玩不轉的,需要學習。雖然 Ansible 相比其他的自動化運維工具,它的學習曲線已經非常平易近人了,但畢竟還是要學一下的不是么
  3. Ansible 自動化運維 Linux 服務器得益于 Linux 上 python 的默認支持,功能非常強大。然而如果拿來跑交換機的話,因為交換機上通常沒有 python 環境,功能就要打很多折扣了。基本上也就是執行一系列的命令組合。而我們這種有大片園區網的傳統單位,運維的大頭正式是交換機~

所以造這個輪子的出發點是基于以下考慮的:

  1. 要跨平臺,木有依賴,開箱即用。用 Go 來擼一個就能很好的滿足這個需求。你看 Open-Falcon 的 agent,ELK 的 beats ,都選擇用 Go 來實現,就是這個原因。
  2. 簡單無腦,無需學習。直接堆砌命令行就行,就像我們初始化交換機的那種命令行組合模板。只要 cli 會玩,直接照搬過來就行。
  3. 要支持并發。這個是 Go 的強項了,無需多言。
  4. 最后當然是學習 Go 啦。

一點都沒有黑 Ansible 的意思。我們也有在用 Ansible 來做自動化運維的工作,我覺得所有運維最好都學習下 Ansible,將來總是要往自動化的方向走的。這個輪子的目的在于學習 Ansible 之前,先有個夠簡單無腦的工具解決下眼前的需求~

建立 ssh 會話

Go 自身不帶 ssh 包。他的 ssh 包放在了 https://godoc.org/golang.org/x/crypto/ssh 這里。import 他就好

import "golang.org/x/crypto/ssh"

首先我們需要建立一個 ssh 會話,比如這樣。

func connect(user, password, host, key string, port int, cipherList []string) (*ssh.Session, error) {  var (    auth     []ssh.AuthMethod    addr     string    clientConfig *ssh.ClientConfig    client    *ssh.Client    config    ssh.Config    session   *ssh.Session    err     error  )  // get auth method  auth = make([]ssh.AuthMethod, 0)  if key == "" {    auth = append(auth, ssh.Password(password))  } else {    pemBytes, err := ioutil.ReadFile(key)    if err != nil {      return nil, err    }    var signer ssh.Signer    if password == "" {      signer, err = ssh.ParsePrivateKey(pemBytes)    } else {      signer, err = ssh.ParsePrivateKeyWithPassphrase(pemBytes, []byte(password))    }    if err != nil {      return nil, err    }    auth = append(auth, ssh.PublicKeys(signer))  }  if len(cipherList) == 0 {    config = ssh.Config{      Ciphers: []string{"aes128-ctr", "aes192-ctr", "aes256-ctr", "aes128-gcm@openssh.com", "arcfour256", "arcfour128", "aes128-cbc", "3des-cbc", "aes192-cbc", "aes256-cbc"},    }  } else {    config = ssh.Config{      Ciphers: cipherList,    }  }  clientConfig = &ssh.ClientConfig{    User:  user,    Auth:  auth,    Timeout: 30 * time.Second,    Config: config,    HostKeyCallback: func(hostname string, remote net.Addr, key ssh.PublicKey) error {      return nil    },  }  // connet to ssh  addr = fmt.Sprintf("%s:%d", host, port)  if client, err = ssh.Dial("tcp", addr, clientConfig); err != nil {    return nil, err  }  // create session  if session, err = client.NewSession(); err != nil {    return nil, err  }  modes := ssh.TerminalModes{    ssh.ECHO:     0,   // disable echoing    ssh.TTY_OP_ISPEED: 14400, // input speed = 14.4kbaud    ssh.TTY_OP_OSPEED: 14400, // output speed = 14.4kbaud  }  if err := session.RequestPty("xterm", 80, 40, modes); err != nil {    return nil, err  }  return session, nil}

ssh.AuthMethod 里存放了 ssh 的認證方式。使用密碼認證的話,就用 ssh.Password()來加載密碼。使用密鑰認證的話,就用 ssh.ParsePrivateKey() 或 ssh.ParsePrivateKeyWithPassphrase() 讀取密鑰,然后通過 ssh.PublicKeys() 加載進去。

ssh.config 這個 struct 存了 ssh 的配置參數,他有以下幾個配置選項,以下引用自GoDoc 。

type Config struct {  // Rand provides the source of entropy for cryptographic  // primitives. If Rand is nil, the cryptographic random reader  // in package crypto/rand will be used.  // 加密時用的種子。默認就好  Rand io.Reader  // The maximum number of bytes sent or received after which a  // new key is negotiated. It must be at least 256. If  // unspecified, a size suitable for the chosen cipher is used.  // 密鑰協商后的最大傳輸字節,默認就好  RekeyThreshold uint64  // The allowed key exchanges algorithms. If unspecified then a  // default set of algorithms is used.  //   KeyExchanges []string  // The allowed cipher algorithms. If unspecified then a sensible  // default is used.  // 連接所允許的加密算法  Ciphers []string  // The allowed MAC algorithms. If unspecified then a sensible default  // is used.  // 連接允許的 MAC (Message Authentication Code 消息摘要)算法,默認就好  MACs []string}

基本上默認的就好啦。但是 Ciphers 需要修改下,默認配置下 Go 的 SSH 包提供的 Ciphers 包含以下加密方式

 

復制代碼 代碼如下:

aes128-ctr aes192-ctr aes256-ctr aes128-gcm@openssh.com arcfour256 arcfour128

 

連 linux 通常沒有問題,但是很多交換機其實默認只提供 aes128-cbc 3des-cbc aes192-cbc aes256-cbc 這些。因此我們還是加全一點比較好。

這里有兩個地方要提一下

1、在 clientConfig 里有這么一段

HostKeyCallback: func(hostname string, remote net.Addr, key ssh.PublicKey) error {  return nil},

這是因為默認密鑰不受信任時,Go 的 ssh 包會在 HostKeyCallback 里把連接干掉(1.8 之后加的應該)。但是我們使用用戶名密碼連接的時候,這個太正常了不是么,所以讓他 return nil 就好了。

2、在 NewSession() 后,我們定義了 modes 和 RequestPty。這是因為為之后使用 session.Shell() 模擬終端時,所建立的終端參數。如果不配的話,默認值可能導致在某些終端上執行失敗。例如一些 H3C 的交換機,連接建立后默認推出來的 Copyright 可能會導致 ssh 連接異常,然后超時或者直接斷掉。例如這樣:

******************************************************************************* Copyright (c) 2004-2016 Hangzhou H3C Tech. Co., Ltd. All rights reserved. ** Without the owner's prior written consent,                 ** no decompiling or reverse-engineering shall be allowed.          *******************************************************************************

配置的參數照搬 GoDoc 上的示例就好了:

// Set up terminal modesmodes := ssh.TerminalModes{  ssh.ECHO:     0,   // disable echoing  ssh.TTY_OP_ISPEED: 14400, // input speed = 14.4kbaud  ssh.TTY_OP_OSPEED: 14400, // output speed = 14.4kbaud}// Request pseudo terminalif err := session.RequestPty("xterm", 40, 80, modes); err != nil {  log.Fatal("request for pseudo terminal failed: ", err)}

執行命令

建立起 session 后,執行命令就很簡單了,用 session.Run() 就可以執行我們的命令,結果則返回到 session.Studout 里。我們跑個簡單的測試。

const (  username = "admin"  password = "password"  ip    = "192.168.15.101"  port   = 22  cmd   = "show clock")func Test_SSH_run(t *testing.T) {  ciphers := []string{}  session, err := connect(username, password, ip, port, ciphers)  if err != nil {    t.Error(err)    return  }  defer session.Close()  var stdoutBuf bytes.Buffer  session.Stdout = &stdoutBuf  session.Run(cmd)  t.Log(session.Stdout)  return}

目標是一臺交換機,測試一下

=== RUN  Test_SSH_run--- PASS: Test_SSH_run (0.69s)  ssh_test.go:30: 07:55:52.598 UTC Wed Jan 17 2018PASS

可以看到 show clock 的命令已經成功執行了,并返回了結果。

session.Run() 僅限定執行單條命令,要執行若干命令組合就需要用到 session.Shell() 了。意思很明確,就是模擬一個終端去一條一條執行命令,并返回結果。就像我們用 Shell 一樣,我們把整過過程打印出來輸出就好了。從 session.StdinPipe() 逐個輸入命令,從session.Stdout 和 session.Stderr 獲取 Shell 上的輸出。一樣來做個測試。

const (  username = "admin"  password = "password"  ip    = "192.168.15.101"  port   = 22  cmds   = "show clock;show env power;exit")func Test_SSH(t *testing.T) {  var cipherList []string  session, err := connect(username, password, ip, key, port, cipherList)  if err != nil {    t.Error(err)    return  }  defer session.Close()  cmdlist := strings.Split(cmd, ";")  stdinBuf, err := session.StdinPipe()  if err != nil {    t.Error(err)    return  }  var outbt, errbt bytes.Buffer  session.Stdout = &outbt  session.Stderr = &errbt  err = session.Shell()  if err != nil {    t.Error(err)    return  }  for _, c := range cmdlist {    c = c + "/n"    stdinBuf.Write([]byte(c))  }  session.Wait()  t.Log((outbt.String() + errbt.String()))  return}

還是那臺交換機,測試一下

=== RUN  Test_SSH--- PASS: Test_SSH (0.69s)  ssh_test.go:51: sw-1#show clock    07:59:52.598 UTC Wed Jan 17 2018    sw-1#show env power    SW PID         Serial#   Status      Sys Pwr PoE Pwr Watts    -- ------------------ ---------- --------------- ------- ------- -----     1 Built-in                     Good        sw-1#exitPASS

可以看到,兩個命令都得到執行了,并在執行完 exit 后退出連接。

比較一下和 session.Run() 的區別,可以發現在 session.Shell() 模式下,輸出的內容包含了主機的名字,輸入的命令等等。因為這是 tty 執行的結果嘛。如果我們只需要執行命令倒也無所謂,但是如果我們還需要從執行命令的結果中讀取一些信息,這些內容就顯得有些臃腫了。比如我們在一臺 ubuntu 上跑一下看看

=== RUN  Test_SSH--- PASS: Test_SSH (0.98s)    ssh_test.go:50: Welcome to Ubuntu 16.04.3 LTS (GNU/Linux 4.4.0-98-generic x86_64)         * Documentation: https://help.ubuntu.com         * Management:   https://landscape.canonical.com         * Support:    https://ubuntu.com/advantage         System information as of Thu Jan 18 16:34:56 CST 2018         System load: 0.0        Processes:       335         Usage of /:  10.0% of 90.18GB  Users logged in:    0         Memory usage: 2%         IP address for eth0:  192.168.80.131         Swap usage:  0%         IP address for docker0: 172.17.0.1         Graph this data and manage this system at:          https://landscape.canonical.com/        16 個可升級軟件包。        16 個安全更新。        New release '17.10' available.        Run 'do-release-upgrade' to upgrade to it.        You have new mail.        Last login: Thu Jan 18 16:31:41 2018 from 192.168.95.104        root@ubuntu-docker-node3:~# root@ubuntu-docker-node3:/opt# /opt        root@ubuntu-docker-node3:/opt# 注銷

最起碼,上面那一堆 System information 就用不著嘛。交換機是沒有辦法,Linux 上能不能通過一條命令,也就是想辦法 session.Run() 來執行命令組合呢?

答案是可以的,把命令通過 && 連接起來就好了嘛。LInux 的 Shell 會幫我們拆開來分別運行的,比如上面的這個命令我們就可以合并成一條命令 cd /opt&&pwd&&exit

=== RUN  Test_SSH_run--- PASS: Test_SSH_run (0.91s)  ssh_test.go:76: /opt

立馬就簡潔了對不對?

輪子

ssh 執行命令這樣就差不多了。要變成一個可以用 ssh 批量操作工具,我們還要給他加上并發執行,并發限制,超時控制,輸入參數解析,輸出格式等等

這里就不展開了,最終這個造出來的輪子長這樣:https://github.com/shanghai-edu/multissh

可以直接命令行來執行,通過 ; 號或者 , 號作為命令和主機的分隔符。

 

復制代碼 代碼如下:

# ./multissh -cmds "show clock" -hosts "192.168.31.21;192.168.15.102" -u admin -p password

 

也可以通過文本來存放主機組和命令組,通過換行符分隔。

 

復制代碼 代碼如下:

# ./multissh -cmdfile cmd1.txt.example -hostfile host.txt.example -u admin -p password

 

特別的,如果輸入的是 IP (-ips 或 -ipfile),那么允許 IP 地址段方式的輸入,例如 192.168.15.101-192.168.15.110 。(還記得 swcollector 么,類似的實現方式)

 

復制代碼 代碼如下:

# ./multissh -cmds "show clock" -ips "192.168.15.101-192.168.15.110" -u admin -p password

 

支持使用 ssh 密鑰認證,此時如果輸入 password ,則為作為 key 的密碼

 

復制代碼 代碼如下:

# ./multissh -hosts "192.168.80.131" -cmds "date;cd /opt;ls" -u root -k "server.key"

 

對于 linux ,支持 linuxMode 模式,也就是將命令組合通過 && 連接后,使用 session.Run() 運行。

 

復制代碼 代碼如下:

# ./multissh -hosts "192.168.80.131" -cmds "date;cd /opt;ls" -u root -k "server.key" -l

 

也可以為每個主機定義不同的配置參數,以 json 格式加載配置。

# ./multissh -c ssh.json.example

輸出可以打成 json 格式,方便程序處理。

# ./multissh -c ssh.json.example -j

也可以把輸出結果存到以主機名命名的文本中,比如用來做配置備份

# ./multissh -c ssh.json.example -outTxt

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。


發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲国产精彩中文乱码av| 日韩在线免费视频观看| 亚洲免费电影在线观看| 久久国产精品久久久久久| 91影院在线免费观看视频| 日韩精品视频在线播放| 国产精品欧美一区二区| 亚洲视屏在线播放| 欧美日韩国产中字| 国产日韩精品在线观看| 国产欧美精品日韩精品| 日韩精品视频免费专区在线播放| 欧亚精品在线观看| 亚洲成人黄色在线观看| 国产在线a不卡| 国产大片精品免费永久看nba| 欧美最猛性xxxx| 国内精品小视频在线观看| 亚洲人午夜精品| 国产精品一区二区电影| 日韩免费精品视频| 日韩电影免费在线观看中文字幕| 国产亚洲精品va在线观看| 国产成人精品一区二区三区| www.久久久久久.com| 97在线精品国自产拍中文| 国产香蕉一区二区三区在线视频| 欧美又大又硬又粗bbbbb| 国产精品第三页| 最新国产精品亚洲| 国产精品久久久久久av下载红粉| 久久乐国产精品| 26uuu亚洲国产精品| 日韩欧美一区视频| 国产一区香蕉久久| 综合国产在线视频| 国产一区香蕉久久| 欧美在线视频免费播放| 亚洲久久久久久久久久久| 亚洲欧洲日韩国产| 国产精品视频导航| 久久国产精品久久精品| 热久久美女精品天天吊色| 久久精品国产96久久久香蕉| 68精品国产免费久久久久久婷婷| 91探花福利精品国产自产在线| 久久久久久中文字幕| 国产亚洲视频中文字幕视频| 亚洲综合av影视| 日韩精品免费一线在线观看| 伊人伊成久久人综合网小说| 日韩精品视频免费在线观看| 国产精品自产拍在线观| 久久精品国产精品亚洲| 国产不卡精品视男人的天堂| 欧美在线中文字幕| 成人激情视频小说免费下载| 欧美精品激情blacked18| 国产欧美一区二区白浆黑人| 欧美日韩国产一区在线| 色妞色视频一区二区三区四区| 久久精品夜夜夜夜夜久久| 4k岛国日韩精品**专区| 亚洲片国产一区一级在线观看| 精品国产一区二区三区久久狼5月| 午夜精品久久久久久久99黑人| 91老司机精品视频| 91精品国产自产在线观看永久| 久久夜色精品国产亚洲aⅴ| 国产不卡精品视男人的天堂| 欧美日韩激情视频| 日韩中文字幕视频在线| 日本久久久a级免费| 日韩av手机在线看| 欧美高跟鞋交xxxxxhd| 国产精品九九久久久久久久| 国内精品模特av私拍在线观看| 亚洲在线免费视频| 久久久久久久一区二区| 欧美网站在线观看| 成人一区二区电影| 92福利视频午夜1000合集在线观看| 91香蕉嫩草影院入口| 亚洲日韩中文字幕在线播放| 国产精品流白浆视频| 国产一区二区三区丝袜| 日日噜噜噜夜夜爽亚洲精品| 欧美日韩亚洲高清| 91精品国产沙发| 精品国产一区av| 最近2019中文免费高清视频观看www99| 日韩精品中文字| 国产亚洲精品久久久优势| 亚洲激情在线观看视频免费| 亚洲精品日韩久久久| 亚洲va久久久噜噜噜| 国产日韩欧美影视| 国产精品永久在线| 国产69精品久久久久久| 久久久免费精品| 国产成人精品视频| 亚洲第一av网站| 久久影视电视剧免费网站| 亚洲97在线观看| 国产乱肥老妇国产一区二| 精品国产91久久久久久老师| 欧美中文在线字幕| 国产精品久久一| 日韩大陆欧美高清视频区| 国语自产精品视频在免费| 一区二区三区四区在线观看视频| 97视频网站入口| 亚洲色图第三页| 欧美日韩精品在线观看| 亚洲精品久久久久久久久久久久| 精品国产一区二区在线| 精品亚洲国产成av人片传媒| 亚洲香蕉伊综合在人在线视看| 久久久久久有精品国产| 成人激情黄色网| 91精品国产色综合久久不卡98口| 亚洲精品久久久久久久久| 国产日韩精品视频| 亚洲国模精品私拍| 中文字幕精品www乱入免费视频| 亚洲最大福利视频网站| 国产精品1区2区在线观看| 日韩在线视频播放| 欧美在线一级视频| 九九热在线精品视频| 亚洲欧美激情一区| 中文字幕视频在线免费欧美日韩综合在线看| 日韩av影片在线观看| 日韩美女在线观看一区| 精品小视频在线| 日韩中文在线视频| 在线播放国产一区二区三区| 在线观看欧美日韩国产| 亚洲影视中文字幕| 欧美寡妇偷汉性猛交| 精品人伦一区二区三区蜜桃免费| 欧美日韩成人在线播放| 91国产一区在线| 神马国产精品影院av| 亚洲另类激情图| 亚洲男人天堂古典| 亚洲男人av在线| 成人情趣片在线观看免费| 欧美在线视频免费播放| 日韩欧美精品免费在线| 国内精品小视频在线观看| 国产91精品黑色丝袜高跟鞋| 91在线视频成人| 欧美激情综合亚洲一二区| 国产美女直播视频一区| 亚洲欧美日韩中文在线| 亚洲欧美日韩精品久久亚洲区| 日韩精品黄色网| 国内成人精品视频| 91亚洲永久免费精品| 亚洲一级黄色片| 91久久国产婷婷一区二区| 欧美电影在线观看网站| 欧美激情第一页xxx|