Commit 1ab9f683 authored by Bryan C. Mills's avatar Bryan C. Mills

cmd/go/internal/modfetch: handle codeRoot == path for paths with major-version suffixes

Fixes #30647

Change-Id: Icbcfdb3907bc003ac17a8c7df17ecb41daf82eb4
Reviewed-on: https://go-review.googlesource.com/c/go/+/166117
Run-TryBot: Bryan C. Mills <bcmills@google.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarJay Conrod <jayconrod@google.com>
parent 514c5593
...@@ -23,55 +23,99 @@ import ( ...@@ -23,55 +23,99 @@ import (
type codeRepo struct { type codeRepo struct {
modPath string modPath string
// code is the repository containing this module.
code codehost.Repo code codehost.Repo
// codeRoot is the import path at the root of code.
codeRoot string codeRoot string
// codeDir is the directory (relative to root) at which we expect to find the module.
// If pathMajor is non-empty and codeRoot is not the full modPath,
// then we look in both codeDir and codeDir+modPath
codeDir string codeDir string
path string // pathMajor is the suffix of modPath that indicates its major version,
pathPrefix string // or the empty string if modPath is at major version 0 or 1.
//
// pathMajor is typically of the form "/vN", but possibly ".vN", or
// ".vN-unstable" for modules resolved using gopkg.in.
pathMajor string pathMajor string
// pathPrefix is the prefix of modPath that excludes pathMajor.
// It is used only for logging.
pathPrefix string
// pseudoMajor is the major version prefix to use when generating
// pseudo-versions for this module, derived from the module path.
//
// TODO(golang.org/issue/29262): We can't distinguish v0 from v1 using the
// path alone: we have to compute it by examining the tags at a particular
// revision.
pseudoMajor string pseudoMajor string
} }
func newCodeRepo(code codehost.Repo, root, path string) (Repo, error) { // newCodeRepo returns a Repo that reads the source code for the module with the
if !hasPathPrefix(path, root) { // given path, from the repo stored in code, with the root of the repo
return nil, fmt.Errorf("mismatched repo: found %s for %s", root, path) // containing the path given by codeRoot.
func newCodeRepo(code codehost.Repo, codeRoot, path string) (Repo, error) {
if !hasPathPrefix(path, codeRoot) {
return nil, fmt.Errorf("mismatched repo: found %s for %s", codeRoot, path)
} }
pathPrefix, pathMajor, ok := module.SplitPathVersion(path) pathPrefix, pathMajor, ok := module.SplitPathVersion(path)
if !ok { if !ok {
return nil, fmt.Errorf("invalid module path %q", path) return nil, fmt.Errorf("invalid module path %q", path)
} }
if codeRoot == path {
pathPrefix = path
}
pseudoMajor := "v0" pseudoMajor := "v0"
if pathMajor != "" { if pathMajor != "" {
pseudoMajor = pathMajor[1:] pseudoMajor = pathMajor[1:]
} }
// Compute codeDir = bar, the subdirectory within the repo
// corresponding to the module root.
//
// At this point we might have: // At this point we might have:
// codeRoot = github.com/rsc/foo
// path = github.com/rsc/foo/bar/v2 // path = github.com/rsc/foo/bar/v2
// codeRoot = github.com/rsc/foo
// pathPrefix = github.com/rsc/foo/bar // pathPrefix = github.com/rsc/foo/bar
// pathMajor = /v2 // pathMajor = /v2
// pseudoMajor = v2 // pseudoMajor = v2
// //
// Compute codeDir = bar, the subdirectory within the repo // which gives
// corresponding to the module root. // codeDir = bar
codeDir := strings.Trim(strings.TrimPrefix(pathPrefix, root), "/") //
if strings.HasPrefix(path, "gopkg.in/") { // We know that pathPrefix is a prefix of path, and codeRoot is a prefix of
// But gopkg.in is a special legacy case, in which pathPrefix does not start with codeRoot. // path, but codeRoot may or may not be a prefix of pathPrefix, because
// For example we might have: // codeRoot may be the entire path (in which case codeDir should be empty).
// That occurs in two situations.
//
// One is when a go-import meta tag resolves the complete module path,
// including the pathMajor suffix:
// path = nanomsg.org/go/mangos/v2
// codeRoot = nanomsg.org/go/mangos/v2
// pathPrefix = nanomsg.org/go/mangos
// pathMajor = /v2
// pseudoMajor = v2
//
// The other is similar: for gopkg.in only, the major version is encoded
// with a dot rather than a slash, and thus can't be in a subdirectory.
// path = gopkg.in/yaml.v2
// codeRoot = gopkg.in/yaml.v2 // codeRoot = gopkg.in/yaml.v2
// pathPrefix = gopkg.in/yaml // pathPrefix = gopkg.in/yaml
// pathMajor = .v2 // pathMajor = .v2
// pseudoMajor = v2 // pseudoMajor = v2
// codeDir = pathPrefix (because codeRoot is not a prefix of pathPrefix) //
// Clear codeDir - the module root is the repo root for gopkg.in repos. codeDir := ""
codeDir = "" if codeRoot != path {
if !hasPathPrefix(pathPrefix, codeRoot) {
return nil, fmt.Errorf("repository rooted at %s cannot contain module %s", codeRoot, path)
}
codeDir = strings.Trim(pathPrefix[len(codeRoot):], "/")
} }
r := &codeRepo{ r := &codeRepo{
modPath: path, modPath: path,
code: code, code: code,
codeRoot: root, codeRoot: codeRoot,
codeDir: codeDir, codeDir: codeDir,
pathPrefix: pathPrefix, pathPrefix: pathPrefix,
pathMajor: pathMajor, pathMajor: pathMajor,
...@@ -149,9 +193,6 @@ func (r *codeRepo) Stat(rev string) (*RevInfo, error) { ...@@ -149,9 +193,6 @@ func (r *codeRepo) Stat(rev string) (*RevInfo, error) {
return r.Latest() return r.Latest()
} }
codeRev := r.revToRev(rev) codeRev := r.revToRev(rev)
if semver.IsValid(codeRev) && r.codeDir != "" {
codeRev = r.codeDir + "/" + codeRev
}
info, err := r.code.Stat(codeRev) info, err := r.code.Stat(codeRev)
if err != nil { if err != nil {
return nil, err return nil, err
...@@ -290,7 +331,7 @@ func (r *codeRepo) findDir(version string) (rev, dir string, gomod []byte, err e ...@@ -290,7 +331,7 @@ func (r *codeRepo) findDir(version string) (rev, dir string, gomod []byte, err e
found1 := err1 == nil && isMajor(mpath1, r.pathMajor) found1 := err1 == nil && isMajor(mpath1, r.pathMajor)
var file2 string var file2 string
if r.pathMajor != "" && !strings.HasPrefix(r.pathMajor, ".") { if r.pathMajor != "" && r.codeRoot != r.modPath && !strings.HasPrefix(r.pathMajor, ".") {
// Suppose pathMajor is "/v2". // Suppose pathMajor is "/v2".
// Either go.mod should claim v2 and v2/go.mod should not exist, // Either go.mod should claim v2 and v2/go.mod should not exist,
// or v2/go.mod should exist and claim v2. Not both. // or v2/go.mod should exist and claim v2. Not both.
...@@ -298,6 +339,9 @@ func (r *codeRepo) findDir(version string) (rev, dir string, gomod []byte, err e ...@@ -298,6 +339,9 @@ func (r *codeRepo) findDir(version string) (rev, dir string, gomod []byte, err e
// because of replacement modules. This might be a fork of // because of replacement modules. This might be a fork of
// the real module, found at a different path, usable only in // the real module, found at a different path, usable only in
// a replace directive. // a replace directive.
//
// TODO(bcmills): This doesn't seem right. Investigate futher.
// (Notably: why can't we replace foo/v2 with fork-of-foo/v3?)
dir2 := path.Join(r.codeDir, r.pathMajor[1:]) dir2 := path.Join(r.codeDir, r.pathMajor[1:])
file2 = path.Join(dir2, "go.mod") file2 = path.Join(dir2, "go.mod")
gomod2, err2 := r.code.ReadFile(rev, file2, codehost.MaxGoMod) gomod2, err2 := r.code.ReadFile(rev, file2, codehost.MaxGoMod)
...@@ -418,7 +462,7 @@ func (r *codeRepo) Zip(dst io.Writer, version string) error { ...@@ -418,7 +462,7 @@ func (r *codeRepo) Zip(dst io.Writer, version string) error {
} }
defer dl.Close() defer dl.Close()
if actualDir != "" && !hasPathPrefix(dir, actualDir) { if actualDir != "" && !hasPathPrefix(dir, actualDir) {
return fmt.Errorf("internal error: downloading %v %v: dir=%q but actualDir=%q", r.path, rev, dir, actualDir) return fmt.Errorf("internal error: downloading %v %v: dir=%q but actualDir=%q", r.modPath, rev, dir, actualDir)
} }
subdir := strings.Trim(strings.TrimPrefix(dir, actualDir), "/") subdir := strings.Trim(strings.TrimPrefix(dir, actualDir), "/")
......
...@@ -323,6 +323,15 @@ var codeRepoTests = []struct { ...@@ -323,6 +323,15 @@ var codeRepoTests = []struct {
time: time.Date(2017, 5, 31, 16, 3, 50, 0, time.UTC), time: time.Date(2017, 5, 31, 16, 3, 50, 0, time.UTC),
gomod: "module gopkg.in/natefinch/lumberjack.v2\n", gomod: "module gopkg.in/natefinch/lumberjack.v2\n",
}, },
{
path: "nanomsg.org/go/mangos/v2",
rev: "v2.0.2",
version: "v2.0.2",
name: "63f66a65137b9a648ac9f7bf0160b4a4d17d7999",
short: "63f66a65137b",
time: time.Date(2018, 12, 1, 15, 7, 40, 0, time.UTC),
gomod: "module nanomsg.org/go/mangos/v2\n\nrequire (\n\tgithub.com/Microsoft/go-winio v0.4.11\n\tgithub.com/droundy/goopt v0.0.0-20170604162106-0b8effe182da\n\tgithub.com/gopherjs/gopherjs v0.0.0-20181103185306-d547d1d9531e // indirect\n\tgithub.com/gorilla/websocket v1.4.0\n\tgithub.com/jtolds/gls v4.2.1+incompatible // indirect\n\tgithub.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d // indirect\n\tgithub.com/smartystreets/goconvey v0.0.0-20181108003508-044398e4856c\n\tgolang.org/x/sys v0.0.0-20181128092732-4ed8d59d0b35 // indirect\n)\n",
},
} }
func TestCodeRepo(t *testing.T) { func TestCodeRepo(t *testing.T) {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment