|
|
|
// Copyright 2013 gopm authors.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License"): you may
|
|
|
|
// not use this file except in compliance with the License. You may obtain
|
|
|
|
// a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
|
|
// WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
|
|
// License for the specific language governing permissions and limitations
|
|
|
|
// under the License.
|
|
|
|
|
|
|
|
package cmd
|
|
|
|
|
|
|
|
import (
|
|
|
|
"archive/zip"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
|
|
|
"os/user"
|
|
|
|
"path"
|
|
|
|
"path/filepath"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"../doc"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
installGOPATH string // The GOPATH that packages are downloaded to.
|
|
|
|
)
|
|
|
|
|
|
|
|
var CmdGet = &Command{
|
|
|
|
UsageLine: "get [flags] <package(s)>",
|
|
|
|
Short: "download and install packages and dependencies",
|
|
|
|
Long: `
|
|
|
|
Get downloads and installs the packages named by the import paths,
|
|
|
|
along with their dependencies.
|
|
|
|
|
|
|
|
This command works even you haven't installed any version control tool
|
|
|
|
such as git, hg, etc.
|
|
|
|
|
|
|
|
The install flags are:
|
|
|
|
|
|
|
|
-d
|
|
|
|
download without installing package(s).
|
|
|
|
-u
|
|
|
|
force to update pakcage(s).
|
|
|
|
-e
|
|
|
|
download dependencies for example(s).
|
|
|
|
|
|
|
|
The list flags accept a space-separated list of strings.
|
|
|
|
|
|
|
|
For more about specifying packages, see 'go help packages'.
|
|
|
|
`,
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
downloadCache = make(map[string]bool)
|
|
|
|
CmdGet.Run = runGet
|
|
|
|
CmdGet.Flags = map[string]bool{
|
|
|
|
"-d": false,
|
|
|
|
"-u": false,
|
|
|
|
"-e": false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func isStandalone() bool {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
// printGetPrompt prints prompt information to users to
|
|
|
|
// let them know what's going on.
|
|
|
|
func printGetPrompt(flag string) {
|
|
|
|
switch flag {
|
|
|
|
case "-d":
|
|
|
|
doc.ColorLog("[INFO] You enabled download without installing.\n")
|
|
|
|
case "-u":
|
|
|
|
doc.ColorLog("[INFO] You enabled force update.\n")
|
|
|
|
case "-e":
|
|
|
|
doc.ColorLog("[INFO] You enabled download dependencies of example(s).\n")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// checkFlags checks if the flag exists with correct format.
|
|
|
|
func checkFlags(flags map[string]bool, args []string, print func(string)) int {
|
|
|
|
num := 0 // Number of valid flags, use to cut out.
|
|
|
|
for i, f := range args {
|
|
|
|
// Check flag prefix '-'.
|
|
|
|
if !strings.HasPrefix(f, "-") {
|
|
|
|
// Not a flag, finish check process.
|
|
|
|
break
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if it a valid flag.
|
|
|
|
if v, ok := flags[f]; ok {
|
|
|
|
flags[f] = !v
|
|
|
|
if !v {
|
|
|
|
print(f)
|
|
|
|
} else {
|
|
|
|
fmt.Println("DISABLE: " + f)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
doc.ColorLog("[ERRO] Unknown flag: %s.\n", f)
|
|
|
|
return -1
|
|
|
|
}
|
|
|
|
num = i + 1
|
|
|
|
}
|
|
|
|
|
|
|
|
return num
|
|
|
|
}
|
|
|
|
|
|
|
|
func runGet(cmd *Command, args []string) {
|
|
|
|
// Check flags.
|
|
|
|
num := checkFlags(cmd.Flags, args, printGetPrompt)
|
|
|
|
if num == -1 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
args = args[num:]
|
|
|
|
|
|
|
|
// Check length of arguments.
|
|
|
|
if len(args) < 1 {
|
|
|
|
doc.ColorLog("[ERROR] Please list the package that you want to install.\n")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(args) > 0 {
|
|
|
|
var ver string = TRUNK
|
|
|
|
if len(args) == 2 {
|
|
|
|
ver = args[1]
|
|
|
|
}
|
|
|
|
pkg := NewPkg(args[0], ver)
|
|
|
|
if pkg == nil {
|
|
|
|
doc.ColorLog("[ERROR] Unrecognized package %v.\n", args[0])
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if isStandalone() {
|
|
|
|
err := getDirect(pkg)
|
|
|
|
if err != nil {
|
|
|
|
doc.ColorLog("[ERROR] %v\n", err)
|
|
|
|
} else {
|
|
|
|
fmt.Println("done.")
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
fmt.Println("Not implemented.")
|
|
|
|
//getSource(pkgName)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func dirExists(dir string) bool {
|
|
|
|
d, e := os.Stat(dir)
|
|
|
|
switch {
|
|
|
|
case e != nil:
|
|
|
|
return false
|
|
|
|
case !d.IsDir():
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
func fileExists(dir string) bool {
|
|
|
|
info, err := os.Stat(dir)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
return !info.IsDir()
|
|
|
|
}
|
|
|
|
|
|
|
|
func joinPath(paths ...string) string {
|
|
|
|
if len(paths) < 1 {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
res := ""
|
|
|
|
for _, p := range paths {
|
|
|
|
res = path.Join(res, p)
|
|
|
|
}
|
|
|
|
return res
|
|
|
|
}
|
|
|
|
|
|
|
|
func download(url string, localfile string) error {
|
|
|
|
fmt.Println("Downloading", url, "...")
|
|
|
|
resp, err := http.Get(url)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
localdir := filepath.Dir(localfile)
|
|
|
|
if !dirExists(localdir) {
|
|
|
|
err = os.MkdirAll(localdir, 0777)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if !fileExists(localfile) {
|
|
|
|
f, err := os.Create(localfile)
|
|
|
|
if err == nil {
|
|
|
|
_, err = io.Copy(f, resp.Body)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func extractPkg(pkg *Pkg, localfile string, update bool) error {
|
|
|
|
fmt.Println("Extracting package", pkg.Name, "...")
|
|
|
|
|
|
|
|
gopath := os.Getenv("GOPATH")
|
|
|
|
var childDirs []string = strings.Split(pkg.Name, "/")
|
|
|
|
|
|
|
|
if pkg.Ver != TRUNK {
|
|
|
|
childDirs[len(childDirs)-1] = fmt.Sprintf("%v_%v_%v", childDirs[len(childDirs)-1], pkg.Ver, pkg.VerId)
|
|
|
|
}
|
|
|
|
dstDir := joinPath(gopath, "src", joinPath(childDirs...))
|
|
|
|
//fmt.Println(dstDir)
|
|
|
|
var err error
|
|
|
|
if !update {
|
|
|
|
if dirExists(dstDir) {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
err = os.MkdirAll(dstDir, 0777)
|
|
|
|
} else {
|
|
|
|
if dirExists(dstDir) {
|
|
|
|
err = os.Remove(dstDir)
|
|
|
|
} else {
|
|
|
|
err = os.MkdirAll(dstDir, 0777)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if path.Ext(localfile) != ".zip" {
|
|
|
|
return errors.New("Not implemented!")
|
|
|
|
}
|
|
|
|
|
|
|
|
r, err := zip.OpenReader(localfile)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer r.Close()
|
|
|
|
|
|
|
|
for _, f := range r.File {
|
|
|
|
//fmt.Printf("Contents of %s:\n", f.Name)
|
|
|
|
paths := strings.Split(f.Name, "/")[1:]
|
|
|
|
//fmt.Println(paths)
|
|
|
|
if len(paths) < 1 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
if f.FileInfo().IsDir() {
|
|
|
|
childDir := joinPath(dstDir, joinPath(paths...))
|
|
|
|
err = os.MkdirAll(childDir, 0777)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
rc, err := f.Open()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
newF, err := os.Create(path.Join(dstDir, joinPath(paths...)))
|
|
|
|
if err == nil {
|
|
|
|
_, err = io.Copy(newF, rc)
|
|
|
|
}
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
rc.Close()
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func getPackage(pkg *Pkg, url string) error {
|
|
|
|
curUser, err := user.Current()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
reposDir = strings.Replace(reposDir, "~", curUser.HomeDir, -1)
|
|
|
|
localdir := path.Join(reposDir, pkg.Name)
|
|
|
|
localdir, err = filepath.Abs(localdir)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
localfile := path.Join(localdir, pkg.FileName())
|
|
|
|
|
|
|
|
err = download(url, localfile)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return extractPkg(pkg, localfile, false)
|
|
|
|
}
|
|
|
|
|
|
|
|
func getDirect(pkg *Pkg) error {
|
|
|
|
return getPackage(pkg, pkg.Url())
|
|
|
|
}
|
|
|
|
|
|
|
|
/*func getFromSource(pkgName string, ver string, source string) error {
|
|
|
|
urlTempl := "https://%v/%v"
|
|
|
|
//urlTempl := "https://%v/archive/master.zip"
|
|
|
|
url := fmt.Sprintf(urlTempl, source, pkgName)
|
|
|
|
|
|
|
|
return getPackage(pkgName, ver, url)
|
|
|
|
}*/
|