// Copyright 2020 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 moddeps_test import ( "bytes" "encoding/json" "fmt" "internal/testenv" "io" "io/fs" "os" "path/filepath" "slices" "sort" "strings" "sync" "testing" "golang.org/x/mod/module" ) // TestAllDependencies ensures dependencies of all // modules in GOROOT are in a consistent state. // // In short mode, it does a limited quick check and stops there. // In long mode, it also makes a copy of the entire GOROOT tree // and requires network access to perform more thorough checks. // Keep this distinction in mind when adding new checks. // // See issues 36852, 41409, and 43687. // (Also see golang.org/issue/27348.) func TestAllDependencies(t *testing.T) { goBin := testenv.GoToolPath(t) // Ensure that all packages imported within GOROOT // are vendored in the corresponding GOROOT module. // // This property allows offline development within the Go project, and ensures // that all dependency changes are presented in the usual code review process. // // As a quick first-order check, avoid network access and the need to copy the // entire GOROOT tree or explicitly invoke version control to check for changes. // Just check that packages are vendored. (In non-short mode, we go on to also // copy the GOROOT tree and perform more rigorous consistency checks. Jump below // for more details.) for _, m := range findGorootModules(t) { // This short test does NOT ensure that the vendored contents match // the unmodified contents of the corresponding dependency versions. t.Run(m.Path+"(quick)", func(t *testing.T) { t.Logf("module %s in directory %s", m.Path, m.Dir) if m.hasVendor { // Load all of the packages in the module to ensure that their // dependencies are vendored. If any imported package is missing, // 'go list -deps' will fail when attempting to load it. cmd := testenv.Command(t, goBin, "list", "-mod=vendor", "-deps", "./...") cmd.Dir = m.Dir cmd.Env = append(cmd.Environ(), "GO111MODULE=on", "GOWORK=off") cmd.Stderr = new(strings.Builder) _, err := cmd.Output() if err != nil { t.Errorf("%s: %v\n%s", strings.Join(cmd.Args, " "), err, cmd.Stderr) t.Logf("(Run 'go mod vendor' in %s to ensure that dependencies have been vendored.)", m.Dir) } return } // There is no vendor directory, so the module must have no dependencies. // Check that the list of active modules contains only the main module. cmd := testenv.Command(t, goBin, "list", "-mod=readonly", "-m", "all") cmd.Dir = m.Dir cmd.Env = append(cmd.Environ(), "GO111MODULE=on", "GOWORK=off") cmd.Stderr = new(strings.Builder) out, err := cmd.Output() if err != nil { t.Fatalf("%s: %v\n%s", strings.Join(cmd.Args, " "), err, cmd.Stderr) } if strings.TrimSpace(string(out)) != m.Path { t.Errorf("'%s' reported active modules other than %s:\n%s", strings.Join(cmd.Args, " "), m.Path, out) t.Logf("(Run 'go mod tidy' in %s to ensure that no extraneous dependencies were added, or 'go mod vendor' to copy in imported packages.)", m.Dir) } }) } // We now get to the slow, but more thorough part of the test. // Only run it in long test mode. if testing.Short() { return } // Ensure that all modules within GOROOT are tidy, vendored, and bundled. // Ensure that the vendored contents match the unmodified contents of the // corresponding dependency versions. // // The non-short section of this test requires network access and the diff // command. // // It makes a temporary copy of the entire GOROOT tree (where it can safely // perform operations that may mutate the tree), executes the same module // maintenance commands that we expect Go developers to run, and then // diffs the potentially modified module copy with the real one in GOROOT. // (We could try to rely on Git to do things differently, but that's not the // path we've chosen at this time. This allows the test to run when the tree // is not checked into Git.) testenv.MustHaveExternalNetwork(t) if haveDiff := func() bool { diff, err := testenv.Command(t, "diff", "--recursive", "--unified", ".", ".").CombinedOutput() if err != nil || len(diff) != 0 { return false } diff, err = testenv.Command(t, "diff", "--recursive", "--unified", ".", "..").CombinedOutput() if err == nil || len(diff) == 0 { return false } return true }(); !haveDiff { // For now, the diff command is a mandatory dependency of this test. // This test will primarily run on longtest builders, since few people // would test the cmd/internal/moddeps package directly, and all.bash // runs tests in short mode. It's fine to skip if diff is unavailable. t.Skip("skipping because a diff command with support for --recursive and --unified flags is unavailable") } // We're going to check the standard modules for tidiness, so we need a usable // GOMODCACHE. If the default directory doesn't exist, use a temporary // directory instead. (That can occur, for example, when running under // run.bash with GO_TEST_SHORT=0: run.bash sets GOPATH=/nonexist-gopath, and // GO_TEST_SHORT=0 causes it to run this portion of the test.) var modcacheEnv []string { out, err := testenv.Command(t, goBin, "env", "GOMODCACHE").Output() if err != nil { t.Fatalf("%s env GOMODCACHE: %v", goBin, err) } modcacheOk := false if gomodcache := string(bytes.TrimSpace(out)); gomodcache != "" { if _, err := os.Stat(gomodcache); err == nil { modcacheOk = true } } if !modcacheOk { modcacheEnv = []string{ "GOMODCACHE=" + t.TempDir(), "GOFLAGS=" + os.Getenv("GOFLAGS") + " -modcacherw", // Allow t.TempDir() to clean up subdirectories. } } } // Build the bundle binary at the golang.org/x/tools // module version specified in GOROOT/src/cmd/go.mod. bundleDir := t.TempDir() r := runner{ Dir: filepath.Join(testenv.GOROOT(t), "src/cmd"), Env: append(os.Environ(), modcacheEnv...), } r.run(t, goBin, "build", "-mod=readonly", "-o", bundleDir, "golang.org/x/tools/cmd/bundle") var gorootCopyDir string for _, m := range findGorootModules(t) { // Create a test-wide GOROOT copy. It can be created once // and reused between subtests whenever they don't fail. // // This is a relatively expensive operation, but it's a pre-requisite to // be able to safely run commands like "go mod tidy", "go mod vendor", and // "go generate" on the GOROOT tree content. Those commands may modify the // tree, and we don't want to happen to the real tree as part of executing // a test. if gorootCopyDir == "" { gorootCopyDir = makeGOROOTCopy(t) } t.Run(m.Path+"(thorough)", func(t *testing.T) { t.Logf("module %s in directory %s", m.Path, m.Dir) defer func() { if t.Failed() { // The test failed, which means it's possible the GOROOT copy // may have been modified. No choice but to reset it for next // module test case. (This is slow, but it happens only during // test failures.) gorootCopyDir = "" } }() rel, err := filepath.Rel(testenv.GOROOT(t), m.Dir) if err != nil { t.Fatalf("filepath.Rel(%q, %q): %v", testenv.GOROOT(t), m.Dir, err) } r := runner{ Dir: filepath.Join(gorootCopyDir, rel), Env: append(append(os.Environ(), modcacheEnv...), // Set GOROOT. "GOROOT="+gorootCopyDir, // Add GOROOTcopy/bin and bundleDir to front of PATH. "PATH="+filepath.Join(gorootCopyDir, "bin")+string(filepath.ListSeparator)+ bundleDir+string(filepath.ListSeparator)+os.Getenv("PATH"), "GOWORK=off", ), } goBinCopy := filepath.Join(gorootCopyDir, "bin", "go") r.run(t, goBinCopy, "mod", "tidy") // See issue 43687. r.run(t, goBinCopy, "mod", "verify") // Verify should be a no-op, but test it just in case. r.run(t, goBinCopy, "mod", "vendor") // See issue 36852. pkgs := packagePattern(m.Path) r.run(t, goBinCopy, "generate", `-run=^//go:generate bundle `, pkgs) // See issue 41409. advice := "$ cd " + m.Dir + "\n" + "$ go mod tidy # to remove extraneous dependencies\n" + "$ go mod vendor # to vendor dependencies\n" + "$ go generate -run=bundle " + pkgs + " # to regenerate bundled packages\n" if m.Path == "std" { r.run(t, goBinCopy, "generate", "syscall", "internal/syscall/...") // See issue 43440. advice += "$ go generate syscall internal/syscall/... # to regenerate syscall packages\n" } // TODO(golang.org/issue/43440): Check anything else influenced by dependency versions. diff, err := testenv.Command(t, "diff", "--recursive", "--unified", r.Dir, m.Dir).CombinedOutput() if err != nil || len(diff) != 0 { t.Errorf(`Module %s in %s is not tidy (-want +got): %s To fix it, run: %s (If module %[1]s is definitely tidy, this could mean there's a problem in the go or bundle command.)`, m.Path, m.Dir, diff, advice) } }) } } // packagePattern returns a package pattern that matches all packages // in the module modulePath, and ideally as few others as possible. func packagePattern(modulePath string) string { if modulePath == "std" { return "std" } return modulePath + "/..." } // makeGOROOTCopy makes a temporary copy of the current GOROOT tree. // The goal is to allow the calling test t to safely mutate a GOROOT // copy without also modifying the original GOROOT. // // It copies the entire tree as is, with the exception of the GOROOT/.git // directory, which is skipped, and the GOROOT/{bin,pkg} directories, // which are symlinked. This is done for speed, since a GOROOT tree is // functional without being in a Git repository, and bin and pkg are // deemed safe to share for the purpose of the TestAllDependencies test. func makeGOROOTCopy(t *testing.T) string { t.Helper() gorootCopyDir := t.TempDir() err := filepath.Walk(testenv.GOROOT(t), func(src string, info os.FileInfo, err error) error { if err != nil { return err } if info.IsDir() && src == filepath.Join(testenv.GOROOT(t), ".git") { return filepath.SkipDir } rel, err := filepath.Rel(testenv.GOROOT(t), src) if err != nil { return fmt.Errorf("filepath.Rel(%q, %q): %v", testenv.GOROOT(t), src, err) } dst := filepath.Join(gorootCopyDir, rel) if info.IsDir() && (src == filepath.Join(testenv.GOROOT(t), "bin") || src == filepath.Join(testenv.GOROOT(t), "pkg")) { // If the OS supports symlinks, use them instead // of copying the bin and pkg directories. if err := os.Symlink(src, dst); err == nil { return filepath.SkipDir } } perm := info.Mode() & os.ModePerm if info.Mode()&os.ModeSymlink != 0 { info, err = os.Stat(src) if err != nil { return err } perm = info.Mode() & os.ModePerm } // If it's a directory, make a corresponding directory. if info.IsDir() { return os.MkdirAll(dst, perm|0200) } // Copy the file bytes. // We can't create a symlink because the file may get modified; // we need to ensure that only the temporary copy is affected. s, err := os.Open(src) if err != nil { return err } defer s.Close() d, err := os.OpenFile(dst, os.O_WRONLY|os.O_CREATE|os.O_EXCL, perm) if err != nil { return err } _, err = io.Copy(d, s) if err != nil { d.Close() return err } return d.Close() }) if err != nil { t.Fatal(err) } t.Logf("copied GOROOT from %s to %s", testenv.GOROOT(t), gorootCopyDir) return gorootCopyDir } type runner struct { Dir string Env []string } // run runs the command and requires that it succeeds. func (r runner) run(t *testing.T, args ...string) { t.Helper() cmd := testenv.Command(t, args[0], args[1:]...) cmd.Dir = r.Dir cmd.Env = slices.Clip(r.Env) if r.Dir != "" { cmd.Env = append(cmd.Env, "PWD="+r.Dir) } out, err := cmd.CombinedOutput() if err != nil { t.Logf("> %s\n", strings.Join(args, " ")) t.Fatalf("command failed: %s\n%s", err, out) } } // TestDependencyVersionsConsistent verifies that each module in GOROOT that // requires a given external dependency requires the same version of that // dependency. // // This property allows us to maintain a single release branch of each such // dependency, minimizing the number of backports needed to pull in critical // fixes. It also ensures that any bug detected and fixed in one GOROOT module // (such as "std") is fixed in all other modules (such as "cmd") as well. func TestDependencyVersionsConsistent(t *testing.T) { // Collect the dependencies of all modules in GOROOT, indexed by module path. type requirement struct { Required module.Version Replacement module.Version } seen := map[string]map[requirement][]gorootModule{} // module path → requirement → set of modules with that requirement for _, m := range findGorootModules(t) { if !m.hasVendor { // TestAllDependencies will ensure that the module has no dependencies. continue } // We want this test to be able to run offline and with an empty module // cache, so we verify consistency only for the module versions listed in // vendor/modules.txt. That includes all direct dependencies and all modules // that provide any imported packages. // // It's ok if there are undetected differences in modules that do not // provide imported packages: we will not have to pull in any backports of // fixes to those modules anyway. vendor, err := os.ReadFile(filepath.Join(m.Dir, "vendor", "modules.txt")) if err != nil { t.Error(err) continue } for _, line := range strings.Split(strings.TrimSpace(string(vendor)), "\n") { parts := strings.Fields(line) if len(parts) < 3 || parts[0] != "#" { continue } // This line is of the form "# module version [=> replacement [version]]". var r requirement r.Required.Path = parts[1] r.Required.Version = parts[2] if len(parts) >= 5 && parts[3] == "=>" { r.Replacement.Path = parts[4] if module.CheckPath(r.Replacement.Path) != nil { // If the replacement is a filesystem path (rather than a module path), // we don't know whether the filesystem contents have changed since // the module was last vendored. // // Fortunately, we do not currently use filesystem-local replacements // in GOROOT modules. t.Errorf("cannot check consistency for filesystem-local replacement in module %s (%s):\n%s", m.Path, m.Dir, line) } if len(parts) >= 6 { r.Replacement.Version = parts[5] } } if seen[r.Required.Path] == nil { seen[r.Required.Path] = make(map[requirement][]gorootModule) } seen[r.Required.Path][r] = append(seen[r.Required.Path][r], m) } } // Now verify that we saw only one distinct version for each module. for path, versions := range seen { if len(versions) > 1 { t.Errorf("Modules within GOROOT require different versions of %s.", path) for r, mods := range versions { desc := new(strings.Builder) desc.WriteString(r.Required.Version) if r.Replacement.Path != "" { fmt.Fprintf(desc, " => %s", r.Replacement.Path) if r.Replacement.Version != "" { fmt.Fprintf(desc, " %s", r.Replacement.Version) } } for _, m := range mods { t.Logf("%s\trequires %v", m.Path, desc) } } } } } type gorootModule struct { Path string Dir string hasVendor bool } // findGorootModules returns the list of modules found in the GOROOT source tree. func findGorootModules(t *testing.T) []gorootModule { t.Helper() goBin := testenv.GoToolPath(t) goroot.once.Do(func() { // If the root itself is a symlink to a directory, // we want to follow it (see https://go.dev/issue/64375). // Add a trailing separator to force that to happen. root := testenv.GOROOT(t) if !os.IsPathSeparator(root[len(root)-1]) { root += string(filepath.Separator) } goroot.err = filepath.WalkDir(root, func(path string, info fs.DirEntry, err error) error { if err != nil { return err } if info.IsDir() && path != root && (info.Name() == "vendor" || info.Name() == "testdata") { return filepath.SkipDir } if info.IsDir() && path == filepath.Join(testenv.GOROOT(t), "pkg") { // GOROOT/pkg contains generated artifacts, not source code. // // In https://golang.org/issue/37929 it was observed to somehow contain // a module cache, so it is important to skip. (That helps with the // running time of this test anyway.) return filepath.SkipDir } if info.IsDir() && path != root && (strings.HasPrefix(info.Name(), "_") || strings.HasPrefix(info.Name(), ".")) { // _ and . prefixed directories can be used for internal modules // without a vendor directory that don't contribute to the build // but might be used for example as code generators. return filepath.SkipDir } if info.IsDir() || info.Name() != "go.mod" { return nil } dir := filepath.Dir(path) // Use 'go list' to describe the module contained in this directory (but // not its dependencies). cmd := testenv.Command(t, goBin, "list", "-json", "-m") cmd.Dir = dir cmd.Env = append(cmd.Environ(), "GO111MODULE=on", "GOWORK=off") cmd.Stderr = new(strings.Builder) out, err := cmd.Output() if err != nil { return fmt.Errorf("'go list -json -m' in %s: %w\n%s", dir, err, cmd.Stderr) } var m gorootModule if err := json.Unmarshal(out, &m); err != nil { return fmt.Errorf("decoding 'go list -json -m' in %s: %w", dir, err) } if m.Path == "" || m.Dir == "" { return fmt.Errorf("'go list -json -m' in %s failed to populate Path and/or Dir", dir) } if _, err := os.Stat(filepath.Join(dir, "vendor")); err == nil { m.hasVendor = true } goroot.modules = append(goroot.modules, m) return nil }) if goroot.err != nil { return } // knownGOROOTModules is a hard-coded list of modules that are known to exist in GOROOT. // If findGorootModules doesn't find a module, it won't be covered by tests at all, // so make sure at least these modules are found. See issue 46254. If this list // becomes a nuisance to update, can be replaced with len(goroot.modules) check. knownGOROOTModules := [...]string{ "std", "cmd", // The "misc" module sometimes exists, but cmd/distpack intentionally removes it. } var seen = make(map[string]bool) // Key is module path. for _, m := range goroot.modules { seen[m.Path] = true } for _, m := range knownGOROOTModules { if !seen[m] { goroot.err = fmt.Errorf("findGorootModules didn't find the well-known module %q", m) break } } sort.Slice(goroot.modules, func(i, j int) bool { return goroot.modules[i].Dir < goroot.modules[j].Dir }) }) if goroot.err != nil { t.Fatal(goroot.err) } return goroot.modules } // goroot caches the list of modules found in the GOROOT source tree. var goroot struct { once sync.Once modules []gorootModule err error }