Go を使用して、1 行の関数宣言を含む動的 bash スクリプトを実行する
我正在用 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 を使用して、1 行の関数宣言を含む動的 bash スクリプトを実行するの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

ホットAIツール

Undresser.AI Undress
リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover
写真から衣服を削除するオンライン AI ツール。

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

AI Hentai Generator
AIヘンタイを無料で生成します。

人気の記事

ホットツール

メモ帳++7.3.1
使いやすく無料のコードエディター

SublimeText3 中国語版
中国語版、とても使いやすい

ゼンドスタジオ 13.0.1
強力な PHP 統合開発環境

ドリームウィーバー CS6
ビジュアル Web 開発ツール

SublimeText3 Mac版
神レベルのコード編集ソフト(SublimeText3)

ホットトピック











この記事では、Goのパッケージインポートメカニズム:名前付きインポート(例:インポート "fmt&quot;)および空白のインポート(例:_&quot; fmt&quot;)について説明しています。 名前付きインポートはパッケージのコンテンツにアクセス可能になり、空白のインポートはtのみを実行します

この記事では、Webアプリケーションでのページ間データ転送のためのBeegoのnewflash()関数について説明します。 newflash()を使用して、コントローラー間で一時的なメッセージ(成功、エラー、警告)を表示し、セッションメカニズムを活用することに焦点を当てています。 リミア

この記事では、MySQLクエリの結果をGO structスライスに効率的に変換することを詳しく説明しています。 データベース/SQLのスキャン方法を使用して、手動で解析することを避けて強調しています。 DBタグとロブを使用した構造フィールドマッピングのベストプラクティス

この記事では、GENICSのGOのカスタムタイプの制約について説明します。 インターフェイスがジェネリック関数の最小タイプ要件をどのように定義するかを詳しく説明し、タイプの安全性とコードの再利用性を改善します。 この記事では、制限とベストプラクティスについても説明しています

この記事では、ユニットテストのためにGOのモックとスタブを作成することを示しています。 インターフェイスの使用を強調し、模擬実装の例を提供し、模擬フォーカスを維持し、アサーションライブラリを使用するなどのベストプラクティスについて説明します。 articl

この記事では、goで効率的なファイルの書き込みを詳しく説明し、os.writefile(小さなファイルに適している)とos.openfileおよびbuffered write(大規模ファイルに最適)と比較します。 延期エラー処理、Deferを使用し、特定のエラーをチェックすることを強調します。

この記事では、GOでユニットテストを書くことで、ベストプラクティス、モッキングテクニック、効率的なテスト管理のためのツールについて説明します。

この記事では、トレースツールを使用してGOアプリケーションの実行フローを分析します。 手動および自動計装技術について説明し、Jaeger、Zipkin、Opentelemetryなどのツールを比較し、効果的なデータの視覚化を強調しています
