我正在尝试实现监视各种Cisco ASR参数,这些参数对路由器执行ssh并使用show命令来获取结果。以下是工作流程的一部分(省略错误处理):
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#
还是一切都好。但是当我尝试发送更长的命令时,f.e。:
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#
我发送的命令在某些时候被拆分,但命令结果的输出正确传递而没有拆分。
所以问题是这个分裂是如何解决的?现在我没有想法。 感谢您的关注!
答案 0 :(得分:1)
愚蠢的错误。我这样请求pty:
session.RequestPty("vt100", 80, 40, modes)
这给了我宽度 40个字符,高度 80个字符。 在我的情况下,正确的呼叫应该是:
session.RequestPty("vt100", 0, 200, modes)
这样一切都按预期工作。并且无需明确更改终端宽度。
答案 1 :(得分:0)
通过在Cisco上设置终端宽度解决了问题。 所以,在会话建立之后,我发送命令:
terminal width 200
到路由器。
现在长命令在回复中正确显示。