// Copyright 2019 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. //go:build explicit package bootstrap_test import ( "bytes" "errors" "internal/testenv" "os" "os/exec" "path/filepath" "runtime" "testing" ) // TestExperimentToolID verifies that GOEXPERIMENT settings built // into the toolchain influence tool ids in the Go command. // This test requires bootstrapping the toolchain twice, so it's very expensive. // It must be run explicitly with -tags=explicit. // Verifies go.dev/issue/33091. func TestExperimentToolID(t *testing.T) { if testing.Short() { t.Skip("skipping test that rebuilds the entire toolchain twice") } switch runtime.GOOS { case "android", "ios", "js", "wasip1": t.Skipf("skipping because the toolchain does not have to bootstrap on GOOS=%s", runtime.GOOS) } realGoroot := testenv.GOROOT(t) // Set up GOROOT. goroot := t.TempDir() gorootSrc := filepath.Join(goroot, "src") if err := overlayDir(gorootSrc, filepath.Join(realGoroot, "src")); err != nil { t.Fatal(err) } gorootLib := filepath.Join(goroot, "lib") if err := overlayDir(gorootLib, filepath.Join(realGoroot, "lib")); err != nil { t.Fatal(err) } if err := os.WriteFile(filepath.Join(goroot, "VERSION"), []byte("go1.999"), 0666); err != nil { t.Fatal(err) } env := append(os.Environ(), "GOROOT=", "GOROOT_BOOTSTRAP="+realGoroot) // Use a clean cache. gocache := t.TempDir() env = append(env, "GOCACHE="+gocache) // Build the toolchain without GOEXPERIMENT. var makeScript string switch runtime.GOOS { case "windows": makeScript = "make.bat" case "plan9": makeScript = "make.rc" default: makeScript = "make.bash" } makeScriptPath := filepath.Join(realGoroot, "src", makeScript) runCmd(t, gorootSrc, env, makeScriptPath) // Verify compiler version string. goCmdPath := filepath.Join(goroot, "bin", "go") gotVersion := bytes.TrimSpace(runCmd(t, gorootSrc, env, goCmdPath, "tool", "compile", "-V=full")) wantVersion := []byte(`compile version go1.999`) if !bytes.Equal(gotVersion, wantVersion) { t.Errorf("compile version without experiment is unexpected:\ngot %q\nwant %q", gotVersion, wantVersion) } // Build a package in a mode not handled by the make script. runCmd(t, gorootSrc, env, goCmdPath, "build", "-race", "archive/tar") // Rebuild the toolchain with GOEXPERIMENT. env = append(env, "GOEXPERIMENT=fieldtrack") runCmd(t, gorootSrc, env, makeScriptPath) // Verify compiler version string. gotVersion = bytes.TrimSpace(runCmd(t, gorootSrc, env, goCmdPath, "tool", "compile", "-V=full")) wantVersion = []byte(`compile version go1.999 X:fieldtrack`) if !bytes.Equal(gotVersion, wantVersion) { t.Errorf("compile version with experiment is unexpected:\ngot %q\nwant %q", gotVersion, wantVersion) } // Build the same package. We should not get a cache conflict. runCmd(t, gorootSrc, env, goCmdPath, "build", "-race", "archive/tar") } func runCmd(t *testing.T, dir string, env []string, path string, args ...string) []byte { cmd := exec.Command(path, args...) cmd.Dir = dir cmd.Env = env out, err := cmd.Output() if err != nil { if ee := (*exec.ExitError)(nil); errors.As(err, &ee) { out = append(out, ee.Stderr...) } t.Fatalf("%s failed:\n%s\n%s", cmd, out, err) } return out }