gpt4 book ai didi

session - Golang - 过滤事件 SSH session 的标准输出

转载 作者:IT王子 更新时间:2023-10-29 00:47:51 27 4
gpt4 key购买 nike

我正在用 Go 编写一个 SSH 客户端,它连接到一个交换机并运行一系列配置命令。到目前为止,我能够成功连接到交换机、运行所需的命令并打印 session 的输出。当命令的输出太长时,开关期望输入 \n、空格或“q”时就会出现问题。例如:

switch#show int status

Port Name Status Vlan Duplex Speed Type
Gi1/0/1 notconnect 100 auto auto 10/100/1000BaseTX
Gi1/0/2 connected 915 a-full a-100 10/100/1000BaseTX
Gi1/0/3 notconnect 100 auto auto 10/100/1000BaseTX
Gi1/0/4 notconnect 100 auto auto 10/100/1000BaseTX
Gi1/0/5 notconnect 230 auto auto 10/100/1000BaseTX
...
Po1 sw-sww-100-sww-0-0 connected trunk a-full 10G
--More-- # Program hangs here; expecting a new line, space, or 'q'

--More-- 提示实际上并没有打印到屏幕上,因此只需检查 Stdout 的当前行是否包含 --More-- 并发送 \n、空格或“q”不起作用。

Gif of <code>--More--</code> prompt disappearing.

除了解决这个问题,我还想过滤 session 的Stdout,这样唯一打印的就是每个命令的输出。换句话说,我不希望将开关的提示打印到终端。

总结:

  1. 如何捕获和打印每个命令的输出?
  2. 如何在出现提示时发送新行、空格或字母“q”?

感谢任何帮助。这是我的代码:

package main

import (
"bufio"
"fmt"
"golang.org/x/crypto/ssh"
"io"
"log"
"os"
"time"
)

type Device struct {
Config *ssh.ClientConfig
Client *ssh.Client
Session *ssh.Session
Stdin io.WriteCloser
Stdout io.Reader
Stderr io.Reader
}

func (d *Device) Connect() error {
client, err := ssh.Dial("tcp", os.Args[1]+":22", d.Config)
if err != nil {
return err
}
session, err := client.NewSession()
if err != nil {
return err
}
sshIn, err := session.StdinPipe()
if err != nil {
return err
}
sshOut, err := session.StdoutPipe()
if err != nil {
return err
}
sshErr, err := session.StderrPipe()
if err != nil {
return err
}
d.Client = client
d.Session = session
d.Stdin = sshIn
d.Stdout = sshOut
d.Stderr = sshErr
return nil
}

func (d *Device) SendCommand(cmd string) error {
if _, err := io.WriteString(d.Stdin, cmd+"\r\n"); err != nil {
return err
}
return nil
}

func (d *Device) SendConfigSet(cmds []string) error {
for _, cmd := range cmds {
if _, err := io.WriteString(d.Stdin, cmd+"\r\n"); err != nil {
return err
}
time.Sleep(time.Second)
}
return nil
}

func (d *Device) PrintOutput() {
r := bufio.NewReader(d.Stdout)
for {
text, err := r.ReadString('\n')
fmt.Printf("%s", text)
if err == io.EOF {
break
}
}
}

func (d *Device) PrintErr() {
r := bufio.NewReader(d.Stderr)
for {
text, err := r.ReadString('\n')
fmt.Printf("%s", text)
if err == io.EOF {
break
}
}
}

func main() {
sshConf := ssh.Config{}
sshConf.Ciphers = append(sshConf.Ciphers, "aes128-cbc", "3des-cbc", "blowfish-cbc", "arcfour")
config := &ssh.ClientConfig{
Config: sshConf,
User: "mwalto7",
Auth: []ssh.AuthMethod{
ssh.Password("Lion$Tiger$Bear$"),
},
HostKeyCallback: ssh.InsecureIgnoreHostKey(),
Timeout: time.Second * 5,
}

sw := &Device{Config: config}

fmt.Println("Connecting to ", os.Args[1])
if err := sw.Connect(); err != nil {
log.Fatal(err)
}
defer sw.Client.Close()
defer sw.Session.Close()
defer sw.Stdin.Close()

if err := sw.Session.Shell(); err != nil {
log.Fatal(err)
}

commands := []string{"show int status", "exit"}
if err := sw.SendConfigSet(commands); err != nil {
log.Fatal(err)
}

sw.Session.Wait()

sw.PrintOutput()
sw.PrintErr()
}

最佳答案

我通过在连接到交换机之后发送命令 terminal length 0 来解决这个问题,然后再发送其余命令。这通过将开关的终端高度设置为 0 来禁用 —More— 提示,允许显示命令的整个输出。

关于session - Golang - 过滤事件 SSH session 的标准输出,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48468318/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com