// Copyright 2018 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 (
|
"archive/zip"
|
"bytes"
|
"encoding/json"
|
"flag"
|
"fmt"
|
"io/ioutil"
|
"log"
|
"net"
|
"net/http"
|
"os"
|
"path/filepath"
|
"strings"
|
"sync"
|
"testing"
|
|
"cmd/go/internal/modfetch"
|
"cmd/go/internal/modfetch/codehost"
|
"cmd/go/internal/module"
|
"cmd/go/internal/par"
|
"cmd/go/internal/semver"
|
"cmd/go/internal/txtar"
|
)
|
|
var (
|
proxyAddr = flag.String("proxy", "", "run proxy on this network address instead of running any tests")
|
proxyURL string
|
)
|
|
var proxyOnce sync.Once
|
|
// StartProxy starts the Go module proxy running on *proxyAddr (like "localhost:1234")
|
// and sets proxyURL to the GOPROXY setting to use to access the proxy.
|
// Subsequent calls are no-ops.
|
//
|
// The proxy serves from testdata/mod. See testdata/mod/README.
|
func StartProxy() {
|
proxyOnce.Do(func() {
|
fmt.Fprintf(os.Stderr, "go test proxy starting\n")
|
readModList()
|
addr := *proxyAddr
|
if addr == "" {
|
addr = "localhost:0"
|
}
|
l, err := net.Listen("tcp", addr)
|
if err != nil {
|
log.Fatal(err)
|
}
|
*proxyAddr = l.Addr().String()
|
proxyURL = "http://" + *proxyAddr + "/mod"
|
fmt.Fprintf(os.Stderr, "go test proxy running at GOPROXY=%s\n", proxyURL)
|
go func() {
|
log.Fatalf("go proxy: http.Serve: %v", http.Serve(l, http.HandlerFunc(proxyHandler)))
|
}()
|
})
|
}
|
|
var modList []module.Version
|
|
func readModList() {
|
infos, err := ioutil.ReadDir("testdata/mod")
|
if err != nil {
|
log.Fatal(err)
|
}
|
for _, info := range infos {
|
name := info.Name()
|
if !strings.HasSuffix(name, ".txt") {
|
continue
|
}
|
name = strings.TrimSuffix(name, ".txt")
|
i := strings.LastIndex(name, "_v")
|
if i < 0 {
|
continue
|
}
|
encPath := strings.ReplaceAll(name[:i], "_", "/")
|
path, err := module.DecodePath(encPath)
|
if err != nil {
|
fmt.Fprintf(os.Stderr, "go proxy_test: %v\n", err)
|
continue
|
}
|
encVers := name[i+1:]
|
vers, err := module.DecodeVersion(encVers)
|
if err != nil {
|
fmt.Fprintf(os.Stderr, "go proxy_test: %v\n", err)
|
continue
|
}
|
modList = append(modList, module.Version{Path: path, Version: vers})
|
}
|
}
|
|
var zipCache par.Cache
|
|
// proxyHandler serves the Go module proxy protocol.
|
// See the proxy section of https://research.swtch.com/vgo-module.
|
func proxyHandler(w http.ResponseWriter, r *http.Request) {
|
if !strings.HasPrefix(r.URL.Path, "/mod/") {
|
http.NotFound(w, r)
|
return
|
}
|
path := strings.TrimPrefix(r.URL.Path, "/mod/")
|
i := strings.Index(path, "/@v/")
|
if i < 0 {
|
http.NotFound(w, r)
|
return
|
}
|
enc, file := path[:i], path[i+len("/@v/"):]
|
path, err := module.DecodePath(enc)
|
if err != nil {
|
fmt.Fprintf(os.Stderr, "go proxy_test: %v\n", err)
|
http.NotFound(w, r)
|
return
|
}
|
if file == "list" {
|
n := 0
|
for _, m := range modList {
|
if m.Path == path && !modfetch.IsPseudoVersion(m.Version) {
|
if err := module.Check(m.Path, m.Version); err == nil {
|
fmt.Fprintf(w, "%s\n", m.Version)
|
n++
|
}
|
}
|
}
|
if n == 0 {
|
http.NotFound(w, r)
|
}
|
return
|
}
|
|
i = strings.LastIndex(file, ".")
|
if i < 0 {
|
http.NotFound(w, r)
|
return
|
}
|
encVers, ext := file[:i], file[i+1:]
|
vers, err := module.DecodeVersion(encVers)
|
if err != nil {
|
fmt.Fprintf(os.Stderr, "go proxy_test: %v\n", err)
|
http.NotFound(w, r)
|
return
|
}
|
|
if codehost.AllHex(vers) {
|
var best string
|
// Convert commit hash (only) to known version.
|
// Use latest version in semver priority, to match similar logic
|
// in the repo-based module server (see modfetch.(*codeRepo).convert).
|
for _, m := range modList {
|
if m.Path == path && semver.Compare(best, m.Version) < 0 {
|
var hash string
|
if modfetch.IsPseudoVersion(m.Version) {
|
hash = m.Version[strings.LastIndex(m.Version, "-")+1:]
|
} else {
|
hash = findHash(m)
|
}
|
if strings.HasPrefix(hash, vers) || strings.HasPrefix(vers, hash) {
|
best = m.Version
|
}
|
}
|
}
|
if best != "" {
|
vers = best
|
}
|
}
|
|
a := readArchive(path, vers)
|
if a == nil {
|
fmt.Fprintf(os.Stderr, "go proxy: no archive %s %s\n", path, vers)
|
http.Error(w, "cannot load archive", 500)
|
return
|
}
|
|
switch ext {
|
case "info", "mod":
|
want := "." + ext
|
for _, f := range a.Files {
|
if f.Name == want {
|
w.Write(f.Data)
|
return
|
}
|
}
|
|
case "zip":
|
type cached struct {
|
zip []byte
|
err error
|
}
|
c := zipCache.Do(a, func() interface{} {
|
var buf bytes.Buffer
|
z := zip.NewWriter(&buf)
|
for _, f := range a.Files {
|
if strings.HasPrefix(f.Name, ".") {
|
continue
|
}
|
var zipName string
|
if strings.HasPrefix(f.Name, "/") {
|
zipName = f.Name[1:]
|
} else {
|
zipName = path + "@" + vers + "/" + f.Name
|
}
|
zf, err := z.Create(zipName)
|
if err != nil {
|
return cached{nil, err}
|
}
|
if _, err := zf.Write(f.Data); err != nil {
|
return cached{nil, err}
|
}
|
}
|
if err := z.Close(); err != nil {
|
return cached{nil, err}
|
}
|
return cached{buf.Bytes(), nil}
|
}).(cached)
|
|
if c.err != nil {
|
fmt.Fprintf(os.Stderr, "go proxy: %v\n", c.err)
|
http.Error(w, c.err.Error(), 500)
|
return
|
}
|
w.Write(c.zip)
|
return
|
|
}
|
http.NotFound(w, r)
|
}
|
|
func findHash(m module.Version) string {
|
a := readArchive(m.Path, m.Version)
|
if a == nil {
|
return ""
|
}
|
var data []byte
|
for _, f := range a.Files {
|
if f.Name == ".info" {
|
data = f.Data
|
break
|
}
|
}
|
var info struct{ Short string }
|
json.Unmarshal(data, &info)
|
return info.Short
|
}
|
|
var archiveCache par.Cache
|
|
var cmdGoDir, _ = os.Getwd()
|
|
func readArchive(path, vers string) *txtar.Archive {
|
enc, err := module.EncodePath(path)
|
if err != nil {
|
fmt.Fprintf(os.Stderr, "go proxy: %v\n", err)
|
return nil
|
}
|
encVers, err := module.EncodeVersion(vers)
|
if err != nil {
|
fmt.Fprintf(os.Stderr, "go proxy: %v\n", err)
|
return nil
|
}
|
|
prefix := strings.ReplaceAll(enc, "/", "_")
|
name := filepath.Join(cmdGoDir, "testdata/mod", prefix+"_"+encVers+".txt")
|
a := archiveCache.Do(name, func() interface{} {
|
a, err := txtar.ParseFile(name)
|
if err != nil {
|
if testing.Verbose() || !os.IsNotExist(err) {
|
fmt.Fprintf(os.Stderr, "go proxy: %v\n", err)
|
}
|
a = nil
|
}
|
return a
|
}).(*txtar.Archive)
|
return a
|
}
|