我正在用 go 寫一個 bash 任務運行程序,它有一個簡單的概念:
taskfile
,這是一個包含任務定義(簡單的 bash 函數宣告)的 bash 腳本這是一個簡化的範例:
package main import ( "fmt" "os/exec" ) func main() { //simplified for a dynamically built script taskfilecontent := "#!/bin/bash\n\ntask:foo (){\n echo \"test\"\n}\n" // simplified for passed arguments task := "\ntask:foo" bash, _ := exec.lookpath("bash") cmd := exec.command(bash, "-c", "\"$(cat << eof\n"+taskfilecontent+task+"\neof\n)\"") fmt.println(cmd.string()) out, _ := cmd.combinedoutput() fmt.println(string(out)) }
我現在的問題是,如果透過 go 執行它就不起作用,並且我收到此錯誤
task:foo: no such file or directory
但如果我直接在 shell 中執行產生的腳本,它確實有效:
$ /opt/opt/homebrew/bin/bash -c "$(cat << EOF #!/bin/bash task:foo (){ echo "test" } task:foo EOF )" test <-- printed out from the `task:foo` above
我在這裡做錯了什麼?
你不會得到任何你不會得到的東西:
cmd := exec.command(bash, "-c", taskfilecontent+"\n"+task)
如果省略它,您的程式碼會更簡單。
#當您在 shell 中執行時:
65be85239床5...圍繞$()
的"
s 不是正在啟動的bash 副本的語法,而是正在解析命令的bash 副本的語法。/em>。它們告訴bash 的副本,命令替換的結果將作為一個字串傳遞,不受字串分割或通配符的影響。
類似地, $(cat <<eof
、eof
和最終的)"
也是互動式shell 的指令,而不是它調用的非互動式shell。它是運行的互動式shell cat
(包含連接到其標準輸入的heredoc內容的臨時檔案),讀取cat
副本的標準輸出,然後將該資料替換為傳遞給bash -c
的單一參數。
在您的 go 程式中,您沒有互動式 shell,因此您應該使用 go 語法(而不是 shell 語法)來執行所有這些步驟。就這些步驟而言,沒有理由在轉到第一個位置(沒有必要將資料檔案寫入臨時文件,沒有必要讓/bin/cat
讀取該文件的內容,沒有必要使用子進程運行命令替換來產生一個字串(由這些內容組成),然後將其放在最終shell 的命令列中),因此忽略所有這些步驟會更明智。
以上是使用 Go 執行動態 bash 腳本,包括一行函數聲明的詳細內容。更多資訊請關注PHP中文網其他相關文章!