Golang Go语言问题请教
各位 golang 大佬(爹),上午好!我有个菜鸡问题:如何使用 golang 的 exec.Command()函数执行 sudo ?
背景:小弟在使用 chromedp 做一个爬虫,目前一切正常,唯一的遗憾就是每次运行之前需要手动在 macOS 的终端执行 sudo /Applications/Google\ Chrome.app/Contents/MacOS/Google\ Chrome --remote-debugging-port=19222
必须加 sudo 才能打开 chrome 调试窗口。因为 golang 可以执行终端命令,所以我想使用代码去唤醒调试窗口,但是测试下来好像无法全自动调用 sudo 命令(不输入密码或密码怎么传进去)。
Golang Go语言问题请教
更多关于Golang Go语言问题请教的实战系列教程也可以访问 https://www.itying.com/category-94-b0.html
cmd := exec.Command(“sudo”, “ls”, “/root”)
试过了,不行。。。会一直卡住,因为没输入密码。
go build -o main .
sudo ./main
刚试了一下,sudo 执行程序还是不会弹出 chrome 调试页。19222 也没有在监听。
你把运行用户的 sudo 设置 NOPASSWD 不就行了嘛。。
使用 pty 相关的库,应该可以?
用 sudo -S 可以从标准输入接收密码,然后你就可以正常把密码传进去了
sudo -S 然后密码通过 stdin 传进去?
抱歉,抱歉。。。找到问题了,确实 sudo 运行程序可以唤醒调试页面。应该是 chrome 路径问题。。。/Applications/Google Chrome.app/Contents/MacOS/Google Chrome 空格不能加转译。。。
在 sudoers 给用户加 NOPASSWD ,就不用输密码了
是的,这样确实可以,但是需要提前修改用户属性不太好。
我去试一下先,有结果了我来说一下。
也是个不错方法。
这是学到精髓了呀,上周的那个帖子上大分!
sudo -A
提问的艺术
辈分有点乱啊,一会爹一会自称小弟。(开个玩笑…
sudo 可以使用 -S 从 stdin 传入密码
虚心学习嘛 不磕碜!!!
提问的艺术得更新了
golang.org/x/crypto/ssh 包可以创建标准管道输出
用 session.StdinPipe()
用这个办法可以输入 sudo 的密码,之前写过一个模仿 ansiable 的程序是这么做的,里面部分代码:
package utils
import (
“batch-execution-tool/core”
“bytes”
“fmt”
“golang.org/x/crypto/ssh”
“os”
“sync”
)
func ExecuteScripts() {
var wg sync.WaitGroup
defer wg.Wait()
for _, ip := range core.Config.ServersIP {
wg.Add(1)
go func(ip string) {
defer wg.Done()
SSHClient, errSSH := createSSHConnect(ip)
if errSSH != nil {
core.Logger.Sugar().Errorf(“创建 SSH 链接失败:\n%v”, errSSH)
return
}
for _, script := range core.Config.Scripts {
for name := range script {
err := exec(SSHClient, ip, name)
if err != nil {
core.Logger.Sugar().Errorf(“在 IP 地址为:%v 的服务器,出现错误:\n%v”, ip, err)
} else {
core.Logger.Sugar().Debugf(“在 IP 地址为:%v 的服务器,执行脚本 %v 成功”, ip, name)
}
}
}
}(ip)
}
}
func exec(client *ssh.Client, ip, script string) error {
core.Logger.Sugar().Debugf(“正在 IP 地址为:%v 的服务器中执行脚本:%s”, ip, script)
session, err := client.NewSession()
if err != nil {
return err
}
var output bytes.Buffer
session.Stderr = &output
remoteDir := “scripts”
if core.Config.Auth.Username == “root” {
if err := session.Start("bash " + remoteDir + “/” + script + " > " + remoteDir + “/” + script + “.log”); err != nil {
return fmt.Errorf(“执行脚本 %s 失败: \n%s:%s”, script, err, output.String())
}
if err := session.Wait(); err != nil {
return fmt.Errorf(“执行脚本 %s 失败: \n%s:%s”, script, err, output.String())
}
} else {
if err := session.RequestPty(“xterm”, 80, 40, ssh.TerminalModes{}); err != nil {
return fmt.Errorf(“请求 TTY 出错:%v ,\nos.Stderr:%v”, err, os.Stderr)
}
stdin, err := session.StdinPipe()
defer stdin.Close()
if err != nil {
return fmt.Errorf(“创建标准管道输出错误: \n%v”, err)
}
if err := session.Start("sudo bash " + remoteDir + “/” + script + " > " + remoteDir + “/” + script + “.log”); err != nil {
return fmt.Errorf(“执行脚本 %s 失败: \n%s:%s”, script, err, output.String())
}
_, err = fmt.Fprintln(stdin, core.Config.Auth.Password)
if err != nil {
return fmt.Errorf(“使用标准管道输如密码错误: \n%v”, err)
}
if err := session.Wait(); err != nil {
return fmt.Errorf(“执行脚本 %s 失败: \n%s:%s”, script, err, output.String())
}
}
session.Close()
return nil
}
看来下次还要再长一个辈分才行了!😂
好的,抽空我试试。
连续俩帖子把我震住了,态度如此诚恳,此子必成大器
哥们都用 golang 了你可以用用 rod 这个库
兄弟我来还愿了。经过这两天的体验,rod 库确实用起来比 chromedp 方便多了!
您好!很高兴看到您对Go语言感兴趣。作为IT领域的专家,我很乐意帮助您解答关于Golang(Go语言)的问题。
Go语言是一种编译型、并发型,并具有垃圾回收功能的编程语言。其设计注重代码的简洁和高效,非常适合构建高性能的服务器和网络应用。
关于您提到的具体问题(由于您没有具体说明,我将提供一些常见问题的解答方向):
-
并发编程:Go语言的goroutine和channel机制是实现并发编程的强大工具。如果您想了解如何高效地使用它们,请告诉我您的具体需求。
-
内存管理:Go语言有自动的内存回收机制(GC),但了解如何避免内存泄漏和优化内存使用仍然很重要。我可以分享一些最佳实践和技巧。
-
标准库和第三方库:Go语言的标准库非常强大,涵盖了网络、文件操作、加密等多个方面。同时,也有许多优质的第三方库可供使用。如果您需要推荐或了解某个库,请告诉我。
-
性能调优:对于需要高性能的应用,了解Go语言的性能调优技巧至关重要。我可以提供一些关于性能分析、优化代码等方面的建议。
请具体描述您遇到的问题或想要了解的内容,以便我能提供更准确的帮助。期待与您交流!