在熟悉Go和goroutine的过程中,我遇到了执行命令的障碍。这些命令的格式为:
sudo find /folder -type f | while read i; do sudo -S chmod 644 "$i"; done
使用从如何在Golang中执行系统命令的代码中获取未知参数的代码,我试图执行此命令,但是我相信由于第一个参数为sudo,该命令未执行,这可能是错误的。我只有两个问题。
当这些命令无法运行时,我将返回“退出状态1”,是否有办法比我正在执行的操作得到更详细的错误?问题二,为什么我使用此脚本会获得“退出状态1”?那到底是怎么回事?
package main
import (
"bufio"
"fmt"
"os"
"os/exec"
"strings"
"sync"
)
func ExeCmd(cmd string, wg *sync.WaitGroup) {
parts := strings.Fields(cmd)
head := parts[0]
// Head at this point is "sudo"
parts = parts[1:len(parts)]
out, err := exec.Command(head,parts...).Output()
if err != nil {
fmt.Printf("%s\n", err)
}
fmt.Printf("%s\n", out)
wg.Done() // Signal to WaitGroup goroutine finished
}
func InArray(a []string, e string) bool {
for _, x := range a {
if x == e {
return true
fmt.Print("True")
}
}
return false
}
func main() {
exec.Command("sudo ls > /dev/null") // Get sudo password just once, doesn't seem to work with Go
wg := new(sync.WaitGroup)
reader := bufio.NewReader(os.Stdin)
fdbslices := []string{"f", "d", "b", "files", "directories", "both"}
commands := []string{}
fdb := ""
filep := ""
dirp := ""
// Grab Path
fmt.Print("Path: ")
findpath, _ := reader.ReadString('\n')
findpath = strings.ToLower(strings.Trim(findpath, "\n"))
// Grab Files, Directories, or Both
for {
fmt.Print("Files, Directories, or Both: ")
fdb, _ = reader.ReadString('\n')
fdb = strings.ToLower(strings.Trim(fdb, "\n"))
if InArray(fdbslices, fdb) == true { break }
}
// Build commands string, these come out as they should
for {
if fdb == "f" || fdb == "files" {
fmt.Print("Permissions for Files: ")
filep, _ = reader.ReadString('\n')
filep = strings.Trim(filep, "\n")
commands = append(commands, "sudo find " + findpath + " -type f | while read i; do sudo -S chmod " + filep + " \"$i\"; done")
}
if fdb == "d" || fdb == "directories" {
fmt.Print("Permissions for Directories: ")
dirp, _ = reader.ReadString('\n')
dirp = strings.Trim(dirp, "\n")
commands = append(commands, "sudo find " + findpath + " -type d | while read i; do sudo -S chmod " + dirp + " \"$i\"; done")
}
if fdb == "b" || fdb == "both" {
fmt.Print("Permissions for Files: ")
filep, _ = reader.ReadString('\n')
filep = strings.Trim(filep, "\n")
commands = append(commands, "sudo find " + findpath + " -type f | while read i; do sudo -S chmod " + filep + " \"$i\"; done")
fmt.Print("Permissions for Directories: ")
dirp, _ = reader.ReadString('\n')
dirp = strings.Trim(dirp, "\n")
commands = append(commands, "sudo find " + findpath + " -type d | while read i; do sudo -S chmod " + dirp + " \"$i\"; done")
}
break
}
// Execute Commands
for _, str := range commands {
wg.Add(1)
fmt.Print("Doing: " + str + "\r\n")
go ExeCmd(str, wg)
}
wg.Wait()
}
终端输出示例:
Path: test
Files, Directories, or Both: b
Permissions for Files: 644
Permissions for Directories: 755
Doing: find test -type f | while read i; do sudo -S chmod 644 "$i"; done
Doing: find test -type d | while read i; do sudo -S chmod 755 "$i"; done
exit status 1
exit status 1
感谢您的时间。
exec.Command()
将要求内核直接执行给定的进程。但是,您传递的命令是由外壳程序脚本链接在一起的多个程序的字符串。
如果要执行Shell脚本,则应使用以下内容:
cmd := exec.Command("/bin/sh", "-c", "sudo find ...")
Go 命令 Go语言自带有一套完整的命令操作工具,你可以通过在命令行中执行go来查看它们: 图1.3 Go命令显示详细的信息 这些命令对于我们平时编写的代码非常有用,接下来就让我们了解一些常用的命令。 go build 这个命令主要用于编译代码。在包的编译过程中,若有必要,会同时编译与之相关联的包。 如果是普通包,就像我们在1.2节中编写的mymath包那样,当你执行go build之后,它不会产
主要内容:安装第三方图形化显式分析数据工具(Graphviz),安装第三方性能分析来分析代码包,性能分析代码Go语言工具链中的 go pprof 可以帮助开发者快速分析及定位各种性能问题,如 CPU 消耗、内存分配及阻塞分析。 性能分析首先需要使用 runtime.pprof 包嵌入到待分析程序的入口和结束处。runtime.pprof 包在运行时对程序进行每秒 100 次的采样,最少采样 1 秒。然后将生成的数据输出,让开发者写入文件或者其他媒介上进行分析。 go pprof 工具链配合 Gr
主要内容:单元测试——测试和验证代码的框架,基准测试——获得代码内存占用和运行效率的性能数据Go语言拥有一套单元测试和性能测试系统,仅需要添加很少的代码就可以快速测试一段需求代码。 go test 命令,会自动读取源码目录下面名为 *_test.go 的文件,生成并运行测试用的可执行文件。输出的信息类似下面所示的样子: ok archive/tar 0.011s FAIL archive/zip 0.022s ok compress/gzip 0.033s ... 性能测试系统可以给出代
命令是在Go语言 1.4 版本里面新添加的一个命令,当运行该命令时,它将扫描与当前包相关的源代码文件,找出所有包含 的特殊注释,提取并执行该特殊注释后面的命令。 使用 命令时有以下几点需要注意: 该特殊注释必须在 .go 源码文件中; 每个源码文件可以包含多个 generate 特殊注释; 运行命令时,才会执行特殊注释后面的命令; 当命令执行出错时,将终止程序的运行; 特殊注释必须以开头,双斜线后
主要内容:远程包的路径格式,go get+ 远程包,go get 使用时的附加参数go get 命令可以借助代码管理工具通过远程拉取或更新代码包及其依赖包,并自动完成编译和安装。整个过程就像安装一个 App 一样简单。 这个命令可以动态获取远程代码包,目前支持的有 BitBucket、GitHub、Google Code 和 Launchpad。在使用 go get 命令前,需要安装与远程包匹配的代码管理工具,如 Git、SVN、HG 等,参数中需要提供一个包名。 这个命令在内
go install 命令的功能和前面一节《 go build命令》中介绍的 go build 命令类似,附加参数绝大多数都可以与 go build 通用。go install 只是将编译的中间文件放在 GOPATH 的 pkg 目录下,以及固定地将编译结果放在 GOPATH 的 bin 目录下。 这个命令在内部实际上分成了两步操作:第一步是生成结果文件(可执行文件或者 .a 包),第二步会把编译