You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

357 lines
9.4 KiB

// Copyright (c) 2013 GPMGo Members. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
12 years ago
package cmd
import (
12 years ago
"encoding/json"
"errors"
"fmt"
"net/http"
12 years ago
"os"
"os/exec"
"regexp"
"strings"
"github.com/GPMGo/gopm/doc"
"github.com/GPMGo/gopm/utils"
"github.com/GPMGo/node"
)
var (
isHasGit, isHasHg bool
downloadCache map[string]bool // Saves packages that have been downloaded.
installGOPATH string // The GOPATH that packages are downloaded to.
)
12 years ago
var CmdInstall = &Command{
UsageLine: "install [install flags] <packages|bundles|snapshots>",
}
func init() {
12 years ago
downloadCache = make(map[string]bool)
12 years ago
CmdInstall.Run = runInstall
CmdInstall.Flags = map[string]bool{
"-v": false,
"-d": false,
"-u": false, // Flag for 'go get'.
"-e": false,
12 years ago
"-s": false,
}
}
// printInstallPrompt prints prompt information to users to
// let them know what's going on.
func printInstallPrompt(flag string) {
switch flag {
case "-v":
12 years ago
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["PureDownload"]))
case "-d":
12 years ago
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["DownloadOnly"]))
case "-e":
12 years ago
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["DownloadExDeps"]))
12 years ago
case "-s":
12 years ago
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["DownloadFromSrcs"]))
}
}
// checkFlags checks if the flag exists with correct format.
func checkFlags(flags map[string]bool, enable []string, args []string, print func(string)) int {
// Check auto-enable.
for _, v := range enable {
flags["-"+v] = true
print("-" + v)
}
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 {
12 years ago
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["UnknownFlag"]), f)
return -1
}
num = i + 1
}
return num
}
// checkVCSTool checks if users have installed version control tools.
func checkVCSTool() {
// git.
if _, err := exec.LookPath("git"); err == nil {
isHasGit = true
}
// hg.
if _, err := exec.LookPath("hg"); err == nil {
isHasHg = true
}
// svn.
}
func runInstall(cmd *Command, args []string) {
// Check flags.
12 years ago
num := checkFlags(cmd.Flags, Config.AutoEnable.Install, args, printInstallPrompt)
if num == -1 {
return
}
args = args[num:]
// Check length of arguments.
if len(args) < 1 {
12 years ago
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["NoPackage"]))
return
}
// Check version control tools.
checkVCSTool()
12 years ago
installGOPATH = utils.GetBestMatchGOPATH(AppPath)
utils.ColorPrint(fmt.Sprintf(fmt.Sprintf("%s\n", PromptMsg["DownloadPath"]), installGOPATH))
// Generate temporary nodes.
nodes := make([]*node.Node, len(args))
for i := range nodes {
nodes[i] = new(node.Node)
nodes[i].ImportPath = args[i]
}
// Download packages.
downloadPackages(nodes)
12 years ago
if !CmdInstall.Flags["-d"] && !CmdInstall.Flags["-v"] {
// Remove old files.
uninstallList := make([]string, 0, len(downloadCache))
for k := range downloadCache {
uninstallList = append(uninstallList, k)
}
removePackageFiles("", uninstallList)
// Install packages all together.
var cmdArgs []string
cmdArgs = append(cmdArgs, "install")
cmdArgs = append(cmdArgs, "<blank>")
for k := range downloadCache {
12 years ago
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["InstallStatus"]), k)
cmdArgs[1] = k
executeCommand("go", cmdArgs)
}
12 years ago
// Save local nodes to file.
12 years ago
fw, err := os.Create(AppPath + "data/nodes.json")
12 years ago
if err != nil {
12 years ago
utils.ColorPrint(fmt.Sprintf(fmt.Sprintf("[ERROR] runInstall -> %s\n", PromptMsg["OpenFile"]), err))
12 years ago
return
}
defer fw.Close()
12 years ago
fbytes, err := json.MarshalIndent(&LocalNodes, "", "\t")
if err != nil {
12 years ago
utils.ColorPrint(fmt.Sprintf(fmt.Sprintf("[ERROR] runInstall -> %s\n", PromptMsg["ParseJSON"]), err))
return
}
12 years ago
fw.Write(fbytes)
}
}
// chekcDeps checks dependencies of nodes.
func chekcDeps(nodes []*node.Node) (depnodes []*node.Node) {
for _, n := range nodes {
// Make sure it will not download all dependencies automatically.
if len(n.Value) == 0 {
n.Value = "B"
}
depnodes = append(depnodes, n)
depnodes = append(depnodes, chekcDeps(n.Deps)...)
}
return depnodes
}
// checkLocalBundles checks if the bundle is in local file system.
func checkLocalBundles(bundle string) (nodes []*node.Node) {
12 years ago
for _, b := range LocalBundles {
if bundle == b.Name {
nodes = append(nodes, chekcDeps(b.Nodes)...)
return nodes
}
}
return nil
}
// downloadPackages downloads packages with certain commit,
// if the commit is empty string, then it downloads all dependencies,
// otherwise, it only downloada package with specific commit only.
func downloadPackages(nodes []*node.Node) {
// Check all packages, they may be bundles, snapshots or raw packages path.
for _, n := range nodes {
// Check if it is a bundle or snapshot.
switch {
case strings.HasSuffix(n.ImportPath, ".b"):
l := len(n.ImportPath)
// Check local bundles.
bnodes := checkLocalBundles(n.ImportPath[:l-2])
if len(bnodes) > 0 {
// Check with users if continue.
12 years ago
utils.ColorPrint(fmt.Sprintf(fmt.Sprintf("%s\n", PromptMsg["BundleInfo"]), n.ImportPath[:l-2]))
for _, bn := range bnodes {
fmt.Printf("[%s] -> %s: %s.\n", bn.ImportPath, bn.Type, bn.Value)
}
12 years ago
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["ContinueDownload"]))
var option string
fmt.Fscan(os.Stdin, &option)
if strings.ToLower(option) != "y" {
os.Exit(0)
}
downloadPackages(bnodes)
} else {
// Check from server.
// TODO: api.GetBundleInfo()
fmt.Println("Unable to find bundle, and we cannot check with server right now.")
}
case strings.HasSuffix(n.ImportPath, ".s"):
// TODO: api.GetSnapshotInfo()
case utils.IsValidRemotePath(n.ImportPath):
if !downloadCache[n.ImportPath] {
// Download package.
nod, imports := downloadPackage(n)
if len(imports) > 0 {
// Need to download dependencies.
// Generate temporary nodes.
nodes := make([]*node.Node, len(imports))
for i := range nodes {
nodes[i] = new(node.Node)
nodes[i].ImportPath = imports[i]
}
downloadPackages(nodes)
}
// Only save package information with specific commit.
if nod != nil {
12 years ago
// Save record in local nodes.
saveNode(nod)
}
} else {
12 years ago
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["SkipDownloaded"]), n.ImportPath)
}
default:
// Invalid import path.
12 years ago
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["SkipInvalidPath"]), n.ImportPath)
}
}
}
12 years ago
// saveNode saves node into local nodes.
func saveNode(n *node.Node) {
12 years ago
// Node dependencies list.
n.Deps = nil
12 years ago
// Check if this node exists.
12 years ago
for i, v := range LocalNodes {
12 years ago
if n.ImportPath == v.ImportPath {
12 years ago
LocalNodes[i] = n
12 years ago
return
}
}
// Add new node.
12 years ago
LocalNodes = append(LocalNodes, n)
12 years ago
}
// downloadPackage downloads package either use version control tools or not.
func downloadPackage(nod *node.Node) (*node.Node, []string) {
// Check if use version control tools.
switch {
12 years ago
case CmdInstall.Flags["-v"] &&
((nod.ImportPath[0] == 'g' && isHasGit) || (nod.ImportPath[0] == 'c' && isHasHg)): // github.com, code.google.com
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["InstallByGoGet"]), nod.ImportPath)
args := checkGoGetFlags()
args = append(args, nod.ImportPath)
executeCommand("go", args)
return nil, nil
default: // Pure download.
12 years ago
if CmdInstall.Flags["-v"] {
CmdInstall.Flags["-v"] = false
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["NoVCSTool"]))
}
fmt.Printf(fmt.Sprintf("%s\n", PromptMsg["DownloadStatus"]), nod.ImportPath)
// Mark as donwloaded.
downloadCache[nod.ImportPath] = true
imports, err := pureDownload(nod)
12 years ago
if err != nil {
utils.ColorPrint(fmt.Sprintf(fmt.Sprintf("[ERROR] %s\n", PromptMsg["DownloadError"]), nod.ImportPath, err))
return nil, nil
}
return nod, imports
}
}
func checkGoGetFlags() (args []string) {
args = append(args, "get")
switch {
12 years ago
case CmdInstall.Flags["-d"]:
args = append(args, "-d")
fallthrough
12 years ago
case CmdInstall.Flags["-u"]:
args = append(args, "-u")
}
return args
}
// service represents a source code control service.
type service struct {
pattern *regexp.Regexp
prefix string
get func(*http.Client, map[string]string, string, *node.Node, map[string]bool) ([]string, error)
}
// services is the list of source code control services handled by gopkgdoc.
var services = []*service{
{doc.GithubPattern, "github.com/", doc.GetGithubDoc},
{doc.GooglePattern, "code.google.com/", doc.GetGoogleDoc},
{doc.BitbucketPattern, "bitbucket.org/", doc.GetBitbucketDoc},
{doc.LaunchpadPattern, "launchpad.net/", doc.GetLaunchpadDoc},
}
// pureDownload downloads package without version control.
func pureDownload(nod *node.Node) ([]string, error) {
for _, s := range services {
if s.get == nil || !strings.HasPrefix(nod.ImportPath, s.prefix) {
continue
}
m := s.pattern.FindStringSubmatch(nod.ImportPath)
if m == nil {
if s.prefix != "" {
return nil,
12 years ago
doc.NotFoundError{fmt.Sprintf("%s", PromptMsg["NotFoundError"])}
}
continue
}
match := map[string]string{"importPath": nod.ImportPath}
for i, n := range s.pattern.SubexpNames() {
if n != "" {
match[n] = m[i]
}
}
return s.get(doc.HttpClient, match, installGOPATH, nod, CmdInstall.Flags)
}
12 years ago
return nil, errors.New(fmt.Sprintf("%s", PromptMsg["NotFoundError"]))
}