linux - Golang 在提示时输入 SSH Sudo 密码(或退出)

标签 linux bash go ssh

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

我的问题是,如果用户具有 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:... 它只打印到[错误] 安装 cs-server:正在检查依赖项:缺少:lib32gcc1 并永远暂停。

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

最佳答案

我设法通过使用 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))

关于linux - Golang 在提示时输入 SSH Sudo 密码(或退出),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44471749/

相关文章:

Bash cd 进入具有增量名称的子目录

design-patterns - 可能有数百个小类的工厂模式的替代方案

linux - 使用 linux 安装 jPOS

android - 从命令行删除持久僵尸进程

linux - 循环内打印语句

oracle - 如何通过 xorm 从 Oracle DB 获取表值?

html - 使用 Golang 渲染模板时,不会读取来自不同文件夹的 CSS 和图像

linux - 使用 linux 脚本的简单聚合

bash - 如何在 bash 中制作管道循环

php - 从 php 网页运行 R 脚本