Keyboard shortcuts

Press or to navigate between chapters

Press S or / to search in the book

Press ? to show this help

Press Esc to hide this help

[インデックス 18240] ファイルの概要

このコミットは、Go言語のツールチェインの一部である cmd/api コマンドに関するものです。具体的には、go install コマンドを実行する際に、GOPATH 環境変数が常に正しく設定されるようにするための修正が含まれています。

コミット

commit fa6ffc6c9b8bf0945921c8710f3c0d74d1af0126
Author: Alex Brainman <alex.brainman@gmail.com>
Date:   Tue Jan 14 16:56:22 2014 +1100

    cmd/api: ensure GOPATH always points to the correct go.tools
    
    R=golang-codereviews, dave, bradfitz
    CC=golang-codereviews
    https://golang.org/cl/51000043

GitHub上でのコミットページへのリンク

https://github.com/golang/go/commit/fa6ffc6c9b8bf0945921c8710f3c0d74d1af0126

元コミット内容

cmd/api: ensure GOPATH always points to the correct go.tools

R=golang-codereviews, dave, bradfitz
CC=golang-codereviews
https://golang.org/cl/51000043

変更の背景

このコミットの背景には、cmd/api ツールが go install コマンドを実行する際に、GOPATH 環境変数の設定が意図通りに行われない可能性があったという問題があります。GOPATH はGoのワークスペースのルートディレクトリを指定する重要な環境変数であり、Goツールがソースコードやコンパイル済みバイナリを探す場所を決定します。

cmd/api は、GoのAPIの変更をチェックするためのツールであり、自身をインストールするために go install を内部的に呼び出します。この際、go install が正しい GOPATH を参照しないと、ツールのインストールが失敗したり、意図しない場所にインストールされたりする可能性がありました。

以前の実装では、os.Environ() で取得した現在の環境変数から GOARCH のみをフィルタリングし、その後に新しい GOPATH を追加していました。しかし、もし os.Environ() に既に GOPATH が含まれており、それが cmd/api が期待する go.toolsGOPATH と異なる場合、新しい GOPATH が正しく適用されない、あるいは既存の GOPATH が優先されてしまうという問題が発生する可能性がありました。このコミットは、この GOPATH の設定の堅牢性を高めることを目的としています。

前提知識の解説

GOPATH

GOPATH はGo言語におけるワークスペースの概念を定義する環境変数です。Go 1.11以降のGo Modulesの導入によりその重要性は薄れましたが、それ以前のGoのバージョンや、特定のレガシーなビルドシステムでは依然として重要な役割を果たします。

GOPATH で指定されたディレクトリ構造は以下のようになります。

  • src: ソースコードが配置される場所。各リポジトリは src/github.com/user/repo のように配置されます。
  • pkg: コンパイルされたパッケージオブジェクトが配置される場所。
  • bin: go install でインストールされた実行可能バイナリが配置される場所。

go install コマンドは、指定されたパッケージをコンパイルし、その結果のバイナリを $GOPATH/bin に配置します。また、パッケージのアーカイブファイルは $GOPATH/pkg に配置されます。

os.Environ()

Go言語の os パッケージは、オペレーティングシステムとのインタラクションを提供します。os.Environ() 関数は、現在のプロセスの環境変数を KEY=VALUE 形式の文字列スライスの形で返します。例えば、["PATH=/usr/local/bin", "HOME=/home/user", "GOPATH=/path/to/go/workspace"] のような形式です。

exec.Command

os/exec パッケージは、外部コマンドの実行を可能にします。exec.Command(name string, arg ...string) 関数は、指定されたコマンドと引数を持つ Cmd 構造体を返します。この Cmd 構造体には、コマンドの実行環境(環境変数、標準入出力など)を設定するためのフィールドが含まれています。

Cmd.Env フィールドは、実行するコマンドに渡される環境変数を設定するために使用されます。これは KEY=VALUE 形式の文字列スライスです。このフィールドが nil の場合、子プロセスは親プロセスの環境変数を継承します。しかし、明示的に設定することで、子プロセスに渡す環境変数を制御できます。

filterOut 関数 (推測)

コミットの差分から filterOut という関数が使われていることがわかります。この関数はGoの標準ライブラリには存在しないため、cmd/api 内部で定義されているユーティリティ関数であると推測されます。その名前と使われ方から、この関数は環境変数のスライス(os.Environ() の結果)と、除去したい環境変数名の可変長引数を受け取り、指定された環境変数を除外した新しい環境変数のスライスを返すものと考えられます。

例えば、filterOut(os.Environ(), "GOARCH", "GOPATH") は、現在の環境変数から GOARCHGOPATH を含むエントリを除外した環境変数のリストを返します。

技術的詳細

このコミットの技術的な核心は、go install コマンドを実行する子プロセスの環境変数を設定する際の GOPATH の扱いを改善した点にあります。

変更前のコード:

cmd.Env = append([]string{"GOPATH=" + gopath}, filterOut(os.Environ(), "GOARCH")...)

この行では、まず {"GOPATH=" + gopath} という新しいスライスを作成し、その後に os.Environ() から GOARCH を除外した残りの環境変数を追加しています。この方法の問題点は、もし os.Environ() の中に既に GOPATH が存在していた場合、filterOut 関数が GOPATH を除外しないため、結果として cmd.Env の中に GOPATH が複数回出現する可能性があったことです。Goの exec パッケージやシェルによっては、環境変数が複数回設定された場合に、最初の値が優先されるか、最後の値が優先されるか、あるいは未定義の動作になる可能性があります。いずれにしても、意図しない GOPATHgo install に渡されるリスクがありました。

変更後のコード:

cmd.Env = append(filterOut(os.Environ(), "GOARCH", "GOPATH"), "GOPATH="+gopath)

この修正では、filterOut 関数に GOPATH も含めてフィルタリングするように変更しています。これにより、os.Environ() から取得した既存の環境変数リストから、GOARCH既存の GOPATH の両方が確実に除去されます。その結果、append の最初の引数として渡されるスライスには、GOARCHGOPATH 以外の環境変数のみが含まれることになります。そして、そのスライスの末尾に、cmd/api が意図する正しい GOPATH ("GOPATH="+gopath) が追加されます。

この変更により、go install コマンドが実行される子プロセスには、cmd/api が明示的に設定した GOPATH のみが確実に渡されるようになり、環境変数による予期せぬ GOPATH の上書きや競合が防止されます。これは、ツールの堅牢性と信頼性を向上させる上で重要な修正です。

コアとなるコードの変更箇所

変更は src/cmd/api/run.go ファイルの main 関数内の一行です。

--- a/src/cmd/api/run.go
+++ b/src/cmd/api/run.go
@@ -46,7 +46,7 @@ func main() {
 	gopath := prepGoPath()

 	cmd := exec.Command("go", "install", "--tags=api_tool", "cmd/api")
-	cmd.Env = append([]string{"GOPATH=" + gopath}, filterOut(os.Environ(), "GOARCH")...)
+	cmd.Env = append(filterOut(os.Environ(), "GOARCH", "GOPATH"), "GOPATH="+gopath)
 	out, err := cmd.CombinedOutput()
 	if err != nil {
 		log.Fatalf("Error installing cmd/api: %v\\n%s", err, out)

コアとなるコードの解説

変更された行は、go install コマンドを実行するための exec.Command オブジェクト (cmd) の環境変数 (cmd.Env) を設定しています。

  • 変更前: cmd.Env = append([]string{"GOPATH=" + gopath}, filterOut(os.Environ(), "GOARCH")...) このコードは、まず {"GOPATH=" + gopath} という新しい GOPATH を含むスライスを作成し、その後に os.Environ() から GOARCH を除外した残りの環境変数を追加していました。もし os.Environ() に既に GOPATH が含まれていた場合、その既存の GOPATHfilterOut によって除去されず、新しい GOPATH と共に cmd.Env に含まれてしまう可能性がありました。

  • 変更後: cmd.Env = append(filterOut(os.Environ(), "GOARCH", "GOPATH"), "GOPATH="+gopath) この修正では、filterOut 関数に GOARCH と共に GOPATH も渡すことで、os.Environ() から取得した環境変数リストから、既存の GOPATH も確実に除去されるようにしました。これにより、cmd.Env には、GOARCHGOPATH 以外の環境変数と、cmd/api が意図する正しい GOPATH のみが含まれることになります。これにより、go install コマンドが常に正しい GOPATH を参照することが保証されます。

この変更は、GOPATH の設定に関する潜在的な競合や不整合を解消し、cmd/api ツールのインストールプロセスをより堅牢にするための重要な改善です。

関連リンク

参考にした情報源リンク

  • Go言語の公式ドキュメント
  • Go言語のソースコード (特に os および os/exec パッケージ)
  • Go言語の環境変数に関する一般的な知識
  • コミットメッセージと差分情報