| // Copyright 2015 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. |
| |
| package main_test |
| |
| import ( |
| "bytes" |
| "debug/elf" |
| "debug/macho" |
| "debug/pe" |
| "encoding/binary" |
| "flag" |
| "fmt" |
| "go/format" |
| "internal/race" |
| "internal/testenv" |
| "io" |
| "io/fs" |
| "log" |
| "os" |
| "os/exec" |
| "path/filepath" |
| "regexp" |
| "runtime" |
| "strconv" |
| "strings" |
| "testing" |
| "time" |
| |
| "cmd/go/internal/cache" |
| "cmd/go/internal/cfg" |
| "cmd/go/internal/robustio" |
| "cmd/go/internal/work" |
| "cmd/internal/sys" |
| ) |
| |
| func init() { |
| // GOVCS defaults to public:git|hg,private:all, |
| // which breaks many tests here - they can't use non-git, non-hg VCS at all! |
| // Change to fully permissive. |
| // The tests of the GOVCS setting itself are in ../../testdata/script/govcs.txt. |
| os.Setenv("GOVCS", "*:all") |
| } |
| |
| var ( |
| canRace = false // whether we can run the race detector |
| canCgo = false // whether we can use cgo |
| canMSan = false // whether we can run the memory sanitizer |
| ) |
| |
| var exeSuffix string = func() string { |
| if runtime.GOOS == "windows" { |
| return ".exe" |
| } |
| return "" |
| }() |
| |
| func tooSlow(t *testing.T) { |
| if testing.Short() { |
| // In -short mode; skip test, except run it on the {darwin,linux,windows}/amd64 builders. |
| if testenv.Builder() != "" && runtime.GOARCH == "amd64" && (runtime.GOOS == "linux" || runtime.GOOS == "darwin" || runtime.GOOS == "windows") { |
| return |
| } |
| t.Helper() |
| t.Skip("skipping test in -short mode") |
| } |
| } |
| |
| // testGOROOT is the GOROOT to use when running testgo, a cmd/go binary |
| // build from this process's current GOROOT, but run from a different |
| // (temp) directory. |
| var testGOROOT string |
| |
| var testCC string |
| var testGOCACHE string |
| |
| var testGo string |
| var testTmpDir string |
| var testBin string |
| |
| // The TestMain function creates a go command for testing purposes and |
| // deletes it after the tests have been run. |
| func TestMain(m *testing.M) { |
| // $GO_GCFLAGS a compiler debug flag known to cmd/dist, make.bash, etc. |
| // It is not a standard go command flag; use os.Getenv, not cfg.Getenv. |
| if os.Getenv("GO_GCFLAGS") != "" { |
| fmt.Fprintf(os.Stderr, "testing: warning: no tests to run\n") // magic string for cmd/go |
| fmt.Printf("cmd/go test is not compatible with $GO_GCFLAGS being set\n") |
| fmt.Printf("SKIP\n") |
| return |
| } |
| |
| flag.Parse() |
| |
| if *proxyAddr != "" { |
| StartProxy() |
| select {} |
| } |
| |
| // Run with a temporary TMPDIR to check that the tests don't |
| // leave anything behind. |
| topTmpdir, err := os.MkdirTemp("", "cmd-go-test-") |
| if err != nil { |
| log.Fatal(err) |
| } |
| if !*testWork { |
| defer removeAll(topTmpdir) |
| } |
| os.Setenv(tempEnvName(), topTmpdir) |
| |
| dir, err := os.MkdirTemp(topTmpdir, "tmpdir") |
| if err != nil { |
| log.Fatal(err) |
| } |
| testTmpDir = dir |
| if !*testWork { |
| defer removeAll(testTmpDir) |
| } |
| |
| testGOCACHE = cache.DefaultDir() |
| if testenv.HasGoBuild() { |
| testBin = filepath.Join(testTmpDir, "testbin") |
| if err := os.Mkdir(testBin, 0777); err != nil { |
| log.Fatal(err) |
| } |
| testGo = filepath.Join(testBin, "go"+exeSuffix) |
| args := []string{"build", "-tags", "testgo", "-o", testGo} |
| if race.Enabled { |
| args = append(args, "-race") |
| } |
| gotool, err := testenv.GoTool() |
| if err != nil { |
| fmt.Fprintln(os.Stderr, err) |
| os.Exit(2) |
| } |
| |
| goEnv := func(name string) string { |
| out, err := exec.Command(gotool, "env", name).CombinedOutput() |
| if err != nil { |
| fmt.Fprintf(os.Stderr, "go env %s: %v\n%s", name, err, out) |
| os.Exit(2) |
| } |
| return strings.TrimSpace(string(out)) |
| } |
| testGOROOT = goEnv("GOROOT") |
| os.Setenv("TESTGO_GOROOT", testGOROOT) |
| // Ensure that GOROOT is set explicitly. |
| // Otherwise, if the toolchain was built with GOROOT_FINAL set but has not |
| // yet been moved to its final location, programs that invoke runtime.GOROOT |
| // may accidentally use the wrong path. |
| os.Setenv("GOROOT", testGOROOT) |
| |
| // The whole GOROOT/pkg tree was installed using the GOHOSTOS/GOHOSTARCH |
| // toolchain (installed in GOROOT/pkg/tool/GOHOSTOS_GOHOSTARCH). |
| // The testgo.exe we are about to create will be built for GOOS/GOARCH, |
| // which means it will use the GOOS/GOARCH toolchain |
| // (installed in GOROOT/pkg/tool/GOOS_GOARCH). |
| // If these are not the same toolchain, then the entire standard library |
| // will look out of date (the compilers in those two different tool directories |
| // are built for different architectures and have different build IDs), |
| // which will cause many tests to do unnecessary rebuilds and some |
| // tests to attempt to overwrite the installed standard library. |
| // Bail out entirely in this case. |
| hostGOOS := goEnv("GOHOSTOS") |
| hostGOARCH := goEnv("GOHOSTARCH") |
| if hostGOOS != runtime.GOOS || hostGOARCH != runtime.GOARCH { |
| fmt.Fprintf(os.Stderr, "testing: warning: no tests to run\n") // magic string for cmd/go |
| fmt.Printf("cmd/go test is not compatible with GOOS/GOARCH != GOHOSTOS/GOHOSTARCH (%s/%s != %s/%s)\n", runtime.GOOS, runtime.GOARCH, hostGOOS, hostGOARCH) |
| fmt.Printf("SKIP\n") |
| return |
| } |
| |
| buildCmd := exec.Command(gotool, args...) |
| buildCmd.Env = append(os.Environ(), "GOFLAGS=-mod=vendor") |
| out, err := buildCmd.CombinedOutput() |
| if err != nil { |
| fmt.Fprintf(os.Stderr, "building testgo failed: %v\n%s", err, out) |
| os.Exit(2) |
| } |
| |
| out, err = exec.Command(gotool, "env", "CC").CombinedOutput() |
| if err != nil { |
| fmt.Fprintf(os.Stderr, "could not find testing CC: %v\n%s", err, out) |
| os.Exit(2) |
| } |
| testCC = strings.TrimSpace(string(out)) |
| |
| cmd := exec.Command(testGo, "env", "CGO_ENABLED") |
| cmd.Stderr = new(strings.Builder) |
| if out, err := cmd.Output(); err != nil { |
| fmt.Fprintf(os.Stderr, "running testgo failed: %v\n%s", err, cmd.Stderr) |
| os.Exit(2) |
| } else { |
| canCgo, err = strconv.ParseBool(strings.TrimSpace(string(out))) |
| if err != nil { |
| fmt.Fprintf(os.Stderr, "can't parse go env CGO_ENABLED output: %v\n", strings.TrimSpace(string(out))) |
| } |
| } |
| |
| out, err = exec.Command(gotool, "env", "GOCACHE").CombinedOutput() |
| if err != nil { |
| fmt.Fprintf(os.Stderr, "could not find testing GOCACHE: %v\n%s", err, out) |
| os.Exit(2) |
| } |
| testGOCACHE = strings.TrimSpace(string(out)) |
| |
| canMSan = canCgo && sys.MSanSupported(runtime.GOOS, runtime.GOARCH) |
| canRace = canCgo && sys.RaceDetectorSupported(runtime.GOOS, runtime.GOARCH) |
| // The race detector doesn't work on Alpine Linux: |
| // golang.org/issue/14481 |
| // gccgo does not support the race detector. |
| if isAlpineLinux() || runtime.Compiler == "gccgo" { |
| canRace = false |
| } |
| } |
| // Don't let these environment variables confuse the test. |
| os.Setenv("GOENV", "off") |
| os.Unsetenv("GOFLAGS") |
| os.Unsetenv("GOBIN") |
| os.Unsetenv("GOPATH") |
| os.Unsetenv("GIT_ALLOW_PROTOCOL") |
| os.Setenv("HOME", "/test-go-home-does-not-exist") |
| // On some systems the default C compiler is ccache. |
| // Setting HOME to a non-existent directory will break |
| // those systems. Disable ccache and use real compiler. Issue 17668. |
| os.Setenv("CCACHE_DISABLE", "1") |
| if cfg.Getenv("GOCACHE") == "" { |
| os.Setenv("GOCACHE", testGOCACHE) // because $HOME is gone |
| } |
| |
| r := m.Run() |
| if !*testWork { |
| removeAll(testTmpDir) // os.Exit won't run defer |
| } |
| |
| if !*testWork { |
| // There shouldn't be anything left in topTmpdir. |
| dirf, err := os.Open(topTmpdir) |
| if err != nil { |
| log.Fatal(err) |
| } |
| names, err := dirf.Readdirnames(0) |
| if err != nil { |
| log.Fatal(err) |
| } |
| if len(names) > 0 { |
| log.Fatalf("unexpected files left in tmpdir: %v", names) |
| } |
| |
| removeAll(topTmpdir) |
| } |
| |
| os.Exit(r) |
| } |
| |
| func isAlpineLinux() bool { |
| if runtime.GOOS != "linux" { |
| return false |
| } |
| fi, err := os.Lstat("/etc/alpine-release") |
| return err == nil && fi.Mode().IsRegular() |
| } |
| |
| // The length of an mtime tick on this system. This is an estimate of |
| // how long we need to sleep to ensure that the mtime of two files is |
| // different. |
| // We used to try to be clever but that didn't always work (see golang.org/issue/12205). |
| var mtimeTick time.Duration = 1 * time.Second |
| |
| // Manage a single run of the testgo binary. |
| type testgoData struct { |
| t *testing.T |
| temps []string |
| env []string |
| tempdir string |
| ran bool |
| inParallel bool |
| stdout, stderr bytes.Buffer |
| execDir string // dir for tg.run |
| } |
| |
| // skipIfGccgo skips the test if using gccgo. |
| func skipIfGccgo(t *testing.T, msg string) { |
| if runtime.Compiler == "gccgo" { |
| t.Skipf("skipping test not supported on gccgo: %s", msg) |
| } |
| } |
| |
| // testgo sets up for a test that runs testgo. |
| func testgo(t *testing.T) *testgoData { |
| t.Helper() |
| testenv.MustHaveGoBuild(t) |
| testenv.SkipIfShortAndSlow(t) |
| |
| return &testgoData{t: t} |
| } |
| |
| // must gives a fatal error if err is not nil. |
| func (tg *testgoData) must(err error) { |
| tg.t.Helper() |
| if err != nil { |
| tg.t.Fatal(err) |
| } |
| } |
| |
| // check gives a test non-fatal error if err is not nil. |
| func (tg *testgoData) check(err error) { |
| tg.t.Helper() |
| if err != nil { |
| tg.t.Error(err) |
| } |
| } |
| |
| // parallel runs the test in parallel by calling t.Parallel. |
| func (tg *testgoData) parallel() { |
| tg.t.Helper() |
| if tg.ran { |
| tg.t.Fatal("internal testsuite error: call to parallel after run") |
| } |
| for _, e := range tg.env { |
| if strings.HasPrefix(e, "GOROOT=") || strings.HasPrefix(e, "GOPATH=") || strings.HasPrefix(e, "GOBIN=") { |
| val := e[strings.Index(e, "=")+1:] |
| if strings.HasPrefix(val, "testdata") || strings.HasPrefix(val, "./testdata") { |
| tg.t.Fatalf("internal testsuite error: call to parallel with testdata in environment (%s)", e) |
| } |
| } |
| } |
| tg.inParallel = true |
| tg.t.Parallel() |
| } |
| |
| // pwd returns the current directory. |
| func (tg *testgoData) pwd() string { |
| tg.t.Helper() |
| wd, err := os.Getwd() |
| if err != nil { |
| tg.t.Fatalf("could not get working directory: %v", err) |
| } |
| return wd |
| } |
| |
| // sleep sleeps for one tick, where a tick is a conservative estimate |
| // of how long it takes for a file modification to get a different |
| // mtime. |
| func (tg *testgoData) sleep() { |
| time.Sleep(mtimeTick) |
| } |
| |
| // setenv sets an environment variable to use when running the test go |
| // command. |
| func (tg *testgoData) setenv(name, val string) { |
| tg.t.Helper() |
| if tg.inParallel && (name == "GOROOT" || name == "GOPATH" || name == "GOBIN") && (strings.HasPrefix(val, "testdata") || strings.HasPrefix(val, "./testdata")) { |
| tg.t.Fatalf("internal testsuite error: call to setenv with testdata (%s=%s) after parallel", name, val) |
| } |
| tg.unsetenv(name) |
| tg.env = append(tg.env, name+"="+val) |
| } |
| |
| // unsetenv removes an environment variable. |
| func (tg *testgoData) unsetenv(name string) { |
| if tg.env == nil { |
| tg.env = append([]string(nil), os.Environ()...) |
| tg.env = append(tg.env, "GO111MODULE=off") |
| } |
| for i, v := range tg.env { |
| if strings.HasPrefix(v, name+"=") { |
| tg.env = append(tg.env[:i], tg.env[i+1:]...) |
| break |
| } |
| } |
| } |
| |
| func (tg *testgoData) goTool() string { |
| return testGo |
| } |
| |
| // doRun runs the test go command, recording stdout and stderr and |
| // returning exit status. |
| func (tg *testgoData) doRun(args []string) error { |
| tg.t.Helper() |
| if tg.inParallel { |
| for _, arg := range args { |
| if strings.HasPrefix(arg, "testdata") || strings.HasPrefix(arg, "./testdata") { |
| tg.t.Fatal("internal testsuite error: parallel run using testdata") |
| } |
| } |
| } |
| |
| hasGoroot := false |
| for _, v := range tg.env { |
| if strings.HasPrefix(v, "GOROOT=") { |
| hasGoroot = true |
| break |
| } |
| } |
| prog := tg.goTool() |
| if !hasGoroot { |
| tg.setenv("GOROOT", testGOROOT) |
| } |
| |
| tg.t.Logf("running testgo %v", args) |
| cmd := exec.Command(prog, args...) |
| tg.stdout.Reset() |
| tg.stderr.Reset() |
| cmd.Dir = tg.execDir |
| cmd.Stdout = &tg.stdout |
| cmd.Stderr = &tg.stderr |
| cmd.Env = tg.env |
| status := cmd.Run() |
| if tg.stdout.Len() > 0 { |
| tg.t.Log("standard output:") |
| tg.t.Log(tg.stdout.String()) |
| } |
| if tg.stderr.Len() > 0 { |
| tg.t.Log("standard error:") |
| tg.t.Log(tg.stderr.String()) |
| } |
| tg.ran = true |
| return status |
| } |
| |
| // run runs the test go command, and expects it to succeed. |
| func (tg *testgoData) run(args ...string) { |
| tg.t.Helper() |
| if status := tg.doRun(args); status != nil { |
| wd, _ := os.Getwd() |
| tg.t.Logf("go %v failed unexpectedly in %s: %v", args, wd, status) |
| tg.t.FailNow() |
| } |
| } |
| |
| // runFail runs the test go command, and expects it to fail. |
| func (tg *testgoData) runFail(args ...string) { |
| tg.t.Helper() |
| if status := tg.doRun(args); status == nil { |
| tg.t.Fatal("testgo succeeded unexpectedly") |
| } else { |
| tg.t.Log("testgo failed as expected:", status) |
| } |
| } |
| |
| // runGit runs a git command, and expects it to succeed. |
| func (tg *testgoData) runGit(dir string, args ...string) { |
| tg.t.Helper() |
| cmd := exec.Command("git", args...) |
| tg.stdout.Reset() |
| tg.stderr.Reset() |
| cmd.Stdout = &tg.stdout |
| cmd.Stderr = &tg.stderr |
| cmd.Dir = dir |
| cmd.Env = tg.env |
| status := cmd.Run() |
| if tg.stdout.Len() > 0 { |
| tg.t.Log("git standard output:") |
| tg.t.Log(tg.stdout.String()) |
| } |
| if tg.stderr.Len() > 0 { |
| tg.t.Log("git standard error:") |
| tg.t.Log(tg.stderr.String()) |
| } |
| if status != nil { |
| tg.t.Logf("git %v failed unexpectedly: %v", args, status) |
| tg.t.FailNow() |
| } |
| } |
| |
| // getStdout returns standard output of the testgo run as a string. |
| func (tg *testgoData) getStdout() string { |
| tg.t.Helper() |
| if !tg.ran { |
| tg.t.Fatal("internal testsuite error: stdout called before run") |
| } |
| return tg.stdout.String() |
| } |
| |
| // getStderr returns standard error of the testgo run as a string. |
| func (tg *testgoData) getStderr() string { |
| tg.t.Helper() |
| if !tg.ran { |
| tg.t.Fatal("internal testsuite error: stdout called before run") |
| } |
| return tg.stderr.String() |
| } |
| |
| // doGrepMatch looks for a regular expression in a buffer, and returns |
| // whether it is found. The regular expression is matched against |
| // each line separately, as with the grep command. |
| func (tg *testgoData) doGrepMatch(match string, b *bytes.Buffer) bool { |
| tg.t.Helper() |
| if !tg.ran { |
| tg.t.Fatal("internal testsuite error: grep called before run") |
| } |
| re := regexp.MustCompile(match) |
| for _, ln := range bytes.Split(b.Bytes(), []byte{'\n'}) { |
| if re.Match(ln) { |
| return true |
| } |
| } |
| return false |
| } |
| |
| // doGrep looks for a regular expression in a buffer and fails if it |
| // is not found. The name argument is the name of the output we are |
| // searching, "output" or "error". The msg argument is logged on |
| // failure. |
| func (tg *testgoData) doGrep(match string, b *bytes.Buffer, name, msg string) { |
| tg.t.Helper() |
| if !tg.doGrepMatch(match, b) { |
| tg.t.Log(msg) |
| tg.t.Logf("pattern %v not found in standard %s", match, name) |
| tg.t.FailNow() |
| } |
| } |
| |
| // grepStdout looks for a regular expression in the test run's |
| // standard output and fails, logging msg, if it is not found. |
| func (tg *testgoData) grepStdout(match, msg string) { |
| tg.t.Helper() |
| tg.doGrep(match, &tg.stdout, "output", msg) |
| } |
| |
| // grepStderr looks for a regular expression in the test run's |
| // standard error and fails, logging msg, if it is not found. |
| func (tg *testgoData) grepStderr(match, msg string) { |
| tg.t.Helper() |
| tg.doGrep(match, &tg.stderr, "error", msg) |
| } |
| |
| // grepBoth looks for a regular expression in the test run's standard |
| // output or stand error and fails, logging msg, if it is not found. |
| func (tg *testgoData) grepBoth(match, msg string) { |
| tg.t.Helper() |
| if !tg.doGrepMatch(match, &tg.stdout) && !tg.doGrepMatch(match, &tg.stderr) { |
| tg.t.Log(msg) |
| tg.t.Logf("pattern %v not found in standard output or standard error", match) |
| tg.t.FailNow() |
| } |
| } |
| |
| // doGrepNot looks for a regular expression in a buffer and fails if |
| // it is found. The name and msg arguments are as for doGrep. |
| func (tg *testgoData) doGrepNot(match string, b *bytes.Buffer, name, msg string) { |
| tg.t.Helper() |
| if tg.doGrepMatch(match, b) { |
| tg.t.Log(msg) |
| tg.t.Logf("pattern %v found unexpectedly in standard %s", match, name) |
| tg.t.FailNow() |
| } |
| } |
| |
| // grepStdoutNot looks for a regular expression in the test run's |
| // standard output and fails, logging msg, if it is found. |
| func (tg *testgoData) grepStdoutNot(match, msg string) { |
| tg.t.Helper() |
| tg.doGrepNot(match, &tg.stdout, "output", msg) |
| } |
| |
| // grepStderrNot looks for a regular expression in the test run's |
| // standard error and fails, logging msg, if it is found. |
| func (tg *testgoData) grepStderrNot(match, msg string) { |
| tg.t.Helper() |
| tg.doGrepNot(match, &tg.stderr, "error", msg) |
| } |
| |
| // grepBothNot looks for a regular expression in the test run's |
| // standard output or standard error and fails, logging msg, if it is |
| // found. |
| func (tg *testgoData) grepBothNot(match, msg string) { |
| tg.t.Helper() |
| if tg.doGrepMatch(match, &tg.stdout) || tg.doGrepMatch(match, &tg.stderr) { |
| tg.t.Log(msg) |
| tg.t.Fatalf("pattern %v found unexpectedly in standard output or standard error", match) |
| } |
| } |
| |
| // doGrepCount counts the number of times a regexp is seen in a buffer. |
| func (tg *testgoData) doGrepCount(match string, b *bytes.Buffer) int { |
| tg.t.Helper() |
| if !tg.ran { |
| tg.t.Fatal("internal testsuite error: doGrepCount called before run") |
| } |
| re := regexp.MustCompile(match) |
| c := 0 |
| for _, ln := range bytes.Split(b.Bytes(), []byte{'\n'}) { |
| if re.Match(ln) { |
| c++ |
| } |
| } |
| return c |
| } |
| |
| // grepCountBoth returns the number of times a regexp is seen in both |
| // standard output and standard error. |
| func (tg *testgoData) grepCountBoth(match string) int { |
| tg.t.Helper() |
| return tg.doGrepCount(match, &tg.stdout) + tg.doGrepCount(match, &tg.stderr) |
| } |
| |
| // creatingTemp records that the test plans to create a temporary file |
| // or directory. If the file or directory exists already, it will be |
| // removed. When the test completes, the file or directory will be |
| // removed if it exists. |
| func (tg *testgoData) creatingTemp(path string) { |
| tg.t.Helper() |
| if filepath.IsAbs(path) && !strings.HasPrefix(path, tg.tempdir) { |
| tg.t.Fatalf("internal testsuite error: creatingTemp(%q) with absolute path not in temporary directory", path) |
| } |
| tg.must(robustio.RemoveAll(path)) |
| tg.temps = append(tg.temps, path) |
| } |
| |
| // makeTempdir makes a temporary directory for a run of testgo. If |
| // the temporary directory was already created, this does nothing. |
| func (tg *testgoData) makeTempdir() { |
| tg.t.Helper() |
| if tg.tempdir == "" { |
| var err error |
| tg.tempdir, err = os.MkdirTemp("", "gotest") |
| tg.must(err) |
| } |
| } |
| |
| // tempFile adds a temporary file for a run of testgo. |
| func (tg *testgoData) tempFile(path, contents string) { |
| tg.t.Helper() |
| tg.makeTempdir() |
| tg.must(os.MkdirAll(filepath.Join(tg.tempdir, filepath.Dir(path)), 0755)) |
| bytes := []byte(contents) |
| if strings.HasSuffix(path, ".go") { |
| formatted, err := format.Source(bytes) |
| if err == nil { |
| bytes = formatted |
| } |
| } |
| tg.must(os.WriteFile(filepath.Join(tg.tempdir, path), bytes, 0644)) |
| } |
| |
| // tempDir adds a temporary directory for a run of testgo. |
| func (tg *testgoData) tempDir(path string) { |
| tg.t.Helper() |
| tg.makeTempdir() |
| if err := os.MkdirAll(filepath.Join(tg.tempdir, path), 0755); err != nil && !os.IsExist(err) { |
| tg.t.Fatal(err) |
| } |
| } |
| |
| // path returns the absolute pathname to file with the temporary |
| // directory. |
| func (tg *testgoData) path(name string) string { |
| tg.t.Helper() |
| if tg.tempdir == "" { |
| tg.t.Fatalf("internal testsuite error: path(%q) with no tempdir", name) |
| } |
| if name == "." { |
| return tg.tempdir |
| } |
| return filepath.Join(tg.tempdir, name) |
| } |
| |
| // mustExist fails if path does not exist. |
| func (tg *testgoData) mustExist(path string) { |
| tg.t.Helper() |
| if _, err := os.Stat(path); err != nil { |
| if os.IsNotExist(err) { |
| tg.t.Fatalf("%s does not exist but should", path) |
| } |
| tg.t.Fatalf("%s stat failed: %v", path, err) |
| } |
| } |
| |
| // mustNotExist fails if path exists. |
| func (tg *testgoData) mustNotExist(path string) { |
| tg.t.Helper() |
| if _, err := os.Stat(path); err == nil || !os.IsNotExist(err) { |
| tg.t.Fatalf("%s exists but should not (%v)", path, err) |
| } |
| } |
| |
| // mustHaveContent succeeds if filePath is a path to a file, |
| // and that file is readable and not empty. |
| func (tg *testgoData) mustHaveContent(filePath string) { |
| tg.mustExist(filePath) |
| f, err := os.Stat(filePath) |
| if err != nil { |
| tg.t.Fatal(err) |
| } |
| if f.Size() == 0 { |
| tg.t.Fatalf("expected %s to have data, but is empty", filePath) |
| } |
| } |
| |
| // wantExecutable fails with msg if path is not executable. |
| func (tg *testgoData) wantExecutable(path, msg string) { |
| tg.t.Helper() |
| if st, err := os.Stat(path); err != nil { |
| if !os.IsNotExist(err) { |
| tg.t.Log(err) |
| } |
| tg.t.Fatal(msg) |
| } else { |
| if runtime.GOOS != "windows" && st.Mode()&0111 == 0 { |
| tg.t.Fatalf("binary %s exists but is not executable", path) |
| } |
| } |
| } |
| |
| // isStale reports whether pkg is stale, and why |
| func (tg *testgoData) isStale(pkg string) (bool, string) { |
| tg.t.Helper() |
| tg.run("list", "-f", "{{.Stale}}:{{.StaleReason}}", pkg) |
| v := strings.TrimSpace(tg.getStdout()) |
| f := strings.SplitN(v, ":", 2) |
| if len(f) == 2 { |
| switch f[0] { |
| case "true": |
| return true, f[1] |
| case "false": |
| return false, f[1] |
| } |
| } |
| tg.t.Fatalf("unexpected output checking staleness of package %v: %v", pkg, v) |
| panic("unreachable") |
| } |
| |
| // wantStale fails with msg if pkg is not stale. |
| func (tg *testgoData) wantStale(pkg, reason, msg string) { |
| tg.t.Helper() |
| stale, why := tg.isStale(pkg) |
| if !stale { |
| tg.t.Fatal(msg) |
| } |
| // We always accept the reason as being "not installed but |
| // available in build cache", because when that is the case go |
| // list doesn't try to sort out the underlying reason why the |
| // package is not installed. |
| if reason == "" && why != "" || !strings.Contains(why, reason) && !strings.Contains(why, "not installed but available in build cache") { |
| tg.t.Errorf("wrong reason for Stale=true: %q, want %q", why, reason) |
| } |
| } |
| |
| // wantNotStale fails with msg if pkg is stale. |
| func (tg *testgoData) wantNotStale(pkg, reason, msg string) { |
| tg.t.Helper() |
| stale, why := tg.isStale(pkg) |
| if stale { |
| tg.t.Fatal(msg) |
| } |
| if reason == "" && why != "" || !strings.Contains(why, reason) { |
| tg.t.Errorf("wrong reason for Stale=false: %q, want %q", why, reason) |
| } |
| } |
| |
| // If -testwork is specified, the test prints the name of the temp directory |
| // and does not remove it when done, so that a programmer can |
| // poke at the test file tree afterward. |
| var testWork = flag.Bool("testwork", false, "") |
| |
| // cleanup cleans up a test that runs testgo. |
| func (tg *testgoData) cleanup() { |
| tg.t.Helper() |
| if *testWork { |
| tg.t.Logf("TESTWORK=%s\n", tg.path(".")) |
| return |
| } |
| for _, path := range tg.temps { |
| tg.check(removeAll(path)) |
| } |
| if tg.tempdir != "" { |
| tg.check(removeAll(tg.tempdir)) |
| } |
| } |
| |
| func removeAll(dir string) error { |
| // module cache has 0444 directories; |
| // make them writable in order to remove content. |
| filepath.WalkDir(dir, func(path string, info fs.DirEntry, err error) error { |
| // chmod not only directories, but also things that we couldn't even stat |
| // due to permission errors: they may also be unreadable directories. |
| if err != nil || info.IsDir() { |
| os.Chmod(path, 0777) |
| } |
| return nil |
| }) |
| return robustio.RemoveAll(dir) |
| } |
| |
| // failSSH puts an ssh executable in the PATH that always fails. |
| // This is to stub out uses of ssh by go get. |
| func (tg *testgoData) failSSH() { |
| tg.t.Helper() |
| wd, err := os.Getwd() |
| if err != nil { |
| tg.t.Fatal(err) |
| } |
| fail := filepath.Join(wd, "testdata/failssh") |
| tg.setenv("PATH", fmt.Sprintf("%v%c%v", fail, filepath.ListSeparator, os.Getenv("PATH"))) |
| } |
| |
| func TestNewReleaseRebuildsStalePackagesInGOPATH(t *testing.T) { |
| if testing.Short() { |
| t.Skip("skipping lengthy test in short mode") |
| } |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| |
| // Copy the runtime packages into a temporary GOROOT |
| // so that we can change files. |
| for _, copydir := range []string{ |
| "src/runtime", |
| "src/internal/abi", |
| "src/internal/bytealg", |
| "src/internal/cpu", |
| "src/math/bits", |
| "src/unsafe", |
| filepath.Join("pkg", runtime.GOOS+"_"+runtime.GOARCH), |
| filepath.Join("pkg/tool", runtime.GOOS+"_"+runtime.GOARCH), |
| "pkg/include", |
| } { |
| srcdir := filepath.Join(testGOROOT, copydir) |
| tg.tempDir(filepath.Join("goroot", copydir)) |
| err := filepath.WalkDir(srcdir, |
| func(path string, info fs.DirEntry, err error) error { |
| if err != nil { |
| return err |
| } |
| if info.IsDir() { |
| return nil |
| } |
| srcrel, err := filepath.Rel(srcdir, path) |
| if err != nil { |
| return err |
| } |
| dest := filepath.Join("goroot", copydir, srcrel) |
| data, err := os.ReadFile(path) |
| if err != nil { |
| return err |
| } |
| tg.tempFile(dest, string(data)) |
| if strings.Contains(copydir, filepath.Join("pkg", "tool")) { |
| os.Chmod(tg.path(dest), 0777) |
| } |
| return nil |
| }) |
| if err != nil { |
| t.Fatal(err) |
| } |
| } |
| tg.setenv("GOROOT", tg.path("goroot")) |
| |
| addVar := func(name string, idx int) (restore func()) { |
| data, err := os.ReadFile(name) |
| if err != nil { |
| t.Fatal(err) |
| } |
| old := data |
| data = append(data, fmt.Sprintf("var DummyUnusedVar%d bool\n", idx)...) |
| if err := os.WriteFile(name, append(data, '\n'), 0666); err != nil { |
| t.Fatal(err) |
| } |
| tg.sleep() |
| return func() { |
| if err := os.WriteFile(name, old, 0666); err != nil { |
| t.Fatal(err) |
| } |
| } |
| } |
| |
| // Every main package depends on the "runtime". |
| tg.tempFile("d1/src/p1/p1.go", `package main; func main(){}`) |
| tg.setenv("GOPATH", tg.path("d1")) |
| // Pass -i flag to rebuild everything outdated. |
| tg.run("install", "-i", "p1") |
| tg.wantNotStale("p1", "", "./testgo list claims p1 is stale, incorrectly, before any changes") |
| |
| // Changing mtime of runtime/internal/sys/sys.go |
| // should have no effect: only the content matters. |
| // In fact this should be true even outside a release branch. |
| sys := tg.path("goroot/src/runtime/internal/sys/sys.go") |
| tg.sleep() |
| restore := addVar(sys, 0) |
| restore() |
| tg.wantNotStale("p1", "", "./testgo list claims p1 is stale, incorrectly, after updating mtime of runtime/internal/sys/sys.go") |
| |
| // But changing content of any file should have an effect. |
| // Previously zversion.go was the only one that mattered; |
| // now they all matter, so keep using sys.go. |
| restore = addVar(sys, 1) |
| defer restore() |
| tg.wantStale("p1", "stale dependency: runtime/internal/sys", "./testgo list claims p1 is NOT stale, incorrectly, after changing sys.go") |
| restore() |
| tg.wantNotStale("p1", "", "./testgo list claims p1 is stale, incorrectly, after changing back to old release") |
| addVar(sys, 2) |
| tg.wantStale("p1", "stale dependency: runtime", "./testgo list claims p1 is NOT stale, incorrectly, after changing sys.go again") |
| tg.run("install", "-i", "p1") |
| tg.wantNotStale("p1", "", "./testgo list claims p1 is stale after building with new release") |
| |
| // Restore to "old" release. |
| restore() |
| tg.wantStale("p1", "stale dependency: runtime/internal/sys", "./testgo list claims p1 is NOT stale, incorrectly, after restoring sys.go") |
| tg.run("install", "-i", "p1") |
| tg.wantNotStale("p1", "", "./testgo list claims p1 is stale after building with old release") |
| } |
| |
| // cmd/go: custom import path checking should not apply to Go packages without import comment. |
| func TestIssue10952(t *testing.T) { |
| testenv.MustHaveExternalNetwork(t) |
| testenv.MustHaveExecPath(t, "git") |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempDir("src") |
| tg.setenv("GOPATH", tg.path(".")) |
| const importPath = "github.com/zombiezen/go-get-issue-10952" |
| tg.run("get", "-d", "-u", importPath) |
| repoDir := tg.path("src/" + importPath) |
| tg.runGit(repoDir, "remote", "set-url", "origin", "https://"+importPath+".git") |
| tg.run("get", "-d", "-u", importPath) |
| } |
| |
| func TestIssue16471(t *testing.T) { |
| testenv.MustHaveExternalNetwork(t) |
| testenv.MustHaveExecPath(t, "git") |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempDir("src") |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.must(os.MkdirAll(tg.path("src/rsc.io/go-get-issue-10952"), 0755)) |
| tg.runGit(tg.path("src/rsc.io"), "clone", "https://github.com/zombiezen/go-get-issue-10952") |
| tg.runFail("get", "-u", "rsc.io/go-get-issue-10952") |
| tg.grepStderr("rsc.io/go-get-issue-10952 is a custom import path for https://github.com/rsc/go-get-issue-10952, but .* is checked out from https://github.com/zombiezen/go-get-issue-10952", "did not detect updated import path") |
| } |
| |
| // Test git clone URL that uses SCP-like syntax and custom import path checking. |
| func TestIssue11457(t *testing.T) { |
| testenv.MustHaveExternalNetwork(t) |
| testenv.MustHaveExecPath(t, "git") |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempDir("src") |
| tg.setenv("GOPATH", tg.path(".")) |
| const importPath = "rsc.io/go-get-issue-11457" |
| tg.run("get", "-d", "-u", importPath) |
| repoDir := tg.path("src/" + importPath) |
| tg.runGit(repoDir, "remote", "set-url", "origin", "git@github.com:rsc/go-get-issue-11457") |
| |
| // At this time, custom import path checking compares remotes verbatim (rather than |
| // just the host and path, skipping scheme and user), so we expect go get -u to fail. |
| // However, the goal of this test is to verify that gitRemoteRepo correctly parsed |
| // the SCP-like syntax, and we expect it to appear in the error message. |
| tg.runFail("get", "-d", "-u", importPath) |
| want := " is checked out from ssh://git@github.com/rsc/go-get-issue-11457" |
| if !strings.HasSuffix(strings.TrimSpace(tg.getStderr()), want) { |
| t.Error("expected clone URL to appear in stderr") |
| } |
| } |
| |
| func TestGetGitDefaultBranch(t *testing.T) { |
| testenv.MustHaveExternalNetwork(t) |
| testenv.MustHaveExecPath(t, "git") |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempDir("src") |
| tg.setenv("GOPATH", tg.path(".")) |
| |
| // This repo has two branches, master and another-branch. |
| // The another-branch is the default that you get from 'git clone'. |
| // The go get command variants should not override this. |
| const importPath = "github.com/rsc/go-get-default-branch" |
| |
| tg.run("get", "-d", importPath) |
| repoDir := tg.path("src/" + importPath) |
| tg.runGit(repoDir, "branch", "--contains", "HEAD") |
| tg.grepStdout(`\* another-branch`, "not on correct default branch") |
| |
| tg.run("get", "-d", "-u", importPath) |
| tg.runGit(repoDir, "branch", "--contains", "HEAD") |
| tg.grepStdout(`\* another-branch`, "not on correct default branch") |
| } |
| |
| // Security issue. Don't disable. See golang.org/issue/22125. |
| func TestAccidentalGitCheckout(t *testing.T) { |
| testenv.MustHaveExternalNetwork(t) |
| testenv.MustHaveExecPath(t, "git") |
| testenv.MustHaveExecPath(t, "svn") |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempDir("src") |
| |
| tg.setenv("GOPATH", tg.path(".")) |
| |
| tg.runFail("get", "-u", "vcs-test.golang.org/go/test1-svn-git") |
| tg.grepStderr("src[\\\\/]vcs-test.* uses git, but parent .*src[\\\\/]vcs-test.* uses svn", "get did not fail for right reason") |
| |
| if _, err := os.Stat(tg.path("SrC")); err == nil { |
| // This case only triggers on a case-insensitive file system. |
| tg.runFail("get", "-u", "vcs-test.golang.org/go/test2-svn-git/test2main") |
| tg.grepStderr("src[\\\\/]vcs-test.* uses git, but parent .*src[\\\\/]vcs-test.* uses svn", "get did not fail for right reason") |
| } |
| } |
| |
| func TestPackageMainTestCompilerFlags(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.tempFile("src/p1/p1.go", "package main\n") |
| tg.tempFile("src/p1/p1_test.go", "package main\nimport \"testing\"\nfunc Test(t *testing.T){}\n") |
| tg.run("test", "-c", "-n", "p1") |
| tg.grepBothNot(`([\\/]compile|gccgo).* (-p main|-fgo-pkgpath=main).*p1\.go`, "should not have run compile -p main p1.go") |
| tg.grepStderr(`([\\/]compile|gccgo).* (-p p1|-fgo-pkgpath=p1).*p1\.go`, "should have run compile -p p1 p1.go") |
| } |
| |
| // Issue 4104. |
| func TestGoTestWithPackageListedMultipleTimes(t *testing.T) { |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.run("test", "errors", "errors", "errors", "errors", "errors") |
| if strings.Contains(strings.TrimSpace(tg.getStdout()), "\n") { |
| t.Error("go test errors errors errors errors errors tested the same package multiple times") |
| } |
| } |
| |
| func TestGoListHasAConsistentOrder(t *testing.T) { |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.run("list", "std") |
| first := tg.getStdout() |
| tg.run("list", "std") |
| if first != tg.getStdout() { |
| t.Error("go list std ordering is inconsistent") |
| } |
| } |
| |
| func TestGoListStdDoesNotIncludeCommands(t *testing.T) { |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.run("list", "std") |
| tg.grepStdoutNot("cmd/", "go list std shows commands") |
| } |
| |
| func TestGoListCmdOnlyShowsCommands(t *testing.T) { |
| skipIfGccgo(t, "gccgo does not have GOROOT") |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.run("list", "cmd") |
| out := strings.TrimSpace(tg.getStdout()) |
| for _, line := range strings.Split(out, "\n") { |
| if !strings.Contains(line, "cmd/") { |
| t.Error("go list cmd shows non-commands") |
| break |
| } |
| } |
| } |
| |
| func TestGoListDeps(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempDir("src/p1/p2/p3/p4") |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.tempFile("src/p1/p.go", "package p1\nimport _ \"p1/p2\"\n") |
| tg.tempFile("src/p1/p2/p.go", "package p2\nimport _ \"p1/p2/p3\"\n") |
| tg.tempFile("src/p1/p2/p3/p.go", "package p3\nimport _ \"p1/p2/p3/p4\"\n") |
| tg.tempFile("src/p1/p2/p3/p4/p.go", "package p4\n") |
| tg.run("list", "-f", "{{.Deps}}", "p1") |
| tg.grepStdout("p1/p2/p3/p4", "Deps(p1) does not mention p4") |
| |
| tg.run("list", "-deps", "p1") |
| tg.grepStdout("p1/p2/p3/p4", "-deps p1 does not mention p4") |
| |
| if runtime.Compiler != "gccgo" { |
| // Check the list is in dependency order. |
| tg.run("list", "-deps", "math") |
| want := "internal/cpu\nunsafe\nmath/bits\nmath\n" |
| out := tg.stdout.String() |
| if !strings.Contains(out, "internal/cpu") { |
| // Some systems don't use internal/cpu. |
| want = "unsafe\nmath/bits\nmath\n" |
| } |
| if tg.stdout.String() != want { |
| t.Fatalf("list -deps math: wrong order\nhave %q\nwant %q", tg.stdout.String(), want) |
| } |
| } |
| } |
| |
| func TestGoListTest(t *testing.T) { |
| skipIfGccgo(t, "gccgo does not have standard packages") |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.setenv("GOCACHE", tg.tempdir) |
| |
| tg.run("list", "-test", "-deps", "sort") |
| tg.grepStdout(`^sort.test$`, "missing test main") |
| tg.grepStdout(`^sort$`, "missing real sort") |
| tg.grepStdout(`^sort \[sort.test\]$`, "missing test copy of sort") |
| tg.grepStdout(`^testing \[sort.test\]$`, "missing test copy of testing") |
| tg.grepStdoutNot(`^testing$`, "unexpected real copy of testing") |
| |
| tg.run("list", "-test", "sort") |
| tg.grepStdout(`^sort.test$`, "missing test main") |
| tg.grepStdout(`^sort$`, "missing real sort") |
| tg.grepStdout(`^sort \[sort.test\]$`, "unexpected test copy of sort") |
| tg.grepStdoutNot(`^testing \[sort.test\]$`, "unexpected test copy of testing") |
| tg.grepStdoutNot(`^testing$`, "unexpected real copy of testing") |
| |
| tg.run("list", "-test", "cmd/dist", "cmd/doc") |
| tg.grepStdout(`^cmd/dist$`, "missing cmd/dist") |
| tg.grepStdout(`^cmd/doc$`, "missing cmd/doc") |
| tg.grepStdout(`^cmd/doc\.test$`, "missing cmd/doc test") |
| tg.grepStdoutNot(`^cmd/dist\.test$`, "unexpected cmd/dist test") |
| tg.grepStdoutNot(`^testing`, "unexpected testing") |
| |
| tg.run("list", "-test", "runtime/cgo") |
| tg.grepStdout(`^runtime/cgo$`, "missing runtime/cgo") |
| |
| tg.run("list", "-deps", "-f", "{{if .DepOnly}}{{.ImportPath}}{{end}}", "sort") |
| tg.grepStdout(`^internal/reflectlite$`, "missing internal/reflectlite") |
| tg.grepStdoutNot(`^sort`, "unexpected sort") |
| } |
| |
| func TestGoListCompiledCgo(t *testing.T) { |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.setenv("GOCACHE", tg.tempdir) |
| |
| tg.run("list", "-f", `{{join .CgoFiles "\n"}}`, "net") |
| if tg.stdout.String() == "" { |
| t.Skip("net does not use cgo") |
| } |
| if strings.Contains(tg.stdout.String(), tg.tempdir) { |
| t.Fatalf(".CgoFiles unexpectedly mentioned cache %s", tg.tempdir) |
| } |
| tg.run("list", "-compiled", "-f", `{{.Dir}}{{"\n"}}{{join .CompiledGoFiles "\n"}}`, "net") |
| if !strings.Contains(tg.stdout.String(), tg.tempdir) { |
| t.Fatalf(".CompiledGoFiles with -compiled did not mention cache %s", tg.tempdir) |
| } |
| dir := "" |
| for _, file := range strings.Split(tg.stdout.String(), "\n") { |
| if file == "" { |
| continue |
| } |
| if dir == "" { |
| dir = file |
| continue |
| } |
| if !strings.Contains(file, "/") && !strings.Contains(file, `\`) { |
| file = filepath.Join(dir, file) |
| } |
| if _, err := os.Stat(file); err != nil { |
| t.Fatalf("cannot find .CompiledGoFiles result %s: %v", file, err) |
| } |
| } |
| } |
| |
| func TestGoListExport(t *testing.T) { |
| skipIfGccgo(t, "gccgo does not have standard packages") |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.setenv("GOCACHE", tg.tempdir) |
| |
| tg.run("list", "-f", "{{.Export}}", "strings") |
| if tg.stdout.String() != "" { |
| t.Fatalf(".Export without -export unexpectedly set") |
| } |
| tg.run("list", "-export", "-f", "{{.Export}}", "strings") |
| file := strings.TrimSpace(tg.stdout.String()) |
| if file == "" { |
| t.Fatalf(".Export with -export was empty") |
| } |
| if _, err := os.Stat(file); err != nil { |
| t.Fatalf("cannot find .Export result %s: %v", file, err) |
| } |
| |
| tg.run("list", "-export", "-f", "{{.BuildID}}", "strings") |
| buildID := strings.TrimSpace(tg.stdout.String()) |
| if buildID == "" { |
| t.Fatalf(".BuildID with -export was empty") |
| } |
| |
| tg.run("tool", "buildid", file) |
| toolBuildID := strings.TrimSpace(tg.stdout.String()) |
| if buildID != toolBuildID { |
| t.Fatalf(".BuildID with -export %q disagrees with 'go tool buildid' %q", buildID, toolBuildID) |
| } |
| } |
| |
| // Issue 4096. Validate the output of unsuccessful go install foo/quxx. |
| func TestUnsuccessfulGoInstallShouldMentionMissingPackage(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.runFail("install", "foo/quxx") |
| if tg.grepCountBoth(`cannot find package "foo/quxx" in any of`) != 1 { |
| t.Error(`go install foo/quxx expected error: .*cannot find package "foo/quxx" in any of`) |
| } |
| } |
| |
| func TestGOROOTSearchFailureReporting(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.runFail("install", "foo/quxx") |
| if tg.grepCountBoth(regexp.QuoteMeta(filepath.Join("foo", "quxx"))+` \(from \$GOROOT\)$`) != 1 { |
| t.Error(`go install foo/quxx expected error: .*foo/quxx (from $GOROOT)`) |
| } |
| } |
| |
| func TestMultipleGOPATHEntriesReportedSeparately(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| sep := string(filepath.ListSeparator) |
| tg.setenv("GOPATH", filepath.Join(tg.pwd(), "testdata", "a")+sep+filepath.Join(tg.pwd(), "testdata", "b")) |
| tg.runFail("install", "foo/quxx") |
| if tg.grepCountBoth(`testdata[/\\].[/\\]src[/\\]foo[/\\]quxx`) != 2 { |
| t.Error(`go install foo/quxx expected error: .*testdata/a/src/foo/quxx (from $GOPATH)\n.*testdata/b/src/foo/quxx`) |
| } |
| } |
| |
| // Test (from $GOPATH) annotation is reported for the first GOPATH entry, |
| func TestMentionGOPATHInFirstGOPATHEntry(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| sep := string(filepath.ListSeparator) |
| tg.setenv("GOPATH", filepath.Join(tg.pwd(), "testdata", "a")+sep+filepath.Join(tg.pwd(), "testdata", "b")) |
| tg.runFail("install", "foo/quxx") |
| if tg.grepCountBoth(regexp.QuoteMeta(filepath.Join("testdata", "a", "src", "foo", "quxx"))+` \(from \$GOPATH\)$`) != 1 { |
| t.Error(`go install foo/quxx expected error: .*testdata/a/src/foo/quxx (from $GOPATH)`) |
| } |
| } |
| |
| // but not on the second. |
| func TestMentionGOPATHNotOnSecondEntry(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| sep := string(filepath.ListSeparator) |
| tg.setenv("GOPATH", filepath.Join(tg.pwd(), "testdata", "a")+sep+filepath.Join(tg.pwd(), "testdata", "b")) |
| tg.runFail("install", "foo/quxx") |
| if tg.grepCountBoth(regexp.QuoteMeta(filepath.Join("testdata", "b", "src", "foo", "quxx"))+`$`) != 1 { |
| t.Error(`go install foo/quxx expected error: .*testdata/b/src/foo/quxx`) |
| } |
| } |
| |
| func homeEnvName() string { |
| switch runtime.GOOS { |
| case "windows": |
| return "USERPROFILE" |
| case "plan9": |
| return "home" |
| default: |
| return "HOME" |
| } |
| } |
| |
| func tempEnvName() string { |
| switch runtime.GOOS { |
| case "windows": |
| return "TMP" |
| case "plan9": |
| return "TMPDIR" // actually plan 9 doesn't have one at all but this is fine |
| default: |
| return "TMPDIR" |
| } |
| } |
| |
| func TestDefaultGOPATH(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempDir("home/go") |
| tg.setenv(homeEnvName(), tg.path("home")) |
| |
| tg.run("env", "GOPATH") |
| tg.grepStdout(regexp.QuoteMeta(tg.path("home/go")), "want GOPATH=$HOME/go") |
| |
| tg.setenv("GOROOT", tg.path("home/go")) |
| tg.run("env", "GOPATH") |
| tg.grepStdoutNot(".", "want unset GOPATH because GOROOT=$HOME/go") |
| |
| tg.setenv("GOROOT", tg.path("home/go")+"/") |
| tg.run("env", "GOPATH") |
| tg.grepStdoutNot(".", "want unset GOPATH because GOROOT=$HOME/go/") |
| } |
| |
| func TestDefaultGOPATHGet(t *testing.T) { |
| testenv.MustHaveExternalNetwork(t) |
| testenv.MustHaveExecPath(t, "git") |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.setenv("GOPATH", "") |
| tg.tempDir("home") |
| tg.setenv(homeEnvName(), tg.path("home")) |
| |
| // warn for creating directory |
| tg.run("get", "-v", "github.com/golang/example/hello") |
| tg.grepStderr("created GOPATH="+regexp.QuoteMeta(tg.path("home/go"))+"; see 'go help gopath'", "did not create GOPATH") |
| |
| // no warning if directory already exists |
| tg.must(robustio.RemoveAll(tg.path("home/go"))) |
| tg.tempDir("home/go") |
| tg.run("get", "github.com/golang/example/hello") |
| tg.grepStderrNot(".", "expected no output on standard error") |
| |
| // error if $HOME/go is a file |
| tg.must(robustio.RemoveAll(tg.path("home/go"))) |
| tg.tempFile("home/go", "") |
| tg.runFail("get", "github.com/golang/example/hello") |
| tg.grepStderr(`mkdir .*[/\\]go: .*(not a directory|cannot find the path)`, "expected error because $HOME/go is a file") |
| } |
| |
| func TestDefaultGOPATHPrintedSearchList(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.setenv("GOPATH", "") |
| tg.tempDir("home") |
| tg.setenv(homeEnvName(), tg.path("home")) |
| |
| tg.runFail("install", "github.com/golang/example/hello") |
| tg.grepStderr(regexp.QuoteMeta(tg.path("home/go/src/github.com/golang/example/hello"))+`.*from \$GOPATH`, "expected default GOPATH") |
| } |
| |
| func TestLdflagsArgumentsWithSpacesIssue3941(t *testing.T) { |
| skipIfGccgo(t, "gccgo does not support -ldflags -X") |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempFile("main.go", `package main |
| var extern string |
| func main() { |
| println(extern) |
| }`) |
| tg.run("run", "-ldflags", `-X "main.extern=hello world"`, tg.path("main.go")) |
| tg.grepStderr("^hello world", `ldflags -X "main.extern=hello world"' failed`) |
| } |
| |
| func TestLdFlagsLongArgumentsIssue42295(t *testing.T) { |
| // Test the extremely long command line arguments that contain '\n' characters |
| // get encoded and passed correctly. |
| skipIfGccgo(t, "gccgo does not support -ldflags -X") |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempFile("main.go", `package main |
| var extern string |
| func main() { |
| print(extern) |
| }`) |
| testStr := "test test test test test \n\\ " |
| var buf bytes.Buffer |
| for buf.Len() < work.ArgLengthForResponseFile+1 { |
| buf.WriteString(testStr) |
| } |
| tg.run("run", "-ldflags", fmt.Sprintf(`-X "main.extern=%s"`, buf.String()), tg.path("main.go")) |
| if tg.stderr.String() != buf.String() { |
| t.Errorf("strings differ") |
| } |
| } |
| |
| func TestGoTestDashCDashOControlsBinaryLocation(t *testing.T) { |
| skipIfGccgo(t, "gccgo has no standard packages") |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.run("test", "-c", "-o", tg.path("myerrors.test"+exeSuffix), "errors") |
| tg.wantExecutable(tg.path("myerrors.test"+exeSuffix), "go test -c -o myerrors.test did not create myerrors.test") |
| } |
| |
| func TestGoTestDashOWritesBinary(t *testing.T) { |
| skipIfGccgo(t, "gccgo has no standard packages") |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.run("test", "-o", tg.path("myerrors.test"+exeSuffix), "errors") |
| tg.wantExecutable(tg.path("myerrors.test"+exeSuffix), "go test -o myerrors.test did not create myerrors.test") |
| } |
| |
| func TestGoTestDashIDashOWritesBinary(t *testing.T) { |
| skipIfGccgo(t, "gccgo has no standard packages") |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| |
| // don't let test -i overwrite runtime |
| tg.wantNotStale("runtime", "", "must be non-stale before test -i") |
| |
| tg.run("test", "-v", "-i", "-o", tg.path("myerrors.test"+exeSuffix), "errors") |
| tg.grepBothNot("PASS|FAIL", "test should not have run") |
| tg.wantExecutable(tg.path("myerrors.test"+exeSuffix), "go test -o myerrors.test did not create myerrors.test") |
| } |
| |
| // Issue 4515. |
| func TestInstallWithTags(t *testing.T) { |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempDir("bin") |
| tg.tempFile("src/example/a/main.go", `package main |
| func main() {}`) |
| tg.tempFile("src/example/b/main.go", `// +build mytag |
| |
| package main |
| func main() {}`) |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.run("install", "-tags", "mytag", "example/a", "example/b") |
| tg.wantExecutable(tg.path("bin/a"+exeSuffix), "go install example/a example/b did not install binaries") |
| tg.wantExecutable(tg.path("bin/b"+exeSuffix), "go install example/a example/b did not install binaries") |
| tg.must(os.Remove(tg.path("bin/a" + exeSuffix))) |
| tg.must(os.Remove(tg.path("bin/b" + exeSuffix))) |
| tg.run("install", "-tags", "mytag", "example/...") |
| tg.wantExecutable(tg.path("bin/a"+exeSuffix), "go install example/... did not install binaries") |
| tg.wantExecutable(tg.path("bin/b"+exeSuffix), "go install example/... did not install binaries") |
| tg.run("list", "-tags", "mytag", "example/b...") |
| if strings.TrimSpace(tg.getStdout()) != "example/b" { |
| t.Error("go list example/b did not find example/b") |
| } |
| } |
| |
| // Issue 17451, 17662. |
| func TestSymlinkWarning(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.setenv("GOPATH", tg.path(".")) |
| |
| tg.tempDir("src/example/xx") |
| tg.tempDir("yy/zz") |
| tg.tempFile("yy/zz/zz.go", "package zz\n") |
| if err := os.Symlink(tg.path("yy"), tg.path("src/example/xx/yy")); err != nil { |
| t.Skipf("symlink failed: %v", err) |
| } |
| tg.run("list", "example/xx/z...") |
| tg.grepStdoutNot(".", "list should not have matched anything") |
| tg.grepStderr("matched no packages", "list should have reported that pattern matched no packages") |
| tg.grepStderrNot("symlink", "list should not have reported symlink") |
| |
| tg.run("list", "example/xx/...") |
| tg.grepStdoutNot(".", "list should not have matched anything") |
| tg.grepStderr("matched no packages", "list should have reported that pattern matched no packages") |
| tg.grepStderr("ignoring symlink", "list should have reported symlink") |
| } |
| |
| func TestCgoShowsFullPathNames(t *testing.T) { |
| if !canCgo { |
| t.Skip("skipping because cgo not enabled") |
| } |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempFile("src/x/y/dirname/foo.go", ` |
| package foo |
| import "C" |
| func f() {`) |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.runFail("build", "x/y/dirname") |
| tg.grepBoth("x/y/dirname", "error did not use full path") |
| } |
| |
| func TestCgoHandlesWlORIGIN(t *testing.T) { |
| tooSlow(t) |
| if !canCgo { |
| t.Skip("skipping because cgo not enabled") |
| } |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempFile("src/origin/origin.go", `package origin |
| // #cgo !darwin LDFLAGS: -Wl,-rpath,$ORIGIN |
| // void f(void) {} |
| import "C" |
| func f() { C.f() }`) |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.run("build", "origin") |
| } |
| |
| func TestCgoPkgConfig(t *testing.T) { |
| tooSlow(t) |
| if !canCgo { |
| t.Skip("skipping because cgo not enabled") |
| } |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| |
| tg.run("env", "PKG_CONFIG") |
| pkgConfig := strings.TrimSpace(tg.getStdout()) |
| testenv.MustHaveExecPath(t, pkgConfig) |
| if out, err := exec.Command(pkgConfig, "--atleast-pkgconfig-version", "0.24").CombinedOutput(); err != nil { |
| t.Skipf("%s --atleast-pkgconfig-version 0.24: %v\n%s", pkgConfig, err, out) |
| } |
| |
| // OpenBSD's pkg-config is strict about whitespace and only |
| // supports backslash-escaped whitespace. It does not support |
| // quotes, which the normal freedesktop.org pkg-config does |
| // support. See https://man.openbsd.org/pkg-config.1 |
| tg.tempFile("foo.pc", ` |
| Name: foo |
| Description: The foo library |
| Version: 1.0.0 |
| Cflags: -Dhello=10 -Dworld=+32 -DDEFINED_FROM_PKG_CONFIG=hello\ world |
| `) |
| tg.tempFile("foo.go", `package main |
| |
| /* |
| #cgo pkg-config: foo |
| int value() { |
| return DEFINED_FROM_PKG_CONFIG; |
| } |
| */ |
| import "C" |
| import "os" |
| |
| func main() { |
| if C.value() != 42 { |
| println("value() =", C.value(), "wanted 42") |
| os.Exit(1) |
| } |
| } |
| `) |
| tg.setenv("PKG_CONFIG_PATH", tg.path(".")) |
| tg.run("run", tg.path("foo.go")) |
| } |
| |
| func TestListTemplateContextFunction(t *testing.T) { |
| t.Parallel() |
| for _, tt := range []struct { |
| v string |
| want string |
| }{ |
| {"GOARCH", runtime.GOARCH}, |
| {"GOOS", runtime.GOOS}, |
| {"GOROOT", filepath.Clean(runtime.GOROOT())}, |
| {"GOPATH", os.Getenv("GOPATH")}, |
| {"CgoEnabled", ""}, |
| {"UseAllFiles", ""}, |
| {"Compiler", ""}, |
| {"BuildTags", ""}, |
| {"ReleaseTags", ""}, |
| {"InstallSuffix", ""}, |
| } { |
| tt := tt |
| t.Run(tt.v, func(t *testing.T) { |
| tg := testgo(t) |
| tg.parallel() |
| defer tg.cleanup() |
| tmpl := "{{context." + tt.v + "}}" |
| tg.run("list", "-f", tmpl) |
| if tt.want == "" { |
| return |
| } |
| if got := strings.TrimSpace(tg.getStdout()); got != tt.want { |
| t.Errorf("go list -f %q: got %q; want %q", tmpl, got, tt.want) |
| } |
| }) |
| } |
| } |
| |
| // Test that you cannot use a local import in a package |
| // accessed by a non-local import (found in a GOPATH/GOROOT). |
| // See golang.org/issue/17475. |
| func TestImportLocal(t *testing.T) { |
| tooSlow(t) |
| |
| tg := testgo(t) |
| tg.parallel() |
| defer tg.cleanup() |
| |
| tg.tempFile("src/dir/x/x.go", `package x |
| var X int |
| `) |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.run("build", "dir/x") |
| |
| // Ordinary import should work. |
| tg.tempFile("src/dir/p0/p.go", `package p0 |
| import "dir/x" |
| var _ = x.X |
| `) |
| tg.run("build", "dir/p0") |
| |
| // Relative import should not. |
| tg.tempFile("src/dir/p1/p.go", `package p1 |
| import "../x" |
| var _ = x.X |
| `) |
| tg.runFail("build", "dir/p1") |
| tg.grepStderr("local import.*in non-local package", "did not diagnose local import") |
| |
| // ... even in a test. |
| tg.tempFile("src/dir/p2/p.go", `package p2 |
| `) |
| tg.tempFile("src/dir/p2/p_test.go", `package p2 |
| import "../x" |
| import "testing" |
| var _ = x.X |
| func TestFoo(t *testing.T) {} |
| `) |
| tg.run("build", "dir/p2") |
| tg.runFail("test", "dir/p2") |
| tg.grepStderr("local import.*in non-local package", "did not diagnose local import") |
| |
| // ... even in an xtest. |
| tg.tempFile("src/dir/p2/p_test.go", `package p2_test |
| import "../x" |
| import "testing" |
| var _ = x.X |
| func TestFoo(t *testing.T) {} |
| `) |
| tg.run("build", "dir/p2") |
| tg.runFail("test", "dir/p2") |
| tg.grepStderr("local import.*in non-local package", "did not diagnose local import") |
| |
| // Relative import starting with ./ should not work either. |
| tg.tempFile("src/dir/d.go", `package dir |
| import "./x" |
| var _ = x.X |
| `) |
| tg.runFail("build", "dir") |
| tg.grepStderr("local import.*in non-local package", "did not diagnose local import") |
| |
| // ... even in a test. |
| tg.tempFile("src/dir/d.go", `package dir |
| `) |
| tg.tempFile("src/dir/d_test.go", `package dir |
| import "./x" |
| import "testing" |
| var _ = x.X |
| func TestFoo(t *testing.T) {} |
| `) |
| tg.run("build", "dir") |
| tg.runFail("test", "dir") |
| tg.grepStderr("local import.*in non-local package", "did not diagnose local import") |
| |
| // ... even in an xtest. |
| tg.tempFile("src/dir/d_test.go", `package dir_test |
| import "./x" |
| import "testing" |
| var _ = x.X |
| func TestFoo(t *testing.T) {} |
| `) |
| tg.run("build", "dir") |
| tg.runFail("test", "dir") |
| tg.grepStderr("local import.*in non-local package", "did not diagnose local import") |
| |
| // Relative import plain ".." should not work. |
| tg.tempFile("src/dir/x/y/y.go", `package dir |
| import ".." |
| var _ = x.X |
| `) |
| tg.runFail("build", "dir/x/y") |
| tg.grepStderr("local import.*in non-local package", "did not diagnose local import") |
| |
| // ... even in a test. |
| tg.tempFile("src/dir/x/y/y.go", `package y |
| `) |
| tg.tempFile("src/dir/x/y/y_test.go", `package y |
| import ".." |
| import "testing" |
| var _ = x.X |
| func TestFoo(t *testing.T) {} |
| `) |
| tg.run("build", "dir/x/y") |
| tg.runFail("test", "dir/x/y") |
| tg.grepStderr("local import.*in non-local package", "did not diagnose local import") |
| |
| // ... even in an x test. |
| tg.tempFile("src/dir/x/y/y_test.go", `package y_test |
| import ".." |
| import "testing" |
| var _ = x.X |
| func TestFoo(t *testing.T) {} |
| `) |
| tg.run("build", "dir/x/y") |
| tg.runFail("test", "dir/x/y") |
| tg.grepStderr("local import.*in non-local package", "did not diagnose local import") |
| |
| // Relative import "." should not work. |
| tg.tempFile("src/dir/x/xx.go", `package x |
| import "." |
| var _ = x.X |
| `) |
| tg.runFail("build", "dir/x") |
| tg.grepStderr("cannot import current directory", "did not diagnose import current directory") |
| |
| // ... even in a test. |
| tg.tempFile("src/dir/x/xx.go", `package x |
| `) |
| tg.tempFile("src/dir/x/xx_test.go", `package x |
| import "." |
| import "testing" |
| var _ = x.X |
| func TestFoo(t *testing.T) {} |
| `) |
| tg.run("build", "dir/x") |
| tg.runFail("test", "dir/x") |
| tg.grepStderr("cannot import current directory", "did not diagnose import current directory") |
| |
| // ... even in an xtest. |
| tg.tempFile("src/dir/x/xx.go", `package x |
| `) |
| tg.tempFile("src/dir/x/xx_test.go", `package x_test |
| import "." |
| import "testing" |
| var _ = x.X |
| func TestFoo(t *testing.T) {} |
| `) |
| tg.run("build", "dir/x") |
| tg.runFail("test", "dir/x") |
| tg.grepStderr("cannot import current directory", "did not diagnose import current directory") |
| } |
| |
| func TestGoInstallPkgdir(t *testing.T) { |
| skipIfGccgo(t, "gccgo has no standard packages") |
| tooSlow(t) |
| |
| tg := testgo(t) |
| tg.parallel() |
| defer tg.cleanup() |
| tg.makeTempdir() |
| pkg := tg.path(".") |
| tg.run("install", "-pkgdir", pkg, "sync") |
| tg.mustExist(filepath.Join(pkg, "sync.a")) |
| tg.mustNotExist(filepath.Join(pkg, "sync/atomic.a")) |
| tg.run("install", "-i", "-pkgdir", pkg, "sync") |
| tg.mustExist(filepath.Join(pkg, "sync.a")) |
| tg.mustExist(filepath.Join(pkg, "sync/atomic.a")) |
| } |
| |
| // For issue 14337. |
| func TestParallelTest(t *testing.T) { |
| tooSlow(t) |
| tg := testgo(t) |
| tg.parallel() |
| defer tg.cleanup() |
| tg.makeTempdir() |
| const testSrc = `package package_test |
| import ( |
| "testing" |
| ) |
| func TestTest(t *testing.T) { |
| }` |
| tg.tempFile("src/p1/p1_test.go", strings.Replace(testSrc, "package_test", "p1_test", 1)) |
| tg.tempFile("src/p2/p2_test.go", strings.Replace(testSrc, "package_test", "p2_test", 1)) |
| tg.tempFile("src/p3/p3_test.go", strings.Replace(testSrc, "package_test", "p3_test", 1)) |
| tg.tempFile("src/p4/p4_test.go", strings.Replace(testSrc, "package_test", "p4_test", 1)) |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.run("test", "-p=4", "p1", "p2", "p3", "p4") |
| } |
| |
| func TestBinaryOnlyPackages(t *testing.T) { |
| tooSlow(t) |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.setenv("GOPATH", tg.path(".")) |
| |
| tg.tempFile("src/p1/p1.go", `//go:binary-only-package |
| |
| package p1 |
| `) |
| tg.wantStale("p1", "binary-only packages are no longer supported", "p1 is binary-only, and this message should always be printed") |
| tg.runFail("install", "p1") |
| tg.grepStderr("binary-only packages are no longer supported", "did not report attempt to compile binary-only package") |
| |
| tg.tempFile("src/p1/p1.go", ` |
| package p1 |
| import "fmt" |
| func F(b bool) { fmt.Printf("hello from p1\n"); if b { F(false) } } |
| `) |
| tg.run("install", "p1") |
| os.Remove(tg.path("src/p1/p1.go")) |
| tg.mustNotExist(tg.path("src/p1/p1.go")) |
| |
| tg.tempFile("src/p2/p2.go", `//go:binary-only-packages-are-not-great |
| |
| package p2 |
| import "p1" |
| func F() { p1.F(true) } |
| `) |
| tg.runFail("install", "p2") |
| tg.grepStderr("no Go files", "did not complain about missing sources") |
| |
| tg.tempFile("src/p1/missing.go", `//go:binary-only-package |
| |
| package p1 |
| import _ "fmt" |
| func G() |
| `) |
| tg.wantStale("p1", "binary-only package", "should NOT want to rebuild p1 (first)") |
| tg.runFail("install", "p2") |
| tg.grepStderr("p1: binary-only packages are no longer supported", "did not report error for binary-only p1") |
| |
| tg.run("list", "-deps", "-f", "{{.ImportPath}}: {{.BinaryOnly}}", "p2") |
| tg.grepStdout("p1: true", "p1 not listed as BinaryOnly") |
| tg.grepStdout("p2: false", "p2 listed as BinaryOnly") |
| } |
| |
| // Issue 16050. |
| func TestAlwaysLinkSysoFiles(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempDir("src/syso") |
| tg.tempFile("src/syso/a.syso", ``) |
| tg.tempFile("src/syso/b.go", `package syso`) |
| tg.setenv("GOPATH", tg.path(".")) |
| |
| // We should see the .syso file regardless of the setting of |
| // CGO_ENABLED. |
| |
| tg.setenv("CGO_ENABLED", "1") |
| tg.run("list", "-f", "{{.SysoFiles}}", "syso") |
| tg.grepStdout("a.syso", "missing syso file with CGO_ENABLED=1") |
| |
| tg.setenv("CGO_ENABLED", "0") |
| tg.run("list", "-f", "{{.SysoFiles}}", "syso") |
| tg.grepStdout("a.syso", "missing syso file with CGO_ENABLED=0") |
| } |
| |
| // Issue 16120. |
| func TestGenerateUsesBuildContext(t *testing.T) { |
| if runtime.GOOS == "windows" { |
| t.Skip("this test won't run under Windows") |
| } |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempDir("src/gen") |
| tg.tempFile("src/gen/gen.go", "package gen\n//go:generate echo $GOOS $GOARCH\n") |
| tg.setenv("GOPATH", tg.path(".")) |
| |
| tg.setenv("GOOS", "linux") |
| tg.setenv("GOARCH", "amd64") |
| tg.run("generate", "gen") |
| tg.grepStdout("linux amd64", "unexpected GOOS/GOARCH combination") |
| |
| tg.setenv("GOOS", "darwin") |
| tg.setenv("GOARCH", "arm64") |
| tg.run("generate", "gen") |
| tg.grepStdout("darwin arm64", "unexpected GOOS/GOARCH combination") |
| } |
| |
| func TestGoEnv(t *testing.T) { |
| tg := testgo(t) |
| tg.parallel() |
| defer tg.cleanup() |
| tg.setenv("GOOS", "freebsd") // to avoid invalid pair errors |
| tg.setenv("GOARCH", "arm") |
| tg.run("env", "GOARCH") |
| tg.grepStdout("^arm$", "GOARCH not honored") |
| |
| tg.run("env", "GCCGO") |
| tg.grepStdout(".", "GCCGO unexpectedly empty") |
| |
| tg.run("env", "CGO_CFLAGS") |
| tg.grepStdout(".", "default CGO_CFLAGS unexpectedly empty") |
| |
| tg.setenv("CGO_CFLAGS", "-foobar") |
| tg.run("env", "CGO_CFLAGS") |
| tg.grepStdout("^-foobar$", "CGO_CFLAGS not honored") |
| |
| tg.setenv("CC", "gcc -fmust -fgo -ffaster") |
| tg.run("env", "CC") |
| tg.grepStdout("gcc", "CC not found") |
| tg.run("env", "GOGCCFLAGS") |
| tg.grepStdout("-ffaster", "CC arguments not found") |
| |
| tg.run("env", "GOVERSION") |
| envVersion := strings.TrimSpace(tg.stdout.String()) |
| |
| tg.run("version") |
| cmdVersion := strings.TrimSpace(tg.stdout.String()) |
| |
| // If 'go version' is "go version <version> <goos>/<goarch>", then |
| // 'go env GOVERSION' is just "<version>". |
| if cmdVersion == envVersion || !strings.Contains(cmdVersion, envVersion) { |
| t.Fatalf("'go env GOVERSION' %q should be a shorter substring of 'go version' %q", envVersion, cmdVersion) |
| } |
| } |
| |
| const ( |
| noMatchesPattern = `(?m)^ok.*\[no tests to run\]` |
| okPattern = `(?m)^ok` |
| ) |
| |
| // Issue 18044. |
| func TestLdBindNow(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.setenv("LD_BIND_NOW", "1") |
| tg.run("help") |
| } |
| |
| // Issue 18225. |
| // This is really a cmd/asm issue but this is a convenient place to test it. |
| func TestConcurrentAsm(t *testing.T) { |
| skipIfGccgo(t, "gccgo does not use cmd/asm") |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| asm := `DATA ·constants<>+0x0(SB)/8,$0 |
| GLOBL ·constants<>(SB),8,$8 |
| ` |
| tg.tempFile("go/src/p/a.s", asm) |
| tg.tempFile("go/src/p/b.s", asm) |
| tg.tempFile("go/src/p/p.go", `package p`) |
| tg.setenv("GOPATH", tg.path("go")) |
| tg.run("build", "p") |
| } |
| |
| // Issue 18975. |
| func TestFFLAGS(t *testing.T) { |
| if !canCgo { |
| t.Skip("skipping because cgo not enabled") |
| } |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| |
| tg.tempFile("p/src/p/main.go", `package main |
| // #cgo FFLAGS: -no-such-fortran-flag |
| import "C" |
| func main() {} |
| `) |
| tg.tempFile("p/src/p/a.f", `! comment`) |
| tg.setenv("GOPATH", tg.path("p")) |
| |
| // This should normally fail because we are passing an unknown flag, |
| // but issue #19080 points to Fortran compilers that succeed anyhow. |
| // To work either way we call doRun directly rather than run or runFail. |
| tg.doRun([]string{"build", "-x", "p"}) |
| |
| tg.grepStderr("no-such-fortran-flag", `missing expected "-no-such-fortran-flag"`) |
| } |
| |
| // Issue 19198. |
| // This is really a cmd/link issue but this is a convenient place to test it. |
| func TestDuplicateGlobalAsmSymbols(t *testing.T) { |
| skipIfGccgo(t, "gccgo does not use cmd/asm") |
| tooSlow(t) |
| if runtime.GOARCH != "386" && runtime.GOARCH != "amd64" { |
| t.Skipf("skipping test on %s", runtime.GOARCH) |
| } |
| if !canCgo { |
| t.Skip("skipping because cgo not enabled") |
| } |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| |
| asm := ` |
| #include "textflag.h" |
| |
| DATA sym<>+0x0(SB)/8,$0 |
| GLOBL sym<>(SB),(NOPTR+RODATA),$8 |
| |
| TEXT ·Data(SB),NOSPLIT,$0 |
| MOVB sym<>(SB), AX |
| MOVB AX, ret+0(FP) |
| RET |
| ` |
| tg.tempFile("go/src/a/a.s", asm) |
| tg.tempFile("go/src/a/a.go", `package a; func Data() uint8`) |
| tg.tempFile("go/src/b/b.s", asm) |
| tg.tempFile("go/src/b/b.go", `package b; func Data() uint8`) |
| tg.tempFile("go/src/p/p.go", ` |
| package main |
| import "a" |
| import "b" |
| import "C" |
| func main() { |
| _ = a.Data() + b.Data() |
| } |
| `) |
| tg.setenv("GOPATH", tg.path("go")) |
| exe := tg.path("p.exe") |
| tg.creatingTemp(exe) |
| tg.run("build", "-o", exe, "p") |
| } |
| |
| func copyFile(src, dst string, perm fs.FileMode) error { |
| sf, err := os.Open(src) |
| if err != nil { |
| return err |
| } |
| defer sf.Close() |
| |
| df, err := os.OpenFile(dst, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, perm) |
| if err != nil { |
| return err |
| } |
| |
| _, err = io.Copy(df, sf) |
| err2 := df.Close() |
| if err != nil { |
| return err |
| } |
| return err2 |
| } |
| |
| func TestNeedVersion(t *testing.T) { |
| skipIfGccgo(t, "gccgo does not use cmd/compile") |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempFile("goversion.go", `package main; func main() {}`) |
| path := tg.path("goversion.go") |
| tg.setenv("TESTGO_VERSION", "go1.testgo") |
| tg.runFail("run", path) |
| tg.grepStderr("compile", "does not match go tool version") |
| } |
| |
| func TestBuildmodePIE(t *testing.T) { |
| if testing.Short() && testenv.Builder() == "" { |
| t.Skipf("skipping in -short mode on non-builder") |
| } |
| |
| platform := fmt.Sprintf("%s/%s", runtime.GOOS, runtime.GOARCH) |
| switch platform { |
| case "linux/386", "linux/amd64", "linux/arm", "linux/arm64", "linux/ppc64le", "linux/riscv64", "linux/s390x", |
| "android/amd64", "android/arm", "android/arm64", "android/386", |
| "freebsd/amd64", |
| "windows/386", "windows/amd64", "windows/arm": |
| case "darwin/amd64": |
| default: |
| t.Skipf("skipping test because buildmode=pie is not supported on %s", platform) |
| } |
| t.Run("non-cgo", func(t *testing.T) { |
| testBuildmodePIE(t, false, true) |
| }) |
| if canCgo { |
| switch runtime.GOOS { |
| case "darwin", "freebsd", "linux", "windows": |
| t.Run("cgo", func(t *testing.T) { |
| testBuildmodePIE(t, true, true) |
| }) |
| } |
| } |
| } |
| |
| func TestWindowsDefaultBuildmodIsPIE(t *testing.T) { |
| if testing.Short() && testenv.Builder() == "" { |
| t.Skipf("skipping in -short mode on non-builder") |
| } |
| |
| if runtime.GOOS != "windows" { |
| t.Skip("skipping windows only test") |
| } |
| |
| t.Run("non-cgo", func(t *testing.T) { |
| testBuildmodePIE(t, false, false) |
| }) |
| if canCgo { |
| t.Run("cgo", func(t *testing.T) { |
| testBuildmodePIE(t, true, false) |
| }) |
| } |
| } |
| |
| func testBuildmodePIE(t *testing.T, useCgo, setBuildmodeToPIE bool) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| |
| var s string |
| if useCgo { |
| s = `import "C";` |
| } |
| tg.tempFile("main.go", fmt.Sprintf(`package main;%s func main() { print("hello") }`, s)) |
| src := tg.path("main.go") |
| obj := tg.path("main.exe") |
| args := []string{"build"} |
| if setBuildmodeToPIE { |
| args = append(args, "-buildmode=pie") |
| } |
| args = append(args, "-o", obj, src) |
| tg.run(args...) |
| |
| switch runtime.GOOS { |
| case "linux", "android", "freebsd": |
| f, err := elf.Open(obj) |
| if err != nil { |
| t.Fatal(err) |
| } |
| defer f.Close() |
| if f.Type != elf.ET_DYN { |
| t.Errorf("PIE type must be ET_DYN, but %s", f.Type) |
| } |
| case "darwin": |
| f, err := macho.Open(obj) |
| if err != nil { |
| t.Fatal(err) |
| } |
| defer f.Close() |
| if f.Flags&macho.FlagDyldLink == 0 { |
| t.Error("PIE must have DyldLink flag, but not") |
| } |
| if f.Flags&macho.FlagPIE == 0 { |
| t.Error("PIE must have PIE flag, but not") |
| } |
| case "windows": |
| f, err := pe.Open(obj) |
| if err != nil { |
| t.Fatal(err) |
| } |
| defer f.Close() |
| if f.Section(".reloc") == nil { |
| t.Error(".reloc section is not present") |
| } |
| if (f.FileHeader.Characteristics & pe.IMAGE_FILE_RELOCS_STRIPPED) != 0 { |
| t.Error("IMAGE_FILE_RELOCS_STRIPPED flag is set") |
| } |
| var dc uint16 |
| switch oh := f.OptionalHeader.(type) { |
| case *pe.OptionalHeader32: |
| dc = oh.DllCharacteristics |
| case *pe.OptionalHeader64: |
| dc = oh.DllCharacteristics |
| if (dc & pe.IMAGE_DLLCHARACTERISTICS_HIGH_ENTROPY_VA) == 0 { |
| t.Error("IMAGE_DLLCHARACTERISTICS_HIGH_ENTROPY_VA flag is not set") |
| } |
| default: |
| t.Fatalf("unexpected optional header type of %T", f.OptionalHeader) |
| } |
| if (dc & pe.IMAGE_DLLCHARACTERISTICS_DYNAMIC_BASE) == 0 { |
| t.Error("IMAGE_DLLCHARACTERISTICS_DYNAMIC_BASE flag is not set") |
| } |
| if useCgo { |
| // Test that only one symbol is exported (#40795). |
| // PIE binaries don´t require .edata section but unfortunately |
| // binutils doesn´t generate a .reloc section unless there is |
| // at least one symbol exported. |
| // See https://sourceware.org/bugzilla/show_bug.cgi?id=19011 |
| section := f.Section(".edata") |
| if section == nil { |
| t.Fatalf(".edata section is not present") |
| } |
| // TODO: deduplicate this struct from cmd/link/internal/ld/pe.go |
| type IMAGE_EXPORT_DIRECTORY struct { |
| _ [2]uint32 |
| _ [2]uint16 |
| _ [2]uint32 |
| NumberOfFunctions uint32 |
| NumberOfNames uint32 |
| _ [3]uint32 |
| } |
| var e IMAGE_EXPORT_DIRECTORY |
| if err := binary.Read(section.Open(), binary.LittleEndian, &e); err != nil { |
| t.Fatalf("binary.Read failed: %v", err) |
| } |
| |
| // Only _cgo_dummy_export should be exported |
| if e.NumberOfFunctions != 1 { |
| t.Fatalf("got %d exported functions; want 1", e.NumberOfFunctions) |
| } |
| if e.NumberOfNames != 1 { |
| t.Fatalf("got %d exported names; want 1", e.NumberOfNames) |
| } |
| } |
| default: |
| panic("unreachable") |
| } |
| |
| out, err := exec.Command(obj).CombinedOutput() |
| if err != nil { |
| t.Fatal(err) |
| } |
| |
| if string(out) != "hello" { |
| t.Errorf("got %q; want %q", out, "hello") |
| } |
| } |
| |
| func TestUpxCompression(t *testing.T) { |
| if runtime.GOOS != "linux" || |
| (runtime.GOARCH != "amd64" && runtime.GOARCH != "386") { |
| t.Skipf("skipping upx test on %s/%s", runtime.GOOS, runtime.GOARCH) |
| } |
| |
| testenv.MustHaveExecPath(t, "upx") |
| out, err := exec.Command("upx", "--version").CombinedOutput() |
| if err != nil { |
| t.Fatalf("upx --version failed: %v", err) |
| } |
| |
| // upx --version prints `upx <version>` in the first line of output: |
| // upx 3.94 |
| // [...] |
| re := regexp.MustCompile(`([[:digit:]]+)\.([[:digit:]]+)`) |
| upxVersion := re.FindStringSubmatch(string(out)) |
| if len(upxVersion) != 3 { |
| t.Fatalf("bad upx version string: %s", upxVersion) |
| } |
| |
| major, err1 := strconv.Atoi(upxVersion[1]) |
| minor, err2 := strconv.Atoi(upxVersion[2]) |
| if err1 != nil || err2 != nil { |
| t.Fatalf("bad upx version string: %s", upxVersion[0]) |
| } |
| |
| // Anything below 3.94 is known not to work with go binaries |
| if (major < 3) || (major == 3 && minor < 94) { |
| t.Skipf("skipping because upx version %v.%v is too old", major, minor) |
| } |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| |
| tg.tempFile("main.go", `package main; import "fmt"; func main() { fmt.Print("hello upx") }`) |
| src := tg.path("main.go") |
| obj := tg.path("main") |
| tg.run("build", "-o", obj, src) |
| |
| out, err = exec.Command("upx", obj).CombinedOutput() |
| if err != nil { |
| t.Logf("executing upx\n%s\n", out) |
| t.Fatalf("upx failed with %v", err) |
| } |
| |
| out, err = exec.Command(obj).CombinedOutput() |
| if err != nil { |
| t.Logf("%s", out) |
| t.Fatalf("running compressed go binary failed with error %s", err) |
| } |
| if string(out) != "hello upx" { |
| t.Fatalf("bad output from compressed go binary:\ngot %q; want %q", out, "hello upx") |
| } |
| } |
| |
| func TestCacheListStale(t *testing.T) { |
| tooSlow(t) |
| if strings.Contains(os.Getenv("GODEBUG"), "gocacheverify") { |
| t.Skip("GODEBUG gocacheverify") |
| } |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.setenv("GOCACHE", tg.path("cache")) |
| tg.tempFile("gopath/src/p/p.go", "package p; import _ \"q\"; func F(){}\n") |
| tg.tempFile("gopath/src/q/q.go", "package q; func F(){}\n") |
| tg.tempFile("gopath/src/m/m.go", "package main; import _ \"q\"; func main(){}\n") |
| |
| tg.setenv("GOPATH", tg.path("gopath")) |
| tg.run("install", "p", "m") |
| tg.run("list", "-f={{.ImportPath}} {{.Stale}}", "m", "q", "p") |
| tg.grepStdout("^m false", "m should not be stale") |
| tg.grepStdout("^q true", "q should be stale") |
| tg.grepStdout("^p false", "p should not be stale") |
| } |
| |
| func TestCacheCoverage(t *testing.T) { |
| tooSlow(t) |
| |
| if strings.Contains(os.Getenv("GODEBUG"), "gocacheverify") { |
| t.Skip("GODEBUG gocacheverify") |
| } |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.setenv("GOPATH", filepath.Join(tg.pwd(), "testdata")) |
| tg.makeTempdir() |
| |
| tg.setenv("GOCACHE", tg.path("c1")) |
| tg.run("test", "-cover", "-short", "strings") |
| tg.run("test", "-cover", "-short", "math", "strings") |
| } |
| |
| func TestIssue22588(t *testing.T) { |
| // Don't get confused by stderr coming from tools. |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| |
| if _, err := os.Stat("/usr/bin/time"); err != nil { |
| t.Skip(err) |
| } |
| |
| tg.run("list", "-f={{.Stale}}", "runtime") |
| tg.run("list", "-toolexec=/usr/bin/time", "-f={{.Stale}}", "runtime") |
| tg.grepStdout("false", "incorrectly reported runtime as stale") |
| } |
| |
| func TestIssue22531(t *testing.T) { |
| tooSlow(t) |
| if strings.Contains(os.Getenv("GODEBUG"), "gocacheverify") { |
| t.Skip("GODEBUG gocacheverify") |
| } |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.setenv("GOPATH", tg.tempdir) |
| tg.setenv("GOCACHE", tg.path("cache")) |
| tg.tempFile("src/m/main.go", "package main /* c1 */; func main() {}\n") |
| tg.run("install", "-x", "m") |
| tg.run("list", "-f", "{{.Stale}}", "m") |
| tg.grepStdout("false", "reported m as stale after install") |
| tg.run("tool", "buildid", tg.path("bin/m"+exeSuffix)) |
| |
| // The link action ID did not include the full main build ID, |
| // even though the full main build ID is written into the |
| // eventual binary. That caused the following install to |
| // be a no-op, thinking the gofmt binary was up-to-date, |
| // even though .Stale could see it was not. |
| tg.tempFile("src/m/main.go", "package main /* c2 */; func main() {}\n") |
| tg.run("install", "-x", "m") |
| tg.run("list", "-f", "{{.Stale}}", "m") |
| tg.grepStdout("false", "reported m as stale after reinstall") |
| tg.run("tool", "buildid", tg.path("bin/m"+exeSuffix)) |
| } |
| |
| func TestIssue22596(t *testing.T) { |
| tooSlow(t) |
| if strings.Contains(os.Getenv("GODEBUG"), "gocacheverify") { |
| t.Skip("GODEBUG gocacheverify") |
| } |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.setenv("GOCACHE", tg.path("cache")) |
| tg.tempFile("gopath1/src/p/p.go", "package p; func F(){}\n") |
| tg.tempFile("gopath2/src/p/p.go", "package p; func F(){}\n") |
| |
| tg.setenv("GOPATH", tg.path("gopath1")) |
| tg.run("list", "-f={{.Target}}", "p") |
| target1 := strings.TrimSpace(tg.getStdout()) |
| tg.run("install", "p") |
| tg.wantNotStale("p", "", "p stale after install") |
| |
| tg.setenv("GOPATH", tg.path("gopath2")) |
| tg.run("list", "-f={{.Target}}", "p") |
| target2 := strings.TrimSpace(tg.getStdout()) |
| tg.must(os.MkdirAll(filepath.Dir(target2), 0777)) |
| tg.must(copyFile(target1, target2, 0666)) |
| tg.wantStale("p", "build ID mismatch", "p not stale after copy from gopath1") |
| tg.run("install", "p") |
| tg.wantNotStale("p", "", "p stale after install2") |
| } |
| |
| func TestTestCache(t *testing.T) { |
| tooSlow(t) |
| |
| if strings.Contains(os.Getenv("GODEBUG"), "gocacheverify") { |
| t.Skip("GODEBUG gocacheverify") |
| } |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.setenv("GOPATH", tg.tempdir) |
| tg.setenv("GOCACHE", tg.path("cache")) |
| |
| // The -p=1 in the commands below just makes the -x output easier to read. |
| |
| t.Log("\n\nINITIAL\n\n") |
| |
| tg.tempFile("src/p1/p1.go", "package p1\nvar X = 1\n") |
| tg.tempFile("src/p2/p2.go", "package p2\nimport _ \"p1\"\nvar X = 1\n") |
| tg.tempFile("src/t/t1/t1_test.go", "package t\nimport \"testing\"\nfunc Test1(*testing.T) {}\n") |
| tg.tempFile("src/t/t2/t2_test.go", "package t\nimport _ \"p1\"\nimport \"testing\"\nfunc Test2(*testing.T) {}\n") |
| tg.tempFile("src/t/t3/t3_test.go", "package t\nimport \"p1\"\nimport \"testing\"\nfunc Test3(t *testing.T) {t.Log(p1.X)}\n") |
| tg.tempFile("src/t/t4/t4_test.go", "package t\nimport \"p2\"\nimport \"testing\"\nfunc Test4(t *testing.T) {t.Log(p2.X)}") |
| tg.run("test", "-x", "-v", "-short", "t/...") |
| |
| t.Log("\n\nREPEAT\n\n") |
| |
| tg.run("test", "-x", "-v", "-short", "t/...") |
| tg.grepStdout(`ok \tt/t1\t\(cached\)`, "did not cache t1") |
| tg.grepStdout(`ok \tt/t2\t\(cached\)`, "did not cache t2") |
| tg.grepStdout(`ok \tt/t3\t\(cached\)`, "did not cache t3") |
| tg.grepStdout(`ok \tt/t4\t\(cached\)`, "did not cache t4") |
| tg.grepStderrNot(`[\\/](compile|gccgo) `, "incorrectly ran compiler") |
| tg.grepStderrNot(`[\\/](link|gccgo) `, "incorrectly ran linker") |
| tg.grepStderrNot(`p[0-9]\.test`, "incorrectly ran test") |
| |
| t.Log("\n\nCOMMENT\n\n") |
| |
| // Changing the program text without affecting the compiled package |
| // should result in the package being rebuilt but nothing more. |
| tg.tempFile("src/p1/p1.go", "package p1\nvar X = 01\n") |
| tg.run("test", "-p=1", "-x", "-v", "-short", "t/...") |
| tg.grepStdout(`ok \tt/t1\t\(cached\)`, "did not cache t1") |
| tg.grepStdout(`ok \tt/t2\t\(cached\)`, "did not cache t2") |
| tg.grepStdout(`ok \tt/t3\t\(cached\)`, "did not cache t3") |
| tg.grepStdout(`ok \tt/t4\t\(cached\)`, "did not cache t4") |
| tg.grepStderrNot(`([\\/](compile|gccgo) ).*t[0-9]_test\.go`, "incorrectly ran compiler") |
| tg.grepStderrNot(`[\\/](link|gccgo) `, "incorrectly ran linker") |
| tg.grepStderrNot(`t[0-9]\.test.*test\.short`, "incorrectly ran test") |
| |
| t.Log("\n\nCHANGE\n\n") |
| |
| // Changing the actual package should have limited effects. |
| tg.tempFile("src/p1/p1.go", "package p1\nvar X = 02\n") |
| tg.run("test", "-p=1", "-x", "-v", "-short", "t/...") |
| |
| // p2 should have been rebuilt. |
| tg.grepStderr(`([\\/]compile|gccgo).*p2.go`, "did not recompile p2") |
| |
| // t1 does not import anything, should not have been rebuilt. |
| tg.grepStderrNot(`([\\/]compile|gccgo).*t1_test.go`, "incorrectly recompiled t1") |
| tg.grepStderrNot(`([\\/]link|gccgo).*t1_test`, "incorrectly relinked t1_test") |
| tg.grepStdout(`ok \tt/t1\t\(cached\)`, "did not cache t/t1") |
| |
| // t2 imports p1 and must be rebuilt and relinked, |
| // but the change should not have any effect on the test binary, |
| // so the test should not have been rerun. |
| tg.grepStderr(`([\\/]compile|gccgo).*t2_test.go`, "did not recompile t2") |
| tg.grepStderr(`([\\/]link|gccgo).*t2\.test`, "did not relink t2_test") |
| // This check does not currently work with gccgo, as garbage |
| // collection of unused variables is not turned on by default. |
| if runtime.Compiler != "gccgo" { |
| tg.grepStdout(`ok \tt/t2\t\(cached\)`, "did not cache t/t2") |
| } |
| |
| // t3 imports p1, and changing X changes t3's test binary. |
| tg.grepStderr(`([\\/]compile|gccgo).*t3_test.go`, "did not recompile t3") |
| tg.grepStderr(`([\\/]link|gccgo).*t3\.test`, "did not relink t3_test") |
| tg.grepStderr(`t3\.test.*-test.short`, "did not rerun t3_test") |
| tg.grepStdoutNot(`ok \tt/t3\t\(cached\)`, "reported cached t3_test result") |
| |
| // t4 imports p2, but p2 did not change, so t4 should be relinked, not recompiled, |
| // and not rerun. |
| tg.grepStderrNot(`([\\/]compile|gccgo).*t4_test.go`, "incorrectly recompiled t4") |
| tg.grepStderr(`([\\/]link|gccgo).*t4\.test`, "did not relink t4_test") |
| // This check does not currently work with gccgo, as garbage |
| // collection of unused variables is not turned on by default. |
| if runtime.Compiler != "gccgo" { |
| tg.grepStdout(`ok \tt/t4\t\(cached\)`, "did not cache t/t4") |
| } |
| } |
| |
| func TestTestSkipVetAfterFailedBuild(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| |
| tg.tempFile("x_test.go", `package x |
| func f() { |
| return 1 |
| } |
| `) |
| |
| tg.runFail("test", tg.path("x_test.go")) |
| tg.grepStderrNot(`vet`, "vet should be skipped after the failed build") |
| } |
| |
| func TestTestVetRebuild(t *testing.T) { |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| |
| // golang.org/issue/23701. |
| // b_test imports b with augmented method from export_test.go. |
| // b_test also imports a, which imports b. |
| // Must not accidentally see un-augmented b propagate through a to b_test. |
| tg.tempFile("src/a/a.go", `package a |
| import "b" |
| type Type struct{} |
| func (*Type) M() b.T {return 0} |
| `) |
| tg.tempFile("src/b/b.go", `package b |
| type T int |
| type I interface {M() T} |
| `) |
| tg.tempFile("src/b/export_test.go", `package b |
| func (*T) Method() *T { return nil } |
| `) |
| tg.tempFile("src/b/b_test.go", `package b_test |
| import ( |
| "testing" |
| "a" |
| . "b" |
| ) |
| func TestBroken(t *testing.T) { |
| x := new(T) |
| x.Method() |
| _ = new(a.Type) |
| } |
| `) |
| |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.run("test", "b") |
| tg.run("vet", "b") |
| } |
| |
| func TestInstallDeps(t *testing.T) { |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.makeTempdir() |
| tg.setenv("GOPATH", tg.tempdir) |
| |
| tg.tempFile("src/p1/p1.go", "package p1\nvar X = 1\n") |
| tg.tempFile("src/p2/p2.go", "package p2\nimport _ \"p1\"\n") |
| tg.tempFile("src/main1/main.go", "package main\nimport _ \"p2\"\nfunc main() {}\n") |
| |
| tg.run("list", "-f={{.Target}}", "p1") |
| p1 := strings.TrimSpace(tg.getStdout()) |
| tg.run("list", "-f={{.Target}}", "p2") |
| p2 := strings.TrimSpace(tg.getStdout()) |
| tg.run("list", "-f={{.Target}}", "main1") |
| main1 := strings.TrimSpace(tg.getStdout()) |
| |
| tg.run("install", "main1") |
| |
| tg.mustExist(main1) |
| tg.mustNotExist(p2) |
| tg.mustNotExist(p1) |
| |
| tg.run("install", "p2") |
| tg.mustExist(p2) |
| tg.mustNotExist(p1) |
| |
| // don't let install -i overwrite runtime |
| tg.wantNotStale("runtime", "", "must be non-stale before install -i") |
| |
| tg.run("install", "-i", "main1") |
| tg.mustExist(p1) |
| tg.must(os.Remove(p1)) |
| |
| tg.run("install", "-i", "p2") |
| tg.mustExist(p1) |
| } |
| |
| // Issue 22986. |
| func TestImportPath(t *testing.T) { |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| |
| tg.tempFile("src/a/a.go", ` |
| package main |
| |
| import ( |
| "log" |
| p "a/p-1.0" |
| ) |
| |
| func main() { |
| if !p.V { |
| log.Fatal("false") |
| } |
| }`) |
| |
| tg.tempFile("src/a/a_test.go", ` |
| package main_test |
| |
| import ( |
| p "a/p-1.0" |
| "testing" |
| ) |
| |
| func TestV(t *testing.T) { |
| if !p.V { |
| t.Fatal("false") |
| } |
| }`) |
| |
| tg.tempFile("src/a/p-1.0/p.go", ` |
| package p |
| |
| var V = true |
| |
| func init() {} |
| `) |
| |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.run("build", "-o", tg.path("a.exe"), "a") |
| tg.run("test", "a") |
| } |
| |
| func TestBadCommandLines(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| |
| tg.tempFile("src/x/x.go", "package x\n") |
| tg.setenv("GOPATH", tg.path(".")) |
| |
| tg.run("build", "x") |
| |
| tg.tempFile("src/x/@y.go", "package x\n") |
| tg.runFail("build", "x") |
| tg.grepStderr("invalid input file name \"@y.go\"", "did not reject @y.go") |
| tg.must(os.Remove(tg.path("src/x/@y.go"))) |
| |
| tg.tempFile("src/x/-y.go", "package x\n") |
| tg.runFail("build", "x") |
| tg.grepStderr("invalid input file name \"-y.go\"", "did not reject -y.go") |
| tg.must(os.Remove(tg.path("src/x/-y.go"))) |
| |
| if runtime.Compiler == "gccgo" { |
| tg.runFail("build", "-gccgoflags=all=@x", "x") |
| } else { |
| tg.runFail("build", "-gcflags=all=@x", "x") |
| } |
| tg.grepStderr("invalid command-line argument @x in command", "did not reject @x during exec") |
| |
| tg.tempFile("src/@x/x.go", "package x\n") |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.runFail("build", "@x") |
| tg.grepStderr("invalid input directory name \"@x\"|can only use path@version syntax with 'go get' and 'go install' in module-aware mode", "did not reject @x directory") |
| |
| tg.tempFile("src/@x/y/y.go", "package y\n") |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.runFail("build", "@x/y") |
| tg.grepStderr("invalid import path \"@x/y\"|can only use path@version syntax with 'go get' and 'go install' in module-aware mode", "did not reject @x/y import path") |
| |
| tg.tempFile("src/-x/x.go", "package x\n") |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.runFail("build", "--", "-x") |
| tg.grepStderr("invalid import path \"-x\"", "did not reject -x import path") |
| |
| tg.tempFile("src/-x/y/y.go", "package y\n") |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.runFail("build", "--", "-x/y") |
| tg.grepStderr("invalid import path \"-x/y\"", "did not reject -x/y import path") |
| } |
| |
| func TestTwoPkgConfigs(t *testing.T) { |
| if !canCgo { |
| t.Skip("no cgo") |
| } |
| if runtime.GOOS == "windows" || runtime.GOOS == "plan9" { |
| t.Skipf("no shell scripts on %s", runtime.GOOS) |
| } |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempFile("src/x/a.go", `package x |
| // #cgo pkg-config: --static a |
| import "C" |
| `) |
| tg.tempFile("src/x/b.go", `package x |
| // #cgo pkg-config: --static a |
| import "C" |
| `) |
| tg.tempFile("pkg-config.sh", `#!/bin/sh |
| echo $* >>`+tg.path("pkg-config.out")) |
| tg.must(os.Chmod(tg.path("pkg-config.sh"), 0755)) |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.setenv("PKG_CONFIG", tg.path("pkg-config.sh")) |
| tg.run("build", "x") |
| out, err := os.ReadFile(tg.path("pkg-config.out")) |
| tg.must(err) |
| out = bytes.TrimSpace(out) |
| want := "--cflags --static --static -- a a\n--libs --static --static -- a a" |
| if !bytes.Equal(out, []byte(want)) { |
| t.Errorf("got %q want %q", out, want) |
| } |
| } |
| |
| func TestCgoCache(t *testing.T) { |
| if !canCgo { |
| t.Skip("no cgo") |
| } |
| tooSlow(t) |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempFile("src/x/a.go", `package main |
| // #ifndef VAL |
| // #define VAL 0 |
| // #endif |
| // int val = VAL; |
| import "C" |
| import "fmt" |
| func main() { fmt.Println(C.val) } |
| `) |
| tg.setenv("GOPATH", tg.path(".")) |
| exe := tg.path("x.exe") |
| tg.run("build", "-o", exe, "x") |
| tg.setenv("CGO_LDFLAGS", "-lnosuchlibraryexists") |
| tg.runFail("build", "-o", exe, "x") |
| tg.grepStderr(`nosuchlibraryexists`, "did not run linker with changed CGO_LDFLAGS") |
| } |
| |
| // Issue 23982 |
| func TestFilepathUnderCwdFormat(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.run("test", "-x", "-cover", "log") |
| tg.grepStderrNot(`\.log\.cover\.go`, "-x output should contain correctly formatted filepath under cwd") |
| } |
| |
| // Issue 24396. |
| func TestDontReportRemoveOfEmptyDir(t *testing.T) { |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempFile("src/a/a.go", `package a`) |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.run("install", "-x", "a") |
| tg.run("install", "-x", "a") |
| // The second install should have printed only a WORK= line, |
| // nothing else. |
| if bytes.Count(tg.stdout.Bytes(), []byte{'\n'})+bytes.Count(tg.stderr.Bytes(), []byte{'\n'}) > 1 { |
| t.Error("unnecessary output when installing installed package") |
| } |
| } |
| |
| // Issue 24704. |
| func TestLinkerTmpDirIsDeleted(t *testing.T) { |
| skipIfGccgo(t, "gccgo does not use cmd/link") |
| if !canCgo { |
| t.Skip("skipping because cgo not enabled") |
| } |
| tooSlow(t) |
| |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempFile("a.go", `package main; import "C"; func main() {}`) |
| tg.run("build", "-ldflags", "-v", "-o", os.DevNull, tg.path("a.go")) |
| // Find line that has "host link:" in linker output. |
| stderr := tg.getStderr() |
| var hostLinkLine string |
| for _, line := range strings.Split(stderr, "\n") { |
| if !strings.Contains(line, "host link:") { |
| continue |
| } |
| hostLinkLine = line |
| break |
| } |
| if hostLinkLine == "" { |
| t.Fatal(`fail to find with "host link:" string in linker output`) |
| } |
| // Find parameter, like "/tmp/go-link-408556474/go.o" inside of |
| // "host link:" line, and extract temp directory /tmp/go-link-408556474 |
| // out of it. |
| tmpdir := hostLinkLine |
| i := strings.Index(tmpdir, `go.o"`) |
| if i == -1 { |
| t.Fatalf(`fail to find "go.o" in "host link:" line %q`, hostLinkLine) |
| } |
| tmpdir = tmpdir[:i-1] |
| i = strings.LastIndex(tmpdir, `"`) |
| if i == -1 { |
| t.Fatalf(`fail to find " in "host link:" line %q`, hostLinkLine) |
| } |
| tmpdir = tmpdir[i+1:] |
| // Verify that temp directory has been removed. |
| _, err := os.Stat(tmpdir) |
| if err == nil { |
| t.Fatalf("temp directory %q has not been removed", tmpdir) |
| } |
| if !os.IsNotExist(err) { |
| t.Fatalf("Stat(%q) returns unexpected error: %v", tmpdir, err) |
| } |
| } |
| |
| // Issue 25093. |
| func TestCoverpkgTestOnly(t *testing.T) { |
| skipIfGccgo(t, "gccgo has no cover tool") |
| tooSlow(t) |
| tg := testgo(t) |
| defer tg.cleanup() |
| tg.parallel() |
| tg.tempFile("src/a/a.go", `package a |
| func F(i int) int { |
| return i*i |
| }`) |
| tg.tempFile("src/atest/a_test.go", ` |
| package a_test |
| import ( "a"; "testing" ) |
| func TestF(t *testing.T) { a.F(2) } |
| `) |
| tg.setenv("GOPATH", tg.path(".")) |
| tg.run("test", "-coverpkg=a", "atest") |
| tg.grepStderrNot("no packages being tested depend on matches", "bad match message") |
| tg.grepStdout("coverage: 100", "no coverage") |
| } |