【问题标题】:Golang Enter SSH Sudo Password on Prompt (or exit)Golang 在提示时输入 SSH Sudo 密码(或退出)
【发布时间】:2017-11-12 07:05:28
【问题描述】:

我正在尝试通过我的 Go 程序中的 SSH package 运行脚本(到目前为止我已经成功)。

我的问题是,如果用户具有 sudo 权限,脚本会尝试使用 sudo 运行命令,这会导致 bash 脚本暂停,直到用户输入密码。

例如:

[ERROR ] Install cs-server: Checking dependencies: missing: lib32gcc1
# It attempts to install the missing dependencies with sudo but pauses here
[sudo] password for guest: 

在我的 Go 程序中,我写了一些类似这样的东西:

// Connect to SSH and retreive session...

out, err := session.StdoutPipe()
if err != nil {
    log.Fatal(err)
}

go func(out io.Reader) {
    r := bufio.NewScanner(out)
    for r.Scan() {
        fmt.Println(r.Text())
    }
}(out)

// Execute ssh command...

我收到与上面示例完全相同的输出,只是在这种情况下,我什至看不到 [sudo] password for guest: 行...它只打印到 [ERROR ] Install cs-server: Checking dependencies: missing: lib32gcc1 并永远暂停。

我怎样才能绕过这个暂停?我的选择是从我的 Go 程序中自动输入密码,或者结束 ssh 执行并接收输出。

【问题讨论】:

  • 如果我的解决方案可以在不依赖 ssh 远程机器的情况下得到解决,那就太好了。
  • 如果可能,您可以让sudo 不需要密码,至少对于该命令。您可以查看this question on a sister site 以了解如何在脚本中包含密码并让sudo 使用它——当然这会将您的密码以纯文本形式放入您的代码中

标签: linux bash go ssh


【解决方案1】:

我设法通过使用session.StdoutPipe()session.StdinPipe() 解决了这个问题。我编写了一个 go 例程,它扫描每个字节并检查最后写入的行是否以 "[sudo] password for " 开头并以 ": " 结尾。它会将password + "\n" 写入session.StdinPipe(),从而继续执行脚本。

这是我拥有的所有代码。

package ssh

import (
    "bufio"
    "io"
    "log"
    "net"
    "strings"

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

type Connection struct {
    *ssh.Client
    password string
}

func Connect(addr, user, password string) (*Connection, error) {
    sshConfig := &ssh.ClientConfig{
        User: user,
        Auth: []ssh.AuthMethod{
            ssh.Password(password),
        },
        HostKeyCallback: ssh.HostKeyCallback(func(hostname string, remote net.Addr, key ssh.PublicKey) error { return nil }),
    }

    conn, err := ssh.Dial("tcp", addr, sshConfig)
    if err != nil {
        return nil, err
    }

    return &Connection{conn, password}, nil

}

func (conn *Connection) SendCommands(cmds ...string) ([]byte, error) {
    session, err := conn.NewSession()
    if err != nil {
        log.Fatal(err)
    }
    defer session.Close()

    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
    }

    err = session.RequestPty("xterm", 80, 40, modes)
    if err != nil {
        return []byte{}, err
    }

    in, err := session.StdinPipe()
    if err != nil {
        log.Fatal(err)
    }

    out, err := session.StdoutPipe()
    if err != nil {
        log.Fatal(err)
    }

    var output []byte

    go func(in io.WriteCloser, out io.Reader, output *[]byte) {
        var (
            line string
            r    = bufio.NewReader(out)
        )
        for {
            b, err := r.ReadByte()
            if err != nil {
                break
            }

            *output = append(*output, b)

            if b == byte('\n') {
                line = ""
                continue
            }

            line += string(b)

            if strings.HasPrefix(line, "[sudo] password for ") && strings.HasSuffix(line, ": ") {
                _, err = in.Write([]byte(conn.password + "\n"))
                if err != nil {
                    break
                }
            }
        }
    }(in, out, &output)

    cmd := strings.Join(cmds, "; ")
    _, err = session.Output(cmd)
    if err != nil {
        return []byte{}, err
    }

    return output, nil
}

以及如何使用它的示例。

// ssh refers to the custom package above
conn, err := ssh.Connect("0.0.0.0:22", "username", "password")
if err != nil {
    log.Fatal(err)
}

output, err := conn.SendCommands("sleep 2", "echo Hello!")
if err != nil {
    log.Fatal(err)
}

fmt.Println(string(output))

【讨论】:

    【解决方案2】:

    这是@acidic 的代码无法完全捕获输出流的问题。 更新后的代码如下

    package main
    import (
        "bytes"
        "fmt"
        "io"
        "log"
        "net"
        "strings"
    
        "golang.org/x/crypto/ssh"
    )
    
    type Connection struct {
        *ssh.Client
        password string
    }
    
    func Connect(addr, user, password string) (*Connection, error) {
        sshConfig := &ssh.ClientConfig{
            User: user,
            Auth: []ssh.AuthMethod{
                ssh.Password(password),
            },
            HostKeyCallback: ssh.HostKeyCallback(func(hostname string, remote net.Addr, key ssh.PublicKey) error { return nil }),
        }
    
        conn, err := ssh.Dial("tcp", addr, sshConfig)
        if err != nil {
            return nil, err
        }
    
        return &Connection{conn, password}, nil
    
    }
    
    func (conn *Connection) SendCommands(cmds string) ([]byte, error) {
        session, err := conn.NewSession()
        if err != nil {
            log.Fatal(err)
        }
        defer session.Close()
    
        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
        }
    
        err = session.RequestPty("xterm", 80, 40, modes)
        if err != nil {
            return []byte{}, err
        }
    
        stdoutB := new(bytes.Buffer)
        session.Stdout = stdoutB
        in, _ := session.StdinPipe()
    
        go func(in io.Writer, output *bytes.Buffer) {
            for {
                if strings.Contains(string(output.Bytes()), "[sudo] password for ") {
                    _, err = in.Write([]byte(conn.password + "\n"))
                    if err != nil {
                        break
                    }
                    fmt.Println("put the password ---  end .")
                    break
                }
            }
        }(in, stdoutB)
    
        err = session.Run(cmds)
        if err != nil {
            return []byte{}, err
        }
        return stdoutB.Bytes(), nil
    }
    
    func main() {
        // ssh refers to the custom package above
        conn, err := Connect("0.0.0.0:22", "username", "password")
        if err != nil {
            log.Fatal(err)
        }
    
        output, err := conn.SendCommands("sudo docker ps")
        if err != nil {
            log.Fatal(err)
        }
        fmt.Println(string(output))
    
    }
    

    【讨论】:

    • 因为你的代码在你的 go func for 循环中没有 BREAK,你应该添加 goroutine quit
    【解决方案3】:

    解决方法是将sudo [cmd] 转换为echo [password] | sudo -S [cmd],这不好,但对我有用。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2021-03-23
      • 2016-04-08
      • 1970-01-01
      • 2012-08-11
      • 2015-08-14
      • 2010-09-30
      • 1970-01-01
      • 2023-03-20
      相关资源
      最近更新 更多