Skip to content

Commit

Permalink
Make go_bazel_test pass with Bzlmod enabled
Browse files Browse the repository at this point in the history
This is a quick fix for the test runner only, the tests themselves still rely on WORKSPACE (via local_repository).
The rules_go repo is also hardcoded, which means that other external repos likely won't work.
  • Loading branch information
fmeum committed Jul 31, 2024
1 parent fb3afab commit b30b469
Show file tree
Hide file tree
Showing 2 changed files with 44 additions and 14 deletions.
55 changes: 41 additions & 14 deletions go/tools/bazel_testing/bazel_testing.go
Original file line number Diff line number Diff line change
Expand Up @@ -101,6 +101,8 @@ type Args struct {
// instead of running tests.
const debug = false

var bzlmodEnabled = os.Getenv("BZLMOD_ENABLED") == "True"

// outputUserRoot is set to the directory where Bazel should put its internal files.
// Since Bazel 2.0.0, this needs to be set explicitly to avoid it defaulting to a
// deeply nested directory within the test, which runs into Windows path length limits.
Expand Down Expand Up @@ -342,29 +344,37 @@ func setupWorkspace(args Args, files []string) (dir string, cleanup func() error
return "", cleanup, fmt.Errorf("building main workspace: %v", err)
}

var runfilesWorkspaceName string
// If some of the path arguments are missing an explicit workspace,
// read the workspace name from WORKSPACE. We need this to map arguments
// to runfiles in specific workspaces.
haveDefaultWorkspace := false
var defaultWorkspaceName string
for _, argPath := range files {
workspace, _, err := parseLocationArg(argPath)
if err == nil && workspace == "" {
haveDefaultWorkspace = true
cleanPath := path.Clean(argPath)
if cleanPath == "WORKSPACE" {
defaultWorkspaceName, err = loadWorkspaceName(cleanPath)
runfilesWorkspaceName, err = loadWorkspaceName(cleanPath)
if err != nil {
return "", cleanup, fmt.Errorf("could not load default workspace name: %v", err)
}
break
}
}
}
if haveDefaultWorkspace && defaultWorkspaceName == "" {
if haveDefaultWorkspace && runfilesWorkspaceName == "" {
return "", cleanup, fmt.Errorf("found files from default workspace, but not WORKSPACE")
}

var mainRepoName string
if bzlmodEnabled {
mainRepoName = runfilesWorkspaceName
runfilesWorkspaceName = "_main"
} else {
mainRepoName = runfilesWorkspaceName
}

// Index runfiles by workspace and short path. We need this to determine
// destination paths when we copy or link files.
runfiles, err := bazel.ListRunfiles()
Expand All @@ -379,20 +389,22 @@ func setupWorkspace(args Args, files []string) (dir string, cleanup func() error
}

// Copy or link file arguments from runfiles into fake workspace dirctories.
// Keep track of the workspace names we see, since we'll generate a WORKSPACE
// Keep track of the repo names we see, since we'll generate a WORKSPACE
// with local_repository rules later.
workspaceNames := make(map[string]bool)
canonicalRepoNames := make(map[string]bool)
for _, argPath := range files {
workspace, shortPath, err := parseLocationArg(argPath)
if err != nil {
return "", cleanup, err
}
key := runfileKey{workspace, shortPath}
if workspace == "" {
workspace = defaultWorkspaceName
workspace = mainRepoName
key.workspace = runfilesWorkspaceName
}
workspaceNames[workspace] = true
canonicalRepoNames[workspace] = true

srcPath, ok := runfileMap[runfileKey{workspace, shortPath}]
srcPath, ok := runfileMap[key]
if !ok {
return "", cleanup, fmt.Errorf("unknown runfile: %s", argPath)
}
Expand Down Expand Up @@ -422,10 +434,12 @@ func setupWorkspace(args Args, files []string) (dir string, cleanup func() error
NogoIncludes: args.NogoIncludes,
NogoExcludes: args.NogoExcludes,
}
for name := range workspaceNames {
info.WorkspaceNames = append(info.WorkspaceNames, name)
for name := range canonicalRepoNames {
info.WorkspaceNames = append(info.WorkspaceNames, makeApparent(name))
}
sort.Strings(info.WorkspaceNames)
sort.Slice(info.WorkspaceNames, func(i, j int) bool {
return info.WorkspaceNames[i].ApparentName < info.WorkspaceNames[j].ApparentName
})
if outBaseDir != "" {
goSDKPath := filepath.Join(outBaseDir, "external", "go_sdk")
rel, err := filepath.Rel(mainDir, goSDKPath)
Expand Down Expand Up @@ -474,6 +488,14 @@ func setupWorkspace(args Args, files []string) (dir string, cleanup func() error
return mainDir, cleanup, nil
}

func makeApparent(canonicalName string) workspaceName {
name := workspaceName{ApparentName: canonicalName, CanonicalName: canonicalName}
if canonicalName == "rules_go~" || canonicalName == "rules_go+" {
name.ApparentName = "io_bazel_rules_go"
}
return name
}

func extractTxtar(dir, txt string) error {
ar := txtar.Parse([]byte(txt))
for _, f := range ar.Files {
Expand Down Expand Up @@ -531,8 +553,13 @@ func loadWorkspaceName(workspacePath string) (string, error) {
return name, nil
}

type workspaceName struct {
ApparentName string
CanonicalName string
}

type workspaceTemplateInfo struct {
WorkspaceNames []string
WorkspaceNames []workspaceName
GoSDKPath string
Nogo string
NogoIncludes []string
Expand All @@ -544,8 +571,8 @@ type workspaceTemplateInfo struct {
var defaultWorkspaceTpl = template.Must(template.New("").Parse(`
{{range .WorkspaceNames}}
local_repository(
name = "{{.}}",
path = "../{{.}}",
name = "{{.ApparentName}}",
path = "../{{.CanonicalName}}",
)
{{end}}
Expand Down
3 changes: 3 additions & 0 deletions go/tools/bazel_testing/def.bzl
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,9 @@ def go_bazel_test(rule_files = None, **kwargs):
["-end_files"] +
kwargs["args"])

kwargs.setdefault("env", {})
kwargs["env"]["BZLMOD_ENABLED"] = str(str(Label("//:foo")).startswith("@@"))

# Set rundir to the workspace root directory to ensure relative paths
# are interpreted correctly.
kwargs.setdefault("rundir", ".")
Expand Down

0 comments on commit b30b469

Please sign in to comment.