185 lines
3.9 KiB
Go
185 lines
3.9 KiB
Go
// Copyright 2017 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.
|
|
|
|
// +build !plan9
|
|
|
|
package main
|
|
|
|
import (
|
|
"archive/tar"
|
|
"archive/zip"
|
|
"compress/gzip"
|
|
"crypto/sha256"
|
|
"fmt"
|
|
"io"
|
|
"io/ioutil"
|
|
"net/http"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
)
|
|
|
|
const (
|
|
currentVersionURL = "https://golang.org/VERSION?m=text"
|
|
downloadURLPrefix = "https://storage.googleapis.com/golang"
|
|
)
|
|
|
|
// downloadGoVersion downloads and upacks the specific go version to dest/go.
|
|
func downloadGoVersion(version, ops, arch, dest string) error {
|
|
suffix := "tar.gz"
|
|
if ops == "windows" {
|
|
suffix = "zip"
|
|
}
|
|
uri := fmt.Sprintf("%s/%s.%s-%s.%s", downloadURLPrefix, version, ops, arch, suffix)
|
|
|
|
verbosef("Downloading %s", uri)
|
|
|
|
req, err := http.NewRequest("GET", uri, nil)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
req.Header.Add("User-Agent", fmt.Sprintf("golang.org-getgo/%s", version))
|
|
|
|
resp, err := http.DefaultClient.Do(req)
|
|
if err != nil {
|
|
return fmt.Errorf("Downloading Go from %s failed: %v", uri, err)
|
|
}
|
|
if resp.StatusCode > 299 {
|
|
return fmt.Errorf("Downloading Go from %s failed with HTTP status %s", uri, resp.Status)
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
tmpf, err := ioutil.TempFile("", "go")
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer os.Remove(tmpf.Name())
|
|
|
|
h := sha256.New()
|
|
|
|
w := io.MultiWriter(tmpf, h)
|
|
if _, err := io.Copy(w, resp.Body); err != nil {
|
|
return err
|
|
}
|
|
|
|
verbosef("Downloading SHA %s.sha256", uri)
|
|
|
|
sresp, err := http.Get(uri + ".sha256")
|
|
if err != nil {
|
|
return fmt.Errorf("Downloading Go sha256 from %s.sha256 failed: %v", uri, err)
|
|
}
|
|
defer sresp.Body.Close()
|
|
if sresp.StatusCode > 299 {
|
|
return fmt.Errorf("Downloading Go sha256 from %s.sha256 failed with HTTP status %s", uri, sresp.Status)
|
|
}
|
|
|
|
shasum, err := ioutil.ReadAll(sresp.Body)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// Check the shasum.
|
|
sum := fmt.Sprintf("%x", h.Sum(nil))
|
|
if sum != string(shasum) {
|
|
return fmt.Errorf("Shasum mismatch %s vs. %s", sum, string(shasum))
|
|
}
|
|
|
|
unpackFunc := unpackTar
|
|
if ops == "windows" {
|
|
unpackFunc = unpackZip
|
|
}
|
|
if err := unpackFunc(tmpf.Name(), dest); err != nil {
|
|
return fmt.Errorf("Unpacking Go to %s failed: %v", dest, err)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func unpack(dest, name string, fi os.FileInfo, r io.Reader) error {
|
|
if strings.HasPrefix(name, "go/") {
|
|
name = name[len("go/"):]
|
|
}
|
|
|
|
path := filepath.Join(dest, name)
|
|
if fi.IsDir() {
|
|
return os.MkdirAll(path, fi.Mode())
|
|
}
|
|
|
|
f, err := os.OpenFile(path, os.O_CREATE|os.O_TRUNC|os.O_WRONLY, fi.Mode())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer f.Close()
|
|
|
|
_, err = io.Copy(f, r)
|
|
return err
|
|
}
|
|
|
|
func unpackTar(src, dest string) error {
|
|
r, err := os.Open(src)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer r.Close()
|
|
|
|
archive, err := gzip.NewReader(r)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer archive.Close()
|
|
|
|
tarReader := tar.NewReader(archive)
|
|
|
|
for {
|
|
header, err := tarReader.Next()
|
|
if err == io.EOF {
|
|
break
|
|
} else if err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := unpack(dest, header.Name, header.FileInfo(), tarReader); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func unpackZip(src, dest string) error {
|
|
zr, err := zip.OpenReader(src)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, f := range zr.File {
|
|
fr, err := f.Open()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := unpack(dest, f.Name, f.FileInfo(), fr); err != nil {
|
|
return err
|
|
}
|
|
fr.Close()
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func getLatestGoVersion() (string, error) {
|
|
resp, err := http.Get(currentVersionURL)
|
|
if err != nil {
|
|
return "", fmt.Errorf("Getting current Go version failed: %v", err)
|
|
}
|
|
defer resp.Body.Close()
|
|
if resp.StatusCode > 299 {
|
|
b, _ := ioutil.ReadAll(io.LimitReader(resp.Body, 1024))
|
|
return "", fmt.Errorf("Could not get current Go version: HTTP %d: %q", resp.StatusCode, b)
|
|
}
|
|
version, err := ioutil.ReadAll(resp.Body)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
return strings.TrimSpace(string(version)), nil
|
|
}
|