- xml - AJAX/Jquery XML 解析
- 具有多重继承的 XML 模式
- .net - 枚举序列化 Json 与 XML
- XML 简单类型、简单内容、复杂类型、复杂内容
我进行了高低搜索,但我不确定我是否使用了错误的关键字,但我无法理解这一点。我正在构建一个应用程序,它接收主机名列表并通过 SSH 连接到这些主机名。它旨在维护这些连接(并在断开连接时重新连接)。定期地,我的程序将接受指令并向部分/所有这些主机执行命令。
我目前的问题是,我知道你不能初始化一个变量而不使用它,我必须为这些 SSH 连接动态创建变量,这样我就可以独立监控/管理它们(读/写,必要时重新连接等) .由于我对 go 的了解有限,而且倾向于不小心使事情过于复杂,到目前为止,我想到的最好的方法是使用一个结构并为每个连接及其参数(主机名、用户名、密码、SSH 配置详细信息、日志文件)附加位置等)。
目前我的狡猾代码看起来像这样:
package main
import (
"os"
"fmt"
"flag"
"golang.org/x/crypto/ssh"
"io"
"log"
"strings"
"time"
"encoding/xml"
"bufio"
"net"
)
// SSH connection struct
type SSHConnections struct {
Host string
User string
Password string
CLILogfile string
SSHConn ssh.Client
SSHConfig ssh.ClientConfig
}
func main() {
//parse list of addresses
var ipaddr = []string{"a.b.c.d","e.f.g.h"} //hard-coded for now
//build out SSHConnections struct
for i := 0; i < len(ipaddr); i++ {
tempsshConfig := &ssh.ClientConfig{
User: "administrator",
Auth: []ssh.AuthMethod{
ssh.Password("Password!"),
},
HostKeyCallback: ssh.InsecureIgnoreHostKey(),
}
tempsshConfig.Config.Ciphers = append(tempsshConfig.Config.Ciphers, "aes128-cbc")
var newitem = SSHConnections{
Host: ipaddr[i],
User: "administrator",
Password: "Password!",
CLILogfile: ipaddr[1],
SSHConn: ssh.Client,
SSHConfig: *tempsshConfig,
}
SSHConnections = append(SSHConnections, newitem)
}
使用上面的代码我得到了编译错误:
type ssh.Client is not an expression
type SSHConnections is not an expression
在此之后还有特定于连接的参数(日志文件声明、额外的 SSH 参数、实际的 SSH 连接过程)我将需要管理,大概在与上面相同的结构中。上面是我目前的绊脚石,我对如何解决这个问题一无所知,更不用说将下面的单连接代码集成到上面了。
//extra SSH parameters required before connecting
sshConfig.Config.Ciphers = append(sshConfig.Config.Ciphers, "aes128-cbc")
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
}
//prepare logfiles
f, ferr := os.OpenFile("outputfile.txt", os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0600)
if ferr != nil {
panic(ferr)
}
defer f.Close()
//SSH connection procedure
connection, err := ssh.Dial("tcp", hostname+":22", sshConfig)
if err != nil {
log.Fatalf("Failed to dial: %s", err)
}
session, err := connection.NewSession()
handleError(err, true, "Failed to create session: %s")
sshOut, err := session.StdoutPipe()
handleError(err, true, "Unable to setup stdin for session: %v")
sshIn, err := session.StdinPipe()
handleError(err, true, "Unable to setup stdout for session: %v")
if err := session.RequestPty("xterm", 0, 200, modes); err != nil {
session.Close()
handleError(err, true, "request for pseudo terminal failed: %s")
}
if err := session.Shell(); err != nil {
session.Close()
handleError(err, true, "request for shell failed: %s")
}
为了使我的帖子更清晰,我删除了一大堆不相关的代码。我希望我在这方面取得了远程成功。
我什至不确定结构是否是我想要的,但接收可变数量的主机名的动态性以及单独控制它们的要求是关键。稍后,我可能还需要单独断开连接,因此我正在寻找可以操纵它们的结构。
非常感谢任何帮助我指明正确方向的人。谢谢!
------------- 在此行下方更新----------------
在 mu 的下面的帖子和一些谷歌搜索之后,我的代码现在看起来像这样(删除了导入):
var connections []SSHConnections
// SSH connection struct
type SSHConnections struct {
Host string
User string
Password string
CLILogfilepath string
CLILogfile *os.File
SSHConn *ssh.Client
Session *ssh.Session
SSHOut io.Reader
SSHIn io.WriteCloser
SSHConfig *ssh.ClientConfig
CLIReady string
SSHConnErr error
SessionErr error
SSHOutErr error
SSHInErr error
CLILogfileErr error
CommandQueue chan string
}
func handleError(e error, fatal bool, customMessage ...string) {
var errorMessage string
if e != nil {
if len(customMessage) > 0 {
errorMessage = strings.Join(customMessage, " ")
} else {
errorMessage = "%s"
}
if fatal == true {
log.Fatalf(errorMessage, e)
} else {
log.Print(errorMessage, e)
}
}
}
func writeToFile(f *os.File, err error, inputString string) {
if _, err = f.WriteString(inputString); err != nil {
panic(err)
}
}
func connectionWorker(i int) {
//this function is to monitor SSH connections and reconnect if/when they are terminated
fmt.Println("goroutine for: "+connections[i].Host)
for { //not sure if this is even necessary? yet to test
select {
case command := <-connections[i].CommandQueue:
fmt.Printf("received %v from queue\n", command)
default:
time.Sleep(1000 * time.Millisecond)
}
}
}
func main() {
//parse list of addresses
var ipaddr = []string{"host1","host2"}
//build out SSHConnections struct
for i := 0; i < len(ipaddr); i++ {
tempsshConfig := &ssh.ClientConfig{
User: "administrator",
Auth: []ssh.AuthMethod{
ssh.Password("Password!"),
},
HostKeyCallback: ssh.InsecureIgnoreHostKey(),
}
tempsshConfig.Config.Ciphers = append(tempsshConfig.Config.Ciphers, "aes128-cbc")
var newitem = SSHConnections{
Host: ipaddr[i],
User: "administrator",
Password: "Password!",
CLILogfilepath: ipaddr[i],
SSHConfig: tempsshConfig,
}
connections = append(connections, newitem)
}
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
}
//establish connections, check each for relevant 'admin:' readiness
for i := 0; i < len(connections); i++ {
connections[i].SSHConn, connections[i].SSHConnErr = ssh.Dial("tcp", connections[i].Host+":22", connections[i].SSHConfig)
if connections[i].SSHConnErr != nil {
fmt.Println("Failed to dial: %s")
} else {
connections[i].Session, connections[i].SessionErr = connections[i].SSHConn.NewSession()
handleError(connections[i].SessionErr, true, "Failed to create session: %s")
connections[i].SSHOut, connections[i].SSHOutErr = connections[i].Session.StdoutPipe()
handleError(connections[i].SSHOutErr, true, "Unable to setup stdin for session: %v")
connections[i].SSHIn, connections[i].SSHInErr = connections[i].Session.StdinPipe()
handleError(connections[i].SSHInErr, true, "Unable to setup stdout for session: %v")
if err := connections[i].Session.RequestPty("xterm", 0, 200, modes); err != nil {
connections[i].Session.Close()
handleError(connections[i].SSHInErr, true, "request for pseudo terminal failed: %s")
}
}
if err := connections[i].Session.Shell(); err != nil {
connections[i].Session.Close()
handleError(err, true, "request for shell failed: %s")
}
//initialise the buffered CommandQueue channel
connections[i].CommandQueue = make(chan string, 1000)
//prepare logfiles
connections[i].CLILogfile, connections[i].CLILogfileErr = os.OpenFile(connections[i].CLILogfilepath+".txt", os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0600)
if connections[i].CLILogfileErr != nil {
panic(connections[i].CLILogfileErr)
}
defer connections[i].CLILogfile.Close()
writeToFile(connections[i].CLILogfile, connections[i].CLILogfileErr, "testing output to file\r\n")
//monitor/maintain connections, monitor work queue and execute
go connectionWorker(i)
}
}
以上内容并不漂亮也不完美,如果有人要复制/粘贴,可能还需要进行一些调整,所以请记住这一点。我敢肯定,通过适当的测试我会偶然发现很多错误,但目前它可以按预期工作。我原来的疑问现在都得到了解答,谢谢!
最佳答案
你的循环看起来像这样:
for i := 0; i < len(ipaddr); i++ {
// ...
var newitem = SSHConnections{
// ...
SSHConn: ssh.Client,
// ...
}
SSHConnections = append(SSHConnections, newitem)
}
当你构建 newitem
时,你没有 SSHConn
的 ssh.Client
所以你应该把它去掉(并且因此使用它的零值)而不是尝试使用 ssh.Client
类型作为表达式;这是你的第一个错误。
第二个错误是类似的:您使用的类型(在本例中为 SSHConnections
)您确实想要使用 slice :
SSHConnections = append(SSHConnections, newitem)
应该是:
someSlice = append(someSlice, newitem)
那个循环应该看起来更像这样:
var connections []SSHConnections
for i := 0; i < len(ipaddr); i++ {
// ...
var newitem = SSHConnections{
// Everything that's there now except the SSHConn ...
}
connections = append(connections, newitem)
}
或者,既然您知道连接
需要多大:
connections := make([]SSHConnections, len(ipaddr))
for i := 0; i < len(ipaddr); i++ {
// ...
var newitem = SSHConnections{
// Everything that's there now except the SSHConn ...
}
connections[i] = newitem
}
然后稍后您调用 ssh.Dial
和 connection.NewClient
来获取可能会在您的终端中的 ssh.Client
连接
。
此外,您最好为 SSHConn
和 SSConfig
使用指向 SSHConnections
的指针,因为这是包的接口(interface)想要使用的.
关于go - 初始化/管理并发 SSH 连接,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48455865/
我知道这个问题可能已经被问过,但我检查了所有这些,我认为我的情况有所不同(请友善)。所以我有两个数据集,第一个是测试数据集,第二个是我保存在数据框中的预测(预测值,这就是没有数据列的原因)。我想合并两
在 .loc 方法的帮助下,我根据同一数据框中另一列中的值来识别 Panda 数据框中某一列中的值。 下面给出了代码片段供您引用: var1 = output_df['Player'].loc[out
当我在 Windows 中使用 WinSCP 通过 Ubuntu 连接到 VMware 时,它提示: The server rejected SFTP connection, but it lis
我正在开发一个使用 xml web 服务的 android 应用程序。在 wi-fi 网络中连接时工作正常,但在 3G 网络中连接时失败(未找到 http 404)。 这不仅仅发生在设备中。为了进行测
我有一个XIB包含我的控件的文件,加载到 Interface Builder(Snow Leopard 上的 Xcode 4.0.2)中。 文件的所有者被设置为 someClassController
我在本地计算机上管理 MySQL 数据库,并通过运行以下程序通过 C 连接到它: #include #include #include int main(int argc, char** arg
我不知道为什么每次有人访问我网站上的页面时,都会打开一个与数据库的新连接。最终我到达了大约 300 并收到错误并且页面不再加载。我认为它应该工作的方式是,我将 maxIdle 设置为 30,这意味着
希望清理 NMEA GPS 中的 .txt 文件。我当前的代码如下。 deletes = ['$GPGGA', '$GPGSA', '$GPGSV', '$PSRF156', ] searchquer
我有一个 URL、一个用户名和一个密码。我想在 C# .Net WinForms 中建立 VPN 连接。 你能告诉我从哪里开始吗?任何第三方 API? 代码示例将受到高度赞赏... 最佳答案 您可以像
有没有更好的方法将字符串 vector 转换为字符 vector ,字符串之间的终止符为零。 因此,如果我有一个包含以下字符串的 vector "test","my","string",那么我想接收一
我正在编写一个库,它不断检查 android 设备的连接,并在设备连接、断开连接或互联网连接变慢时给出回调。 https://github.com/muddassir235/connection_ch
我的操作系统:Centos 7 + CLOUDLINUX 7.7当我尝试从服务器登录Mysql时 [root@server3 ~]# Mysql -u root -h localhost -P 330
我收到错误:Puma 发现此错误:无法打开到本地主机的 TCP 连接:9200(连接被拒绝 - 连接(2)用于“本地主机”端口 9200)(Faraday::ConnectionFailed)在我的
请给我一些解决以下错误的方法。 这是一个聊天应用....代码和错误如下:: conversations_controller.rb def create if Conversation.bet
我想将两个单元格中的数据连接到一个单元格中。我还想只组合那些具有相同 ID 的单元格。 任务 ID 名称 4355.2 参与者 4355.2 领袖 4462.1 在线 4462.1 快速 4597.1
我经常需要连接 TSQL 中的字段... 使用“+”运算符时 TSQL 强制您处理的两个问题是 Data Type Precedence和 NULL 值。 使用数据类型优先级,问题是转换错误。 1)
有没有在 iPad 或 iPhone 应用程序中使用 Facebook 连接。 这个想法是登录这个应用程序,然后能够看到我的哪些 facebook 用户也在使用该应用程序及其功能。 最佳答案 是的。
我在连接或打印字符串时遇到了一个奇怪的问题。我有一个 char * ,可以将其设置为字符串文字的几个值之一。 char *myStrLiteral = NULL; ... if(blah) myS
对于以下数据 - let $x := "Yahooooo !!!! Select one number - " let $y := 1 2 3 4 5 6 7 我想得到
我正在看 UDEMY for perl 的培训视频,但是视频不清晰,看起来有错误。 培训展示了如何使用以下示例连接 2 个字符串: #!usr/bin/perl print $str = "Hi";
我是一名优秀的程序员,十分优秀!