]> Cypherpunks repositories - gostls13.git/commitdiff
cmd/go/internal/modfetch/codehost: return VCSError for unimplemented functions and...
authorBryan C. Mills <bcmills@google.com>
Mon, 10 Dec 2018 19:58:08 +0000 (14:58 -0500)
committerBryan C. Mills <bcmills@google.com>
Tue, 11 Dec 2018 21:35:21 +0000 (21:35 +0000)
Updates #28943
Updates #26092

Change-Id: I07af2731ef5af046b9f7c7280ccb3976cdf41ca4
Reviewed-on: https://go-review.googlesource.com/c/153458
Run-TryBot: Bryan C. Mills <bcmills@google.com>
Reviewed-by: Jay Conrod <jayconrod@google.com>
src/cmd/go/internal/modfetch/codehost/vcs.go
src/cmd/go/testdata/script/mod_get_svn.txt [new file with mode: 0644]

index 190f47cf8d1f01847951973de8c8bfcdf77c120e..59c2b15d19a06c4fb6b525f45ca3cf7db68be201 100644 (file)
@@ -28,12 +28,19 @@ import (
 // to get the code, but we can't access it due to the error.
 // The caller should report this error instead of continuing to probe
 // other possible module paths.
+//
+// TODO(bcmills): See if we can invert this. (Return a distinguished error for
+// “repo not found” and treat everything else as terminal.)
 type VCSError struct {
        Err error
 }
 
 func (e *VCSError) Error() string { return e.Err.Error() }
 
+func vcsErrorf(format string, a ...interface{}) error {
+       return &VCSError{Err: fmt.Errorf(format, a...)}
+}
+
 func NewRepo(vcs, remote string) (Repo, error) {
        type key struct {
                vcs    string
@@ -339,7 +346,7 @@ func (r *vcsRepo) fetch() {
 func (r *vcsRepo) statLocal(rev string) (*RevInfo, error) {
        out, err := Run(r.dir, r.cmd.statLocal(rev, r.remote))
        if err != nil {
-               return nil, fmt.Errorf("unknown revision %s", rev)
+               return nil, vcsErrorf("unknown revision %s", rev)
        }
        return r.cmd.parseStat(rev, string(out))
 }
@@ -381,7 +388,7 @@ func (r *vcsRepo) ReadFileRevs(revs []string, file string, maxSize int64) (map[s
        }
        defer unlock()
 
-       return nil, fmt.Errorf("ReadFileRevs not implemented")
+       return nil, vcsErrorf("ReadFileRevs not implemented")
 }
 
 func (r *vcsRepo) RecentTag(rev, prefix string) (tag string, err error) {
@@ -394,10 +401,14 @@ func (r *vcsRepo) RecentTag(rev, prefix string) (tag string, err error) {
        }
        defer unlock()
 
-       return "", fmt.Errorf("RecentTags not implemented")
+       return "", vcsErrorf("RecentTag not implemented")
 }
 
 func (r *vcsRepo) ReadZip(rev, subdir string, maxSize int64) (zip io.ReadCloser, actualSubdir string, err error) {
+       if r.cmd.readZip == nil {
+               return nil, "", vcsErrorf("ReadZip not implemented for %s", r.cmd.vcs)
+       }
+
        unlock, err := r.mu.Lock()
        if err != nil {
                return nil, "", err
@@ -448,7 +459,7 @@ func (d *deleteCloser) Close() error {
 func hgParseStat(rev, out string) (*RevInfo, error) {
        f := strings.Fields(string(out))
        if len(f) < 3 {
-               return nil, fmt.Errorf("unexpected response from hg log: %q", out)
+               return nil, vcsErrorf("unexpected response from hg log: %q", out)
        }
        hash := f[0]
        version := rev
@@ -457,7 +468,7 @@ func hgParseStat(rev, out string) (*RevInfo, error) {
        }
        t, err := strconv.ParseInt(f[1], 10, 64)
        if err != nil {
-               return nil, fmt.Errorf("invalid time from hg log: %q", out)
+               return nil, vcsErrorf("invalid time from hg log: %q", out)
        }
 
        var tags []string
@@ -486,12 +497,12 @@ func svnParseStat(rev, out string) (*RevInfo, error) {
                } `xml:"logentry"`
        }
        if err := xml.Unmarshal([]byte(out), &log); err != nil {
-               return nil, fmt.Errorf("unexpected response from svn log --xml: %v\n%s", err, out)
+               return nil, vcsErrorf("unexpected response from svn log --xml: %v\n%s", err, out)
        }
 
        t, err := time.Parse(time.RFC3339, log.Logentry.Date)
        if err != nil {
-               return nil, fmt.Errorf("unexpected response from svn log --xml: %v\n%s", err, out)
+               return nil, vcsErrorf("unexpected response from svn log --xml: %v\n%s", err, out)
        }
 
        info := &RevInfo{
@@ -527,23 +538,23 @@ func bzrParseStat(rev, out string) (*RevInfo, error) {
                        }
                        i, err := strconv.ParseInt(val, 10, 64)
                        if err != nil {
-                               return nil, fmt.Errorf("unexpected revno from bzr log: %q", line)
+                               return nil, vcsErrorf("unexpected revno from bzr log: %q", line)
                        }
                        revno = i
                case "timestamp":
                        j := strings.Index(val, " ")
                        if j < 0 {
-                               return nil, fmt.Errorf("unexpected timestamp from bzr log: %q", line)
+                               return nil, vcsErrorf("unexpected timestamp from bzr log: %q", line)
                        }
                        t, err := time.Parse("2006-01-02 15:04:05 -0700", val[j+1:])
                        if err != nil {
-                               return nil, fmt.Errorf("unexpected timestamp from bzr log: %q", line)
+                               return nil, vcsErrorf("unexpected timestamp from bzr log: %q", line)
                        }
                        tm = t.UTC()
                }
        }
        if revno == 0 || tm.IsZero() {
-               return nil, fmt.Errorf("unexpected response from bzr log: %q", out)
+               return nil, vcsErrorf("unexpected response from bzr log: %q", out)
        }
 
        info := &RevInfo{
@@ -560,11 +571,11 @@ func fossilParseStat(rev, out string) (*RevInfo, error) {
                if strings.HasPrefix(line, "uuid:") {
                        f := strings.Fields(line)
                        if len(f) != 5 || len(f[1]) != 40 || f[4] != "UTC" {
-                               return nil, fmt.Errorf("unexpected response from fossil info: %q", line)
+                               return nil, vcsErrorf("unexpected response from fossil info: %q", line)
                        }
                        t, err := time.Parse("2006-01-02 15:04:05", f[2]+" "+f[3])
                        if err != nil {
-                               return nil, fmt.Errorf("unexpected response from fossil info: %q", line)
+                               return nil, vcsErrorf("unexpected response from fossil info: %q", line)
                        }
                        hash := f[1]
                        version := rev
@@ -580,5 +591,5 @@ func fossilParseStat(rev, out string) (*RevInfo, error) {
                        return info, nil
                }
        }
-       return nil, fmt.Errorf("unexpected response from fossil info: %q", out)
+       return nil, vcsErrorf("unexpected response from fossil info: %q", out)
 }
diff --git a/src/cmd/go/testdata/script/mod_get_svn.txt b/src/cmd/go/testdata/script/mod_get_svn.txt
new file mode 100644 (file)
index 0000000..ad96fa1
--- /dev/null
@@ -0,0 +1,21 @@
+[!net] skip
+[!exec:svn] skip
+
+env GO111MODULE=on
+env GOPROXY=direct # obtain llvm.org directory, not via svn.
+
+# Attempting to get a module zip using svn should fail with a reasonable
+# message instead of a panic.
+# TODO(golang.org/issue/26092): Really, it shouldn't fail at all.
+! go get -d llvm.org/llvm/bindings/go/llvm
+stderr 'ReadZip not implemented for svn'
+! go install .
+# TODO(bcmills): The error message here should mention ReadZip.
+stderr 'cannot find module for path llvm.org'
+
+-- go.mod --
+module golang/go/issues/28943/main
+-- main.go --
+package main
+import _ "llvm.org/llvm/bindings/go/llvm"
+func main() {}