Browse Source

command install add support for use tag, branch or commit for packages in github.com

pull/103/head
Unknown 12 years ago
parent
commit
2c429375fb
  1. 6
      README.md
  2. 30
      doc/bitbucket.go
  3. 49
      doc/github.go
  4. 33
      doc/google.go
  5. 27
      doc/launchpad.go
  6. 6
      doc/struct.go
  7. 3
      gpm.go
  8. 110
      install.go

6
README.md

@ -1,7 +1,7 @@
gpm - Go Package Manager gpm - Go Package Manager
=== ===
![GPMGo_Logo](https://raw.github.com/GPMGo/gpm-site/master/static/img/gpmgo.png?raw=true) ![GPMGo_Logo](https://raw.github.com/GPMGo/gpm-site/master/static/img/gpmgo2.png?raw=true)
gpm(Go Package Manager) is a Go package manage tool for search, install, update and share packages in Go. gpm(Go Package Manager) is a Go package manage tool for search, install, update and share packages in Go.
@ -22,7 +22,7 @@ gpm(Go Package Manager) is a Go package manage tool for search, install, update
- Add bundle and snapshot parser code for downloading by bundle or snapshot id. - Add bundle and snapshot parser code for downloading by bundle or snapshot id.
- Add user system to create, edit, upload, and download bundles or snapshots through gpm client program. - Add user system to create, edit, upload, and download bundles or snapshots through gpm client program.
- Download package from code.google.com only support hg as version control system, probably support git and svn. - Download package from code.google.com only support hg as version control system, probably support git and svn.
- Add support for downloading by tag for packages in github.com, bitbucket.org, git.oschina.net, gitcafe.com. - Add support for downloading by tag and branch for packages in bitbucket.org, git.oschina.net, gitcafe.com.
- Get author commit time and save in node. - Get author commit time and save in node.
- Collect download and installation results and report to users in the end. - Collect download and installation results and report to users in the end.
- Command `install` add support for downloading code from git.oschina.net, gitcafe.com, *.codeplex.com; - Command `install` add support for downloading code from git.oschina.net, gitcafe.com, *.codeplex.com;
@ -30,3 +30,5 @@ gpm(Go Package Manager) is a Go package manage tool for search, install, update
- Command `daemon` is for auto-compile web applications when debug it locally. - Command `daemon` is for auto-compile web applications when debug it locally.
- Command `update` is for checking updates. - Command `update` is for checking updates.
- Command `remove` is for removing packages. - Command `remove` is for removing packages.
- Command `search` is for searching packages.
- Add feature "struct generator".

30
doc/bitbucket.go

@ -22,7 +22,8 @@ var (
) )
// GetBitbucketDoc downloads tarball from bitbucket.org. // GetBitbucketDoc downloads tarball from bitbucket.org.
func GetBitbucketDoc(client *http.Client, match map[string]string, installGOPATH, commit string, cmdFlags map[string]bool) (*Node, []string, error) { func GetBitbucketDoc(client *http.Client, match map[string]string, installGOPATH string, node *Node, cmdFlags map[string]bool) ([]string, error) {
commit := node.Value
// Check version control. // Check version control.
if m := bitbucketEtagRe.FindStringSubmatch(commit); m != nil { if m := bitbucketEtagRe.FindStringSubmatch(commit); m != nil {
match["vcs"] = m[1] match["vcs"] = m[1]
@ -31,7 +32,7 @@ func GetBitbucketDoc(client *http.Client, match map[string]string, installGOPATH
Scm string Scm string
} }
if err := httpGetJSON(client, expand("https://api.bitbucket.org/1.0/repositories/{owner}/{repo}", match), &repo); err != nil { if err := httpGetJSON(client, expand("https://api.bitbucket.org/1.0/repositories/{owner}/{repo}", match), &repo); err != nil {
return nil, nil, err return nil, err
} }
match["vcs"] = repo.Scm match["vcs"] = repo.Scm
} }
@ -48,7 +49,7 @@ func GetBitbucketDoc(client *http.Client, match map[string]string, installGOPATH
Node string Node string
} }
if err := httpGetJSON(client, expand("https://api.bitbucket.org/1.0/repositories/{owner}/{repo}/{0}", match, nodeType), &nodes); err != nil { if err := httpGetJSON(client, expand("https://api.bitbucket.org/1.0/repositories/{owner}/{repo}/{0}", match, nodeType), &nodes); err != nil {
return nil, nil, err return nil, err
} }
for t, n := range nodes { for t, n := range nodes {
tags[t] = n.Node tags[t] = n.Node
@ -59,7 +60,7 @@ func GetBitbucketDoc(client *http.Client, match map[string]string, installGOPATH
var err error var err error
match["tag"], match["commit"], err = bestTag(tags, defaultTags[match["vcs"]]) match["tag"], match["commit"], err = bestTag(tags, defaultTags[match["vcs"]])
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
} else { } else {
match["commit"] = commit match["commit"] = commit
@ -72,7 +73,7 @@ func GetBitbucketDoc(client *http.Client, match map[string]string, installGOPATH
// Downlaod archive. // Downlaod archive.
p, err := httpGetBytes(client, expand("https://bitbucket.org/{owner}/{repo}/get/{commit}.tar.gz", match), nil) p, err := httpGetBytes(client, expand("https://bitbucket.org/{owner}/{repo}/get/{commit}.tar.gz", match), nil)
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
projectPath := expand("bitbucket.org/{owner}/{repo}", match) projectPath := expand("bitbucket.org/{owner}/{repo}", match)
@ -85,7 +86,7 @@ func GetBitbucketDoc(client *http.Client, match map[string]string, installGOPATH
gzr, err := gzip.NewReader(bytes.NewReader(p)) gzr, err := gzip.NewReader(bytes.NewReader(p))
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
defer gzr.Close() defer gzr.Close()
@ -99,7 +100,7 @@ func GetBitbucketDoc(client *http.Client, match map[string]string, installGOPATH
if err == io.EOF { if err == io.EOF {
break break
} else if err != nil { } else if err != nil {
return nil, nil, err return nil, err
} }
fn := h.FileInfo().Name() fn := h.FileInfo().Name()
@ -125,26 +126,27 @@ func GetBitbucketDoc(client *http.Client, match map[string]string, installGOPATH
// Get data from archive. // Get data from archive.
fbytes := make([]byte, h.Size) fbytes := make([]byte, h.Size)
if _, err := io.ReadFull(tr, fbytes); err != nil { if _, err := io.ReadFull(tr, fbytes); err != nil {
return nil, nil, err return nil, err
} }
// Write data to file // Write data to file
fw, err := os.Create(absPath) fw, err := os.Create(absPath)
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
_, err = fw.Write(fbytes) _, err = fw.Write(fbytes)
fw.Close() fw.Close()
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
} }
node := &Node{ node.Value = commit
/* node := &Node{
ImportPath: projectPath, ImportPath: projectPath,
Commit: commit, Commit: commit,
} }*/
var imports []string var imports []string
@ -153,13 +155,13 @@ func GetBitbucketDoc(client *http.Client, match map[string]string, installGOPATH
for _, d := range dirs { for _, d := range dirs {
importPkgs, err := checkImports(d+"/", match["importPath"]) importPkgs, err := checkImports(d+"/", match["importPath"])
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
imports = append(imports, importPkgs...) imports = append(imports, importPkgs...)
} }
} }
return node, imports, err return imports, err
} }
// checkDir checks if current directory has been saved. // checkDir checks if current directory has been saved.

49
doc/github.go

@ -7,6 +7,7 @@ package doc
import ( import (
"archive/zip" "archive/zip"
"bytes" "bytes"
"errors"
"io" "io"
"net/http" "net/http"
"os" "os"
@ -26,7 +27,7 @@ func SetGithubCredentials(id, secret string) {
} }
// GetGithubDoc downloads tarball from github.com. // GetGithubDoc downloads tarball from github.com.
func GetGithubDoc(client *http.Client, match map[string]string, installGOPATH, commit string, cmdFlags map[string]bool) (*Node, []string, error) { func GetGithubDoc(client *http.Client, match map[string]string, installGOPATH string, node *Node, cmdFlags map[string]bool) ([]string, error) {
SetGithubCredentials("1862bcb265171f37f36c", "308d71ab53ccd858416cfceaed52d5d5b7d53c5f") SetGithubCredentials("1862bcb265171f37f36c", "308d71ab53ccd858416cfceaed52d5d5b7d53c5f")
match["cred"] = githubCred match["cred"] = githubCred
@ -43,14 +44,14 @@ func GetGithubDoc(client *http.Client, match map[string]string, installGOPATH, c
// bundle and snapshot will have commit 'B' and 'S', // bundle and snapshot will have commit 'B' and 'S',
// but does not need to download dependencies. // but does not need to download dependencies.
isCheckImport := len(commit) == 0 isCheckImport := len(node.Value) == 0
// Check if download with specific revision. switch {
if isCheckImport || len(commit) == 1 { case isCheckImport || len(node.Value) == 1:
// Get up-to-date version. // Get up-to-date version.
err := httpGetJSON(client, expand("https://api.github.com/repos/{owner}/{repo}/git/refs?{cred}", match), &refs) err := httpGetJSON(client, expand("https://api.github.com/repos/{owner}/{repo}/git/refs?{cred}", match), &refs)
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
tags := make(map[string]string) tags := make(map[string]string)
@ -64,9 +65,19 @@ func GetGithubDoc(client *http.Client, match map[string]string, installGOPATH, c
} }
// Check revision tag. // Check revision tag.
match["tag"], commit, err = bestTag(tags, "master") match["tag"], match["sha"], err = bestTag(tags, "master")
if err != nil { if err != nil {
return nil, nil, err return nil, err
}
node.Type = "commit"
case !isCheckImport: // Bundle or snapshot.
// Check downlaod type.
switch node.Type {
case "tag", "commit", "branch":
match["sha"] = node.Value
default:
return nil, errors.New("Unknown node type: " + node.Type)
} }
} }
@ -74,19 +85,22 @@ func GetGithubDoc(client *http.Client, match map[string]string, installGOPATH, c
// zip : https://github.com/{owner}/{repo}/archive/{sha}.zip // zip : https://github.com/{owner}/{repo}/archive/{sha}.zip
// tarball : https://github.com/{owner}/{repo}/tarball/{sha} // tarball : https://github.com/{owner}/{repo}/tarball/{sha}
match["sha"] = commit
// Downlaod archive. // Downlaod archive.
p, err := httpGetBytes(client, expand("https://github.com/{owner}/{repo}/archive/{sha}.zip", match), nil) p, err := httpGetBytes(client, expand("https://github.com/{owner}/{repo}/archive/{sha}.zip", match), nil)
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
r, err := zip.NewReader(bytes.NewReader(p), int64(len(p))) r, err := zip.NewReader(bytes.NewReader(p), int64(len(p)))
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
shaName := expand("{repo}-{sha}", match) shaName := expand("{repo}-{sha}", match)
if node.Type == "tag" {
shaName = strings.Replace(shaName, "-v", "-", 1)
}
projectPath := expand("github.com/{owner}/{repo}", match) projectPath := expand("github.com/{owner}/{repo}", match)
installPath := installGOPATH + "/src/" + projectPath installPath := installGOPATH + "/src/" + projectPath
@ -114,7 +128,7 @@ func GetGithubDoc(client *http.Client, match map[string]string, installGOPATH, c
// Get file from archive. // Get file from archive.
rc, err := f.Open() rc, err := f.Open()
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
// Create diretory before create file. // Create diretory before create file.
@ -123,20 +137,19 @@ func GetGithubDoc(client *http.Client, match map[string]string, installGOPATH, c
// Write data to file // Write data to file
fw, _ := os.Create(absPath) fw, _ := os.Create(absPath)
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
_, err = io.Copy(fw, rc) _, err = io.Copy(fw, rc)
rc.Close() rc.Close()
fw.Close() fw.Close()
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
} }
node := &Node{ if node.Type == "commit" {
ImportPath: projectPath, node.Value = match["sha"]
Commit: commit,
} }
var imports []string var imports []string
@ -146,11 +159,11 @@ func GetGithubDoc(client *http.Client, match map[string]string, installGOPATH, c
for _, d := range dirs { for _, d := range dirs {
importPkgs, err := checkImports(d, match["importPath"]) importPkgs, err := checkImports(d, match["importPath"])
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
imports = append(imports, importPkgs...) imports = append(imports, importPkgs...)
} }
} }
return node, imports, err return imports, err
} }

33
doc/google.go

@ -48,13 +48,14 @@ func getGoogleVCS(client *http.Client, match map[string]string) error {
} }
// GetGoogleDoc downloads raw files from code.google.com. // GetGoogleDoc downloads raw files from code.google.com.
func GetGoogleDoc(client *http.Client, match map[string]string, installGOPATH, commit string, cmdFlags map[string]bool) (*Node, []string, error) { func GetGoogleDoc(client *http.Client, match map[string]string, installGOPATH string, node *Node, cmdFlags map[string]bool) ([]string, error) {
setupGoogleMatch(match) setupGoogleMatch(match)
commit := node.Value
// Check version control. // Check version control.
if m := googleEtagRe.FindStringSubmatch(commit); m != nil { if m := googleEtagRe.FindStringSubmatch(commit); m != nil {
match["vcs"] = m[1] match["vcs"] = m[1]
} else if err := getGoogleVCS(client, match); err != nil { } else if err := getGoogleVCS(client, match); err != nil {
return nil, nil, err return nil, err
} }
// bundle and snapshot will have commit 'B' and 'S', // bundle and snapshot will have commit 'B' and 'S',
@ -66,12 +67,12 @@ func GetGoogleDoc(client *http.Client, match map[string]string, installGOPATH, c
// Scrape the repo browser to find the project revision and individual Go files. // Scrape the repo browser to find the project revision and individual Go files.
p, err := httpGetBytes(client, rootPath+"?r="+commit, nil) p, err := httpGetBytes(client, rootPath+"?r="+commit, nil)
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
// Check revision tag. // Check revision tag.
if m := googleRevisionRe.FindSubmatch(p); m == nil { if m := googleRevisionRe.FindSubmatch(p); m == nil {
return nil, nil, return nil,
errors.New("doc.GetGoogleDoc(): Could not find revision for " + match["importPath"]) errors.New("doc.GetGoogleDoc(): Could not find revision for " + match["importPath"])
} }
@ -86,7 +87,7 @@ func GetGoogleDoc(client *http.Client, match map[string]string, installGOPATH, c
// Get source files in root path. // Get source files in root path.
files := make([]*source, 0, 5) files := make([]*source, 0, 5)
for _, m := range googleFileRe.FindAllSubmatch(p, -1) { for _, m := range googleFileRe.FindAllSubmatch(p, -1) {
fname := string(m[1]) fname := strings.Split(string(m[1]), "?")[0]
files = append(files, &source{ files = append(files, &source{
name: fname, name: fname,
rawURL: expand("http://{subrepo}{dot}{repo}.googlecode.com/{vcs}{dir}/{0}", match, fname) + "?r=" + commit, rawURL: expand("http://{subrepo}{dot}{repo}.googlecode.com/{vcs}{dir}/{0}", match, fname) + "?r=" + commit,
@ -95,7 +96,7 @@ func GetGoogleDoc(client *http.Client, match map[string]string, installGOPATH, c
// Fetch files from VCS. // Fetch files from VCS.
if err := fetchFiles(client, files, nil); err != nil { if err := fetchFiles(client, files, nil); err != nil {
return nil, nil, err return nil, err
} }
// Save files. // Save files.
@ -108,13 +109,13 @@ func GetGoogleDoc(client *http.Client, match map[string]string, installGOPATH, c
// Write data to file // Write data to file
fw, err := os.Create(absPath + f.name) fw, err := os.Create(absPath + f.name)
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
_, err = fw.Write(f.data) _, err = fw.Write(f.data)
fw.Close() fw.Close()
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
} }
@ -129,26 +130,28 @@ func GetGoogleDoc(client *http.Client, match map[string]string, installGOPATH, c
err = downloadFiles(client, match, rootPath, installPath+"/", commit, dirs) err = downloadFiles(client, match, rootPath, installPath+"/", commit, dirs)
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
node := &Node{ node.Value = commit
/* node := &Node{
ImportPath: projectPath, ImportPath: projectPath,
Commit: commit, Commit: commit,
} }*/
var imports []string var imports []string
// Check if need to check imports. // Check if need to check imports.
if isCheckImport { if isCheckImport {
rootdir, err := os.Open(installPath + "/") rootdir, err := os.Open(installPath + "/")
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
defer rootdir.Close() defer rootdir.Close()
dirs, err := rootdir.Readdir(0) dirs, err := rootdir.Readdir(0)
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
for _, d := range dirs { for _, d := range dirs {
@ -156,14 +159,14 @@ func GetGoogleDoc(client *http.Client, match map[string]string, installGOPATH, c
absPath := installPath + "/" + d.Name() + "/" absPath := installPath + "/" + d.Name() + "/"
importPkgs, err := checkImports(absPath, match["importPath"]) importPkgs, err := checkImports(absPath, match["importPath"])
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
imports = append(imports, importPkgs...) imports = append(imports, importPkgs...)
} }
} }
} }
return node, imports, err return imports, err
} }
func downloadFiles(client *http.Client, match map[string]string, rootPath, installPath, commit string, dirs []string) error { func downloadFiles(client *http.Client, match map[string]string, rootPath, installPath, commit string, dirs []string) error {

27
doc/launchpad.go

@ -19,7 +19,7 @@ import (
var LaunchpadPattern = regexp.MustCompile(`^launchpad\.net/(?P<repo>(?P<project>[a-z0-9A-Z_.\-]+)(?P<series>/[a-z0-9A-Z_.\-]+)?|~[a-z0-9A-Z_.\-]+/(\+junk|[a-z0-9A-Z_.\-]+)/[a-z0-9A-Z_.\-]+)(?P<dir>/[a-z0-9A-Z_.\-/]+)*$`) var LaunchpadPattern = regexp.MustCompile(`^launchpad\.net/(?P<repo>(?P<project>[a-z0-9A-Z_.\-]+)(?P<series>/[a-z0-9A-Z_.\-]+)?|~[a-z0-9A-Z_.\-]+/(\+junk|[a-z0-9A-Z_.\-]+)/[a-z0-9A-Z_.\-]+)(?P<dir>/[a-z0-9A-Z_.\-/]+)*$`)
// GetLaunchpadDoc downloads tarball from launchpad.net. // GetLaunchpadDoc downloads tarball from launchpad.net.
func GetLaunchpadDoc(client *http.Client, match map[string]string, installGOPATH, commit string, cmdFlags map[string]bool) (*Node, []string, error) { func GetLaunchpadDoc(client *http.Client, match map[string]string, installGOPATH string, node *Node, cmdFlags map[string]bool) ([]string, error) {
if match["project"] != "" && match["series"] != "" { if match["project"] != "" && match["series"] != "" {
rc, err := httpGet(client, expand("https://code.launchpad.net/{project}{series}/.bzr/branch-format", match), nil) rc, err := httpGet(client, expand("https://code.launchpad.net/{project}{series}/.bzr/branch-format", match), nil)
@ -32,10 +32,10 @@ func GetLaunchpadDoc(client *http.Client, match map[string]string, installGOPATH
match["repo"] = match["project"] match["repo"] = match["project"]
match["dir"] = expand("{series}{dir}", match) match["dir"] = expand("{series}{dir}", match)
default: default:
return nil, nil, err return nil, err
} }
} }
commit := node.Value
// bundle and snapshot will have commit 'B' and 'S', // bundle and snapshot will have commit 'B' and 'S',
// but does not need to download dependencies. // but does not need to download dependencies.
isCheckImport := len(commit) == 0 isCheckImport := len(commit) == 0
@ -51,7 +51,7 @@ func GetLaunchpadDoc(client *http.Client, match map[string]string, installGOPATH
// Scrape the repo browser to find the project revision and individual Go files. // Scrape the repo browser to find the project revision and individual Go files.
p, err := httpGetBytes(client, downloadPath, nil) p, err := httpGetBytes(client, downloadPath, nil)
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
projectPath := expand("launchpad.net/{repo}", match) projectPath := expand("launchpad.net/{repo}", match)
@ -64,7 +64,7 @@ func GetLaunchpadDoc(client *http.Client, match map[string]string, installGOPATH
gzr, err := gzip.NewReader(bytes.NewReader(p)) gzr, err := gzip.NewReader(bytes.NewReader(p))
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
defer gzr.Close() defer gzr.Close()
@ -78,7 +78,7 @@ func GetLaunchpadDoc(client *http.Client, match map[string]string, installGOPATH
if err == io.EOF { if err == io.EOF {
break break
} else if err != nil { } else if err != nil {
return nil, nil, err return nil, err
} }
fn := h.FileInfo().Name() fn := h.FileInfo().Name()
@ -104,26 +104,27 @@ func GetLaunchpadDoc(client *http.Client, match map[string]string, installGOPATH
// Get data from archive. // Get data from archive.
fbytes := make([]byte, h.Size) fbytes := make([]byte, h.Size)
if _, err := io.ReadFull(tr, fbytes); err != nil { if _, err := io.ReadFull(tr, fbytes); err != nil {
return nil, nil, err return nil, err
} }
// Write data to file // Write data to file
fw, err := os.Create(absPath) fw, err := os.Create(absPath)
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
_, err = fw.Write(fbytes) _, err = fw.Write(fbytes)
fw.Close() fw.Close()
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
} }
node := &Node{ node.Value = commit
/* node := &Node{
ImportPath: projectPath, ImportPath: projectPath,
Commit: commit, Commit: commit,
} }*/
var imports []string var imports []string
@ -132,11 +133,11 @@ func GetLaunchpadDoc(client *http.Client, match map[string]string, installGOPATH
for _, d := range dirs { for _, d := range dirs {
importPkgs, err := checkImports(d+"/", match["importPath"]) importPkgs, err := checkImports(d+"/", match["importPath"])
if err != nil { if err != nil {
return nil, nil, err return nil, err
} }
imports = append(imports, importPkgs...) imports = append(imports, importPkgs...)
} }
} }
return node, imports, err return imports, err
} }

6
doc/struct.go

@ -12,9 +12,9 @@ import (
// Node represents a node. // Node represents a node.
type Node struct { type Node struct {
ImportPath string `json:"import_path"` ImportPath string `json:"import_path"`
Commit string Type, Value string
Date string Deps []*Node `json:"-"` // Dependencies.
} }
// Bundle represents a bundle. // Bundle represents a bundle.

3
gpm.go

@ -183,7 +183,7 @@ func loadLocalBundles() bool {
for _, fi := range fis { for _, fi := range fis {
// In case this folder contains unexpected directories. // In case this folder contains unexpected directories.
if !fi.IsDir() { if !fi.IsDir() && strings.HasSuffix(fi.Name(), ".json") {
fr, err := os.Open(appPath + "repo/bundles/" + fi.Name()) fr, err := os.Open(appPath + "repo/bundles/" + fi.Name())
if err != nil { if err != nil {
fmt.Println(err) fmt.Println(err)
@ -202,6 +202,7 @@ func loadLocalBundles() bool {
if len(bundle.Name) == 0 { if len(bundle.Name) == 0 {
bundle.Name = fi.Name()[:strings.Index(fi.Name(), ".")] bundle.Name = fi.Name()[:strings.Index(fi.Name(), ".")]
} }
localBundles = append(localBundles, bundle) localBundles = append(localBundles, bundle)
} }
} }

110
install.go

@ -113,9 +113,14 @@ func runInstall(cmd *Command, args []string) {
installGOPATH = utils.GetBestMatchGOPATH(appPath) installGOPATH = utils.GetBestMatchGOPATH(appPath)
fmt.Printf("Packages will be downloaded to GOPATH(%s).\n", installGOPATH) fmt.Printf("Packages will be downloaded to GOPATH(%s).\n", installGOPATH)
// Generate temporary nodes.
nodes := make([]*doc.Node, len(args))
for i := range nodes {
nodes[i] = new(doc.Node)
nodes[i].ImportPath = args[i]
}
// Download packages. // Download packages.
commits := make([]string, len(args)) downloadPackages(nodes)
downloadPackages(args, commits)
if !cmdInstall.Flags["-d"] && cmdInstall.Flags["-p"] { if !cmdInstall.Flags["-d"] && cmdInstall.Flags["-p"] {
// Install packages all together. // Install packages all together.
@ -144,40 +149,48 @@ func runInstall(cmd *Command, args []string) {
} }
// checkLocalBundles checks if the bundle is in local file system. // checkLocalBundles checks if the bundle is in local file system.
func checkLocalBundles(bundle string) (pkgs, commits []string) { func checkLocalBundles(bundle string) (nodes []*doc.Node) {
for _, b := range localBundles { for _, b := range localBundles {
if bundle == b.Name { if bundle == b.Name {
for _, n := range b.Nodes { for _, n := range b.Nodes {
pkgs = append(pkgs, n.ImportPath)
// Make sure it will not download all dependencies automatically. // Make sure it will not download all dependencies automatically.
if len(n.Commit) == 0 { if len(n.Value) == 0 {
commits = append(commits, "B") n.Value = "B"
} else { }
commits = append(commits, n.Commit) nodes = append(nodes, n)
// Check dependencies.
for _, d := range n.Deps {
// Make sure it will not download all dependencies automatically.
if len(d.Value) == 0 {
d.Value = "B"
}
nodes = append(nodes, d)
} }
} }
return pkgs, commits return nodes
} }
} }
return nil, nil return nil
} }
// downloadPackages downloads packages with certain commit, // downloadPackages downloads packages with certain commit,
// if the commit is empty string, then it downloads all dependencies, // if the commit is empty string, then it downloads all dependencies,
// otherwise, it only downloada package with specific commit only. // otherwise, it only downloada package with specific commit only.
func downloadPackages(pkgs, commits []string) { func downloadPackages(nodes []*doc.Node) {
// Check all packages, they may be bundles, snapshots or raw packages path. // Check all packages, they may be bundles, snapshots or raw packages path.
for i, p := range pkgs { for _, n := range nodes {
// Check if it is a bundle or snapshot. // Check if it is a bundle or snapshot.
switch { switch {
case p[0] == 'B': case n.ImportPath[0] == 'B':
// Check local bundles. // Check local bundles.
bpkgs, bcommits := checkLocalBundles(p[1:]) bnodes := checkLocalBundles(n.ImportPath[1:])
if len(bpkgs) > 0 { if len(nodes) > 0 {
// Check with users if continue. // Check with users if continue.
fmt.Printf("Bundle(%s) contains following nodes:\n", p[1:]) fmt.Printf("Bundle(%s) contains following nodes:\n",
for i := range bpkgs { n.ImportPath[1:])
fmt.Printf("import path: %s, commit: %s.\n", bpkgs[i], bcommits[i]) for _, bn := range bnodes {
fmt.Printf("[%s] -> %s: %s.\n", bn.ImportPath, bn.Type, bn.Value)
} }
fmt.Print("Continue to download?(Y/n).") fmt.Print("Continue to download?(Y/n).")
var option string var option string
@ -185,36 +198,40 @@ func downloadPackages(pkgs, commits []string) {
if strings.ToLower(option) != "y" { if strings.ToLower(option) != "y" {
os.Exit(0) os.Exit(0)
} }
downloadPackages(bpkgs, bcommits) downloadPackages(bnodes)
} else { } else {
// Check from server. // Check from server.
// TODO: api.GetBundleInfo() // TODO: api.GetBundleInfo()
fmt.Println("Unable to check with server right now.") fmt.Println("Unable to check with server right now.")
} }
case p[0] == 'S': case n.ImportPath[0] == 'S':
// TODO: api.GetSnapshotInfo() // TODO: api.GetSnapshotInfo()
case utils.IsValidRemotePath(p): case utils.IsValidRemotePath(n.ImportPath):
if !downloadCache[p] { if !downloadCache[n.ImportPath] {
// Download package. // Download package.
node, imports := downloadPackage(p, commits[i]) node, imports := downloadPackage(n)
if len(imports) > 0 { if len(imports) > 0 {
// Need to download dependencies. // Need to download dependencies.
tags := make([]string, len(imports)) // Generate temporary nodes.
downloadPackages(imports, tags) nodes := make([]*doc.Node, len(imports))
for i := range nodes {
nodes[i] = new(doc.Node)
nodes[i].ImportPath = imports[i]
}
downloadPackages(nodes)
} }
// Only save package information with specific commit. // Only save package information with specific commit.
if node != nil { if node != nil {
// Save record in local nodes. // Save record in local nodes.
saveNode(node) saveNode(node)
//fmt.Printf("Saved information: %s:%s.\n", pkg.ImportPath, pkg.Commit)
} }
} else { } else {
fmt.Printf("Skipped downloaded package: %s.\n", p) fmt.Printf("Skipped downloaded package: %s.\n", n.ImportPath)
} }
default: default:
// Invalid import path. // Invalid import path.
fmt.Printf("Skipped invalid import path: %s.\n", p) fmt.Printf("Skipped invalid import path: %s.\n", n.ImportPath)
} }
} }
} }
@ -222,9 +239,9 @@ func downloadPackages(pkgs, commits []string) {
// saveNode saves node into local nodes. // saveNode saves node into local nodes.
func saveNode(n *doc.Node) { func saveNode(n *doc.Node) {
// Check if this node exists. // Check if this node exists.
for _, v := range localNodes { for i, v := range localNodes {
if n.ImportPath == v.ImportPath { if n.ImportPath == v.ImportPath {
v = n localNodes[i] = n
return return
} }
} }
@ -234,14 +251,14 @@ func saveNode(n *doc.Node) {
} }
// downloadPackage download package either use version control tools or not. // downloadPackage download package either use version control tools or not.
func downloadPackage(path, commit string) (node *doc.Node, imports []string) { func downloadPackage(node *doc.Node) (*doc.Node, []string) {
// Check if use version control tools. // Check if use version control tools.
switch { switch {
case !cmdInstall.Flags["-p"] && case !cmdInstall.Flags["-p"] &&
((path[0] == 'g' && isHasGit) || (path[0] == 'c' && isHasHg)): // github.com, code.google.com ((node.ImportPath[0] == 'g' && isHasGit) || (node.ImportPath[0] == 'c' && isHasHg)): // github.com, code.google.com
fmt.Printf("Installing package(%s) through 'go get'.\n", path) fmt.Printf("Installing package(%s) through 'go get'.\n", node.ImportPath)
args := checkGoGetFlags() args := checkGoGetFlags()
args = append(args, path) args = append(args, node.ImportPath)
executeGoCommand(args) executeGoCommand(args)
return nil, nil return nil, nil
default: // Pure download. default: // Pure download.
@ -250,14 +267,13 @@ func downloadPackage(path, commit string) (node *doc.Node, imports []string) {
fmt.Printf("No version control tool is available, pure download enabled!\n") fmt.Printf("No version control tool is available, pure download enabled!\n")
} }
fmt.Printf("Downloading package: %s.\n", path) fmt.Printf("Downloading package: %s.\n", node.ImportPath)
// Mark as donwloaded. // Mark as donwloaded.
downloadCache[path] = true downloadCache[node.ImportPath] = true
var err error imports, err := pureDownload(node)
node, imports, err = pureDownload(path, commit)
if err != nil { if err != nil {
fmt.Printf("Fail to download package(%s) with error: %s.\n", path, err) fmt.Printf("Fail to download package(%s) with error: %s.\n", node.ImportPath, err)
return nil, nil return nil, nil
} }
@ -282,7 +298,7 @@ func checkGoGetFlags() (args []string) {
type service struct { type service struct {
pattern *regexp.Regexp pattern *regexp.Regexp
prefix string prefix string
get func(*http.Client, map[string]string, string, string, map[string]bool) (*doc.Node, []string, error) get func(*http.Client, map[string]string, string, *doc.Node, map[string]bool) ([]string, error)
} }
// services is the list of source code control services handled by gopkgdoc. // services is the list of source code control services handled by gopkgdoc.
@ -294,26 +310,26 @@ var services = []*service{
} }
// pureDownload downloads package without version control. // pureDownload downloads package without version control.
func pureDownload(path, commit string) (*doc.Node, []string, error) { func pureDownload(node *doc.Node) ([]string, error) {
for _, s := range services { for _, s := range services {
if s.get == nil || !strings.HasPrefix(path, s.prefix) { if s.get == nil || !strings.HasPrefix(node.ImportPath, s.prefix) {
continue continue
} }
m := s.pattern.FindStringSubmatch(path) m := s.pattern.FindStringSubmatch(node.ImportPath)
if m == nil { if m == nil {
if s.prefix != "" { if s.prefix != "" {
return nil, nil, return nil,
doc.NotFoundError{"Import path prefix matches known service, but regexp does not."} doc.NotFoundError{"Import path prefix matches known service, but regexp does not."}
} }
continue continue
} }
match := map[string]string{"importPath": path} match := map[string]string{"importPath": node.ImportPath}
for i, n := range s.pattern.SubexpNames() { for i, n := range s.pattern.SubexpNames() {
if n != "" { if n != "" {
match[n] = m[i] match[n] = m[i]
} }
} }
return s.get(doc.HttpClient, match, installGOPATH, commit, cmdInstall.Flags) return s.get(doc.HttpClient, match, installGOPATH, node, cmdInstall.Flags)
} }
return nil, nil, doc.ErrNoMatch return nil, doc.ErrNoMatch
} }

Loading…
Cancel
Save