使用 go ssh 库与 Cisco 设备通信

我正在尝试通过 ssh 到路由器并使用 show 命令来实现对各种 Cisco ASR 参数的监控。这是工作流程的一部分(省略了错误处理):


client, err := ssh.Dial("tcp", "172.16.32.95:22", config)

session, err := client.NewSession()

sshOut, err := session.StdoutPipe()

sshIn, err := session.StdinPipe()

err := session.RequestPty("xterm", 80, 40, modes)

err := session.Shell()

在这一点上,我可以写入sshIn和读取sshOut. 由于 Cisco 的(和其他供应商的)路由器不支持 exec'ing 命令(如果我错了,请纠正我)我所能做的就是将命令传递给 shell 并读取输入,直到找到命令提示符。


好的。这是我用来在第一个命令提示符之前跳过初始路由器问候语的部分:


buf := make([]byte, 1000)

n, err := sshOut.Read(buf) //this reads the ssh terminal welcome message

loadStr := ""

if err == nil {

    loadStr = string(buf[:n])

}

for (err == nil) && (!strings.Contains(loadStr, "[local]")) {

    n, err = sshOut.Read(buf)

    loadStr += string(buf[:n])

}

fmt.Println(loadStr)

产生:


Last login: Tue Jan 14 17:29:06 -0500 2014 on pts/1 from 172.16.35.101.

Cisco Systems SSI

[local]ewag# 

要运行我写的命令sshIn:


if _, err := sshIn.Write([]byte("show clock\r")); err != nil {

    panic("Failed to run: " + err.Error())

}

并以与阅读初始问候语相同的方式阅读响应,从而产生(带有问候语):


Last login: Tue Jan 14 18:06:14 -0500 2014 on pts/2 from 10.7.7.14.

Cisco Systems SSI

[local]ewag# 

show clock

Tuesday January 14 18:09:19 EST 2014

[local]ewag# 

还是一切正常。但是当我尝试发送更长的命令时,fe:


if _, err := sshIn.Write([]byte("show session progress ipsg-service ipsg-gprs-svc\r")); err != nil {

    panic("Failed to run: " + err.Error())

}

输出中断:


Last login: Tue Jan 14 18:09:19 -0500 2014 on pts/2 from 10.7.7.14.

Cisco Systems SSI

[local]ewag# 

show session progress ipsg

-service ipsg-gprs-svc

Unknown command - "ipsg-service", unrecognized keyword

[local]ewag# 

我发送的命令在某个时候被拆分,但命令结果的输出正确传递而没有拆分。


所以问题是如何修复这种分裂?现在我没有想法。谢谢关注!


九州编程
浏览 297回答 2
2回答

四季花海

愚蠢的错误。我是这样请求 pty 的:session.RequestPty("vt100", 80, 40, modes)这给了我40 个字符的宽度和80 个字符的高度。在我的情况下,正确的调用应该是:session.RequestPty("vt100", 0, 200, modes)这样一切都按预期工作。并且无需显式更改终端宽度。

BIG阳

通过在 Cisco 上设置终端宽度解决了问题。因此,在会话建立后,我发送命令:terminal width 200到路由器。现在长命令在回复中正确显示。
打开App,查看更多内容
随时随地看视频慕课网APP

相关分类

Go