首頁 >後端開發 >Golang >使用 Go 執行動態 bash 腳本,包括一行函數聲明

使用 Go 執行動態 bash 腳本,包括一行函數聲明

WBOY
WBOY轉載
2024-02-05 23:09:11851瀏覽

"使用

問題內容

我正在用 go 寫一個 bash 任務運行程序,它有一個簡單的概念:

  1. 它會讀取 taskfile ,這是一個包含任務定義(簡單的 bash 函數宣告)的 bash 腳本
  2. 它動態添加附加內容
  3. 根據傳遞的參數執行指令

這是一個簡化的範例:

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 <<eofeof 和最終的)" 也是互動式shell 的指令,而不是它調用的非互動式shell。它是運行的互動式shell cat (包含連接到其標準輸入的heredoc內容的臨時檔案),讀取cat 副本的標準輸出,然後將該資料替換為傳遞給bash -c 的單一參數。

在您的 go 程式中,您沒有互動式 shell,因此您應該使用 go 語法(而不是 shell 語法)來執行所有這些步驟。就這些步驟而言,沒有理由在轉到第一個位置(沒有必要將資料檔案寫入臨時文件,沒有必要讓/bin/cat 讀取該文件的內容,沒有必要使用子進程運行命令替換來產生一個字串(由這些內容組成),然後將其放在最終shell 的命令列中),因此忽略所有這些步驟會更明智。

以上是使用 Go 執行動態 bash 腳本,包括一行函數聲明的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:stackoverflow.com。如有侵權,請聯絡admin@php.cn刪除