Browse Source

Fixed bugs in gopm get

pull/103/head
Unknown 11 years ago
parent
commit
a3c123b929
  1. 105
      README.md
  2. 9
      beewatch.json
  3. 23
      doc/bitbucket.go
  4. 39
      doc/github.go
  5. 199
      doc/google.go
  6. 19
      doc/launchpad.go
  7. 11
      doc/oschina.go
  8. 3
      doc/utils.go
  9. 119
      doc/vcs.go
  10. 2
      gopm.go

105
README.md

@ -1,16 +1,103 @@
gopm - Go Package Manager validation
========================= ==============
![GPMGo_Logo](https://raw.github.com/gpmgo/gopmweb/master/static/img/gpmgo.png?raw=true) validation is a form validation for a data validation and error collecting using Go.
gopm(Go Package Manager) is a Go package manage tool for search, install, update and share packages in Go. ## Installation and tests
**Attention** This application still in experiment, we'are working on new break version, you may use [old version](https://github.com/gpmgo/gopm/tree/v0.1.0) for now. Install:
## Credits go get github.com/astaxie/beego/validation
- [Go Walker](https://github.com/Unknwon/gowalker) Test:
## License go test github.com/astaxie/beego/validation
[Apache License, Version 2.0](http://www.apache.org/licenses/LICENSE-2.0.html). ## Example
Direct Use:
import (
"github.com/astaxie/beego/validation"
"log"
)
type User struct {
Name string
Age int
}
func main() {
u := User{"man", 40}
valid := validation.Validation{}
valid.Required(u.Name, "name")
valid.MaxSize(u.Name, 15, "nameMax")
valid.Range(u.Age, 0, 140, "age")
if valid.HasErrors {
// validation does not pass
// print invalid message
for _, err := range valid.Errors {
log.Println(err.Key, err.Message)
}
}
// or use like this
if v := valid.Max(u.Age, 140); !v.Ok {
log.Println(v.Error.Key, v.Error.Message)
}
}
Struct Tag Use:
import (
"github.com/astaxie/beego/validation"
)
// validation function follow with "valid" tag
// functions divide with ";"
// parameters in parentheses "()" and divide with ","
// Match function's pattern string must in "//"
type user struct {
Id int
Name string `valid:"Required;Match(/^(test)?\\w*@;com$/)"`
Age int `valid:"Required;Range(1, 140)"`
}
func main() {
valid := Validation{}
u := user{Name: "test", Age: 40}
b, err := valid.Valid(u)
if err != nil {
// handle error
}
if !b {
// validation does not pass
// blabla...
}
}
Struct Tag Functions:
Required
Min(min int)
Max(max int)
Range(min, max int)
MinSize(min int)
MaxSize(max int)
Length(length int)
Alpha
Numeric
AlphaNumeric
Match(pattern string)
AlphaDash
Email
IP
Base64
Mobile
Tel
Phone
ZipCode
## LICENSE
BSD License http://creativecommons.org/licenses/BSD/

9
beewatch.json

@ -1,9 +0,0 @@
{
"app_name": "Go Package Manager",
"http_port": 23456,
"watch_enabled": true,
"cmd_mode": true,
"skip_suspend": false,
"print_stack": true,
"print_source": true
}

23
doc/bitbucket.go

@ -43,7 +43,7 @@ func getBitbucketDoc(client *http.Client, match map[string]string, installRepoPa
var repo struct { var repo struct {
Scm string Scm string
} }
if err := com.HttpGetJSON(client, expand("https://api.bitbucket.org/1.0/repositories/{owner}/{repo}", match), &repo); err != nil { if err := com.HttpGetJSON(client, com.Expand("https://api.bitbucket.org/1.0/repositories/{owner}/{repo}", match), &repo); err != nil {
return nil, err return nil, err
} }
match["vcs"] = repo.Scm match["vcs"] = repo.Scm
@ -64,7 +64,7 @@ func getBitbucketDoc(client *http.Client, match map[string]string, installRepoPa
var nodes map[string]struct { var nodes map[string]struct {
Node string Node string
} }
if err := com.HttpGetJSON(client, expand("https://api.bitbucket.org/1.0/repositories/{owner}/{repo}/{0}", match, nodeType), &nodes); err != nil { if err := com.HttpGetJSON(client, com.Expand("https://api.bitbucket.org/1.0/repositories/{owner}/{repo}/{0}", match, nodeType), &nodes); err != nil {
return nil, err return nil, err
} }
for t, n := range nodes { for t, n := range nodes {
@ -96,7 +96,7 @@ func getBitbucketDoc(client *http.Client, match map[string]string, installRepoPa
// tarball : https://bitbucket.org/{owner}/{repo}/get/{commit}.tar.gz // tarball : https://bitbucket.org/{owner}/{repo}/get/{commit}.tar.gz
// Downlaod archive. // Downlaod archive.
p, err := com.HttpGetBytes(client, expand("https://bitbucket.org/{owner}/{repo}/get/{commit}.tar.gz", match), nil) p, err := com.HttpGetBytes(client, com.Expand("https://bitbucket.org/{owner}/{repo}/get/{commit}.tar.gz", match), nil)
if err != nil { if err != nil {
return nil, err return nil, err
} }
@ -107,7 +107,7 @@ func getBitbucketDoc(client *http.Client, match map[string]string, installRepoPa
if len(suf) == 1 { if len(suf) == 1 {
suf = "" suf = ""
} }
projectPath := expand("bitbucket.org/{owner}/{repo}", match) projectPath := com.Expand("bitbucket.org/{owner}/{repo}", match)
installPath = installRepoPath + "/" + projectPath + suf installPath = installRepoPath + "/" + projectPath + suf
nod.ImportPath = projectPath nod.ImportPath = projectPath
} else { } else {
@ -137,7 +137,7 @@ func getBitbucketDoc(client *http.Client, match map[string]string, installRepoPa
return nil, err return nil, err
} }
fn := h.FileInfo().Name() fn := h.Name
// In case that we find directory, usually we should not. // In case that we find directory, usually we should not.
if strings.HasSuffix(fn, "/") { if strings.HasSuffix(fn, "/") {
@ -157,24 +157,13 @@ func getBitbucketDoc(client *http.Client, match map[string]string, installRepoPa
os.MkdirAll(dir+"/", os.ModePerm) os.MkdirAll(dir+"/", os.ModePerm)
} }
if strings.HasPrefix(fn, ".") {
continue
}
// 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, err return nil, err
} }
// Write data to file _, err = com.SaveFile(absPath, fbytes)
fw, err := os.Create(absPath)
if err != nil {
return nil, err
}
_, err = fw.Write(fbytes)
fw.Close()
if err != nil { if err != nil {
return nil, err return nil, err
} }

39
doc/github.go

@ -29,25 +29,11 @@ import (
) )
var ( var (
githubRawHeader = http.Header{"Accept": {"application/vnd.github-blob.raw"}}
githubPattern = regexp.MustCompile(`^github\.com/(?P<owner>[a-z0-9A-Z_.\-]+)/(?P<repo>[a-z0-9A-Z_.\-]+)(?P<dir>/[a-z0-9A-Z_.\-/]*)?$`) githubPattern = regexp.MustCompile(`^github\.com/(?P<owner>[a-z0-9A-Z_.\-]+)/(?P<repo>[a-z0-9A-Z_.\-]+)(?P<dir>/[a-z0-9A-Z_.\-/]*)?$`)
githubCred string
) )
/*func SetGithubCredentials(id, secret string) {
//githubCred = "client_id=" + id + "&client_secret=" + secret
}*/
func SetGithubCredentials(token string) {
if len(token) > 0 {
githubCred = "access_token=" + token
}
}
// getGithubDoc downloads tarball from github.com. // getGithubDoc downloads tarball from github.com.
func getGithubDoc(client *http.Client, match map[string]string, installRepoPath string, nod *Node, cmdFlags map[string]bool) ([]string, error) { func getGithubDoc(client *http.Client, match map[string]string, installRepoPath string, nod *Node, cmdFlags map[string]bool) ([]string, error) {
match["cred"] = githubCred
// Check downlaod type. // Check downlaod type.
switch nod.Type { switch nod.Type {
case BRANCH: case BRANCH:
@ -67,12 +53,12 @@ func getGithubDoc(client *http.Client, match map[string]string, installRepoPath
// tarball: https://github.com/{owner}/{repo}/tarball/{sha} // tarball: https://github.com/{owner}/{repo}/tarball/{sha}
// Downlaod archive. // Downlaod archive.
p, err := com.HttpGetBytes(client, expand("https://github.com/{owner}/{repo}/archive/{sha}.zip", match), nil) p, err := com.HttpGetBytes(client, com.Expand("https://github.com/{owner}/{repo}/archive/{sha}.zip", match), nil)
if err != nil { if err != nil {
return nil, errors.New("Fail to donwload Github repo -> " + err.Error()) return nil, errors.New("Fail to donwload Github repo -> " + err.Error())
} }
shaName := expand("{repo}-{sha}", match) shaName := com.Expand("{repo}-{sha}", match)
if nod.Type == "tag" { if nod.Type == "tag" {
shaName = strings.Replace(shaName, "-v", "-", 1) shaName = strings.Replace(shaName, "-v", "-", 1)
} }
@ -83,7 +69,7 @@ func getGithubDoc(client *http.Client, match map[string]string, installRepoPath
if len(suf) == 1 { if len(suf) == 1 {
suf = "" suf = ""
} }
projectPath := expand("github.com/{owner}/{repo}", match) projectPath := com.Expand("github.com/{owner}/{repo}", match)
installPath = installRepoPath + "/" + projectPath + suf installPath = installRepoPath + "/" + projectPath + suf
nod.ImportPath = projectPath nod.ImportPath = projectPath
} else { } else {
@ -96,14 +82,14 @@ func getGithubDoc(client *http.Client, match map[string]string, installRepoPath
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, err return nil, errors.New(nod.ImportPath + " -> new zip: " + err.Error())
} }
dirs := make([]string, 0, 5) dirs := make([]string, 0, 5)
// Need to add root path because we cannot get from tarball. // Need to add root path because we cannot get from tarball.
dirs = append(dirs, installPath+"/") dirs = append(dirs, installPath+"/")
for _, f := range r.File { for _, f := range r.File {
absPath := strings.Replace(f.FileInfo().Name(), shaName, installPath, 1) absPath := strings.Replace(f.Name, shaName, installPath, 1)
// Create diretory before create file. // Create diretory before create file.
os.MkdirAll(path.Dir(absPath)+"/", os.ModePerm) os.MkdirAll(path.Dir(absPath)+"/", os.ModePerm)
@ -119,7 +105,7 @@ func getGithubDoc(client *http.Client, match map[string]string, installRepoPath
} }
dirs = append(dirs, absPath) dirs = append(dirs, absPath)
} }
case !strings.HasPrefix(f.FileInfo().Name(), "."): default:
// Get file from archive. // Get file from archive.
rc, err := f.Open() rc, err := f.Open()
if err != nil { if err != nil {
@ -157,18 +143,5 @@ func getGithubDoc(client *http.Client, match map[string]string, installRepoPath
imports = append(imports, importPkgs...) imports = append(imports, importPkgs...)
} }
} }
/*fpath := appPath + "repo/tarballs/" + node.ImportPath + "-" + node.Value + ".zip"
// Save tarball.
if autoBackup && !utils.IsExist(fpath) {
os.MkdirAll(path.Dir(fpath)+"/", os.ModePerm)
f, err := os.Create(fpath)
if err != nil {
return nil, err
}
defer f.Close()
_, err = f.Write(p)
}*/
return imports, err return imports, err
} }

199
doc/google.go

@ -15,28 +15,47 @@
package doc package doc
import ( import (
"archive/zip"
"bytes"
"errors" "errors"
"io"
"net/http" "net/http"
"os" "os"
"path" "path"
"regexp" "regexp"
"strings"
"github.com/Unknwon/com" "github.com/Unknwon/com"
"github.com/Unknwon/ctw/packer"
) )
var ( var (
googleRepoRe = regexp.MustCompile(`id="checkoutcmd">(hg|git|svn)`)
googleFileRe = regexp.MustCompile(`<li><a href="([^"/]+)"`)
googleDirRe = regexp.MustCompile(`<li><a href="([^".]+)"`)
googlePattern = regexp.MustCompile(`^code\.google\.com/p/(?P<repo>[a-z0-9\-]+)(:?\.(?P<subrepo>[a-z0-9\-]+))?(?P<dir>/[a-z0-9A-Z_.\-/]+)?$`) googlePattern = regexp.MustCompile(`^code\.google\.com/p/(?P<repo>[a-z0-9\-]+)(:?\.(?P<subrepo>[a-z0-9\-]+))?(?P<dir>/[a-z0-9A-Z_.\-/]+)?$`)
) )
// 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, installRepoPath string, nod *Node, cmdFlags map[string]bool) ([]string, error) { func getGoogleDoc(client *http.Client, match map[string]string, installRepoPath string, nod *Node, cmdFlags map[string]bool) ([]string, error) {
packer.SetupGoogleMatch(match) setupGoogleMatch(match)
// Check version control. // Check version control.
if err := packer.GetGoogleVCS(client, match); err != nil { if err := getGoogleVCS(client, match); err != nil {
return nil, err return nil, err
} }
switch nod.Type {
case BRANCH:
if len(nod.Value) == 0 {
match["tag"] = defaultTags[match["vcs"]]
} else {
match["tag"] = nod.Value
}
case TAG, COMMIT:
match["tag"] = nod.Value
default:
return nil, errors.New("Unknown node type: " + nod.Type)
}
var installPath string var installPath string
projectPath := GetProjectPath(nod.ImportPath) projectPath := GetProjectPath(nod.ImportPath)
if nod.ImportPath == nod.DownloadURL { if nod.ImportPath == nod.DownloadURL {
@ -51,35 +70,67 @@ func getGoogleDoc(client *http.Client, match map[string]string, installRepoPath
// Remove old files. // Remove old files.
os.RemoveAll(installPath + "/") os.RemoveAll(installPath + "/")
match["tag"] = nod.Value os.MkdirAll(installPath+"/", os.ModePerm)
ext := ".zip"
if match["vcs"] == "svn" { if match["vcs"] == "svn" {
ext = ".tar.gz"
com.ColorLog("[WARN] SVN detected, may take very long time.\n") com.ColorLog("[WARN] SVN detected, may take very long time.\n")
rootPath := com.Expand("http://{subrepo}{dot}{repo}.googlecode.com/{vcs}", match)
d, f := path.Split(rootPath)
err := downloadFiles(client, match, d, installPath+"/", match["tag"],
[]string{f + "/"})
if err != nil {
return nil, errors.New("Fail to download " + nod.ImportPath + " : " + err.Error())
}
} }
err := packer.PackToFile(match["importPath"], installPath+ext, match) p, err := com.HttpGetBytes(client, com.Expand("http://{subrepo}{dot}{repo}.googlecode.com/archive/{tag}.zip", match), nil)
if err != nil { if err != nil {
return nil, err return nil, errors.New("Fail to download " + nod.ImportPath + " : " + err.Error())
} }
var dirs []string r, err := zip.NewReader(bytes.NewReader(p), int64(len(p)))
if match["vcs"] != "svn" { if err != nil {
dirs, err = com.Unzip(installPath+ext, path.Dir(installPath)) return nil, errors.New(nod.ImportPath + " -> new zip: " + err.Error())
} else { }
dirs, err = com.UnTarGz(installPath+ext, path.Dir(installPath))
nameLen := strings.Index(r.File[0].Name, "/")
dirPrefix := match["dir"]
if len(dirPrefix) != 0 {
dirPrefix = dirPrefix[1:] + "/"
}
dirs := make([]string, 0, 5)
for _, f := range r.File {
absPath := strings.Replace(f.Name, f.Name[:nameLen], installPath, 1)
// Create diretory before create file.
dir := path.Dir(absPath)
if !checkDir(dir, dirs) && !(!cmdFlags["-e"] && strings.Contains(absPath, "example")) {
dirs = append(dirs, dir)
os.MkdirAll(dir+"/", os.ModePerm)
} }
if len(dirs) == 0 { // Get file from archive.
return nil, errors.New("No file in repository") rc, err := f.Open()
if err != nil {
return nil, err
} }
// Write data to file
fw, _ := os.Create(absPath)
if err != nil { if err != nil {
return nil, err return nil, err
} }
os.Remove(installPath + ext)
os.Rename(path.Dir(installPath)+"/"+dirs[0], installPath) _, err = io.Copy(fw, rc)
// Close files.
rc.Close()
fw.Close()
if err != nil {
return nil, err
}
}
// Check if need to check imports. // Check if need to check imports.
if nod.IsGetDeps { if nod.IsGetDeps {
@ -89,3 +140,117 @@ func getGoogleDoc(client *http.Client, match map[string]string, installRepoPath
return nil, err return nil, err
} }
type rawFile struct {
name string
rawURL string
data []byte
}
func (rf *rawFile) Name() string {
return rf.name
}
func (rf *rawFile) RawUrl() string {
return rf.rawURL
}
func (rf *rawFile) Data() []byte {
return rf.data
}
func (rf *rawFile) SetData(p []byte) {
rf.data = p
}
func downloadFiles(client *http.Client, match map[string]string, rootPath, installPath, commit string, dirs []string) error {
suf := "?r=" + commit
if len(commit) == 0 {
suf = ""
}
for _, d := range dirs {
p, err := com.HttpGetBytes(client, rootPath+d+suf, nil)
if err != nil {
return err
}
// Create destination directory.
os.MkdirAll(installPath+d, os.ModePerm)
// Get source files in current path.
files := make([]com.RawFile, 0, 5)
for _, m := range googleFileRe.FindAllSubmatch(p, -1) {
fname := strings.Split(string(m[1]), "?")[0]
files = append(files, &rawFile{
name: fname,
rawURL: rootPath + d + fname + suf,
})
}
// Fetch files from VCS.
if err := com.FetchFilesCurl(files); err != nil {
return err
}
// Save files.
for _, f := range files {
absPath := installPath + d
// Create diretory before create file.
os.MkdirAll(path.Dir(absPath), os.ModePerm)
// Write data to file
fw, err := os.Create(absPath + f.Name())
if err != nil {
return err
}
_, err = fw.Write(f.Data())
fw.Close()
if err != nil {
return err
}
}
files = nil
subdirs := make([]string, 0, 3)
// Get subdirectories.
for _, m := range googleDirRe.FindAllSubmatch(p, -1) {
dirName := strings.Split(string(m[1]), "?")[0]
if strings.HasSuffix(dirName, "/") {
subdirs = append(subdirs, d+dirName)
}
}
err = downloadFiles(client, match, rootPath, installPath, commit, subdirs)
if err != nil {
return err
}
}
return nil
}
func setupGoogleMatch(match map[string]string) {
if s := match["subrepo"]; s != "" {
match["dot"] = "."
match["query"] = "?repo=" + s
} else {
match["dot"] = ""
match["query"] = ""
}
}
func getGoogleVCS(client *http.Client, match map[string]string) error {
// Scrape the HTML project page to find the VCS.
p, err := com.HttpGetBytes(client, com.Expand("http://code.google.com/p/{repo}/source/checkout", match), nil)
if err != nil {
return errors.New("doc.getGoogleVCS(" + match["importPath"] + ") -> " + err.Error())
}
m := googleRepoRe.FindSubmatch(p)
if m == nil {
return com.NotFoundError{"Could not VCS on Google Code project page."}
}
match["vcs"] = string(m[1])
return nil
}

19
doc/launchpad.go

@ -33,7 +33,7 @@ var launchpadPattern = regexp.MustCompile(`^launchpad\.net/(?P<repo>(?P<project>
func getLaunchpadDoc(client *http.Client, match map[string]string, installRepoPath string, nod *Node, cmdFlags map[string]bool) ([]string, error) { func getLaunchpadDoc(client *http.Client, match map[string]string, installRepoPath string, nod *Node, cmdFlags map[string]bool) ([]string, error) {
if match["project"] != "" && match["series"] != "" { if match["project"] != "" && match["series"] != "" {
rc, err := com.HttpGet(client, expand("https://code.launchpad.net/{project}{series}/.bzr/branch-format", match), nil) rc, err := com.HttpGet(client, com.Expand("https://code.launchpad.net/{project}{series}/.bzr/branch-format", match), nil)
_, isNotFound := err.(com.NotFoundError) _, isNotFound := err.(com.NotFoundError)
switch { switch {
case err == nil: case err == nil:
@ -42,7 +42,7 @@ func getLaunchpadDoc(client *http.Client, match map[string]string, installRepoPa
case isNotFound: case isNotFound:
// The structure of the import path is is launchpad.net/{project}/{dir}. // The structure of the import path is is launchpad.net/{project}/{dir}.
match["repo"] = match["project"] match["repo"] = match["project"]
match["dir"] = expand("{series}{dir}", match) match["dir"] = com.Expand("{series}{dir}", match)
default: default:
return nil, err return nil, err
} }
@ -51,9 +51,9 @@ func getLaunchpadDoc(client *http.Client, match map[string]string, installRepoPa
var downloadPath string var downloadPath string
// Check if download with specific revision. // Check if download with specific revision.
if len(nod.Value) == 0 { if len(nod.Value) == 0 {
downloadPath = expand("https://bazaar.launchpad.net/+branch/{repo}/tarball", match) downloadPath = com.Expand("https://bazaar.launchpad.net/+branch/{repo}/tarball", match)
} else { } else {
downloadPath = expand("https://bazaar.launchpad.net/+branch/{repo}/tarball/"+nod.Value, match) downloadPath = com.Expand("https://bazaar.launchpad.net/+branch/{repo}/tarball/"+nod.Value, match)
} }
// 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.
@ -87,7 +87,7 @@ func getLaunchpadDoc(client *http.Client, match map[string]string, installRepoPa
return nil, err return nil, err
} }
fn := h.FileInfo().Name() fn := h.Name
// Check root path. // Check root path.
if len(autoPath) == 0 { if len(autoPath) == 0 {
autoPath = fn[:strings.Index(fn, match["repo"])+len(match["repo"])] autoPath = fn[:strings.Index(fn, match["repo"])+len(match["repo"])]
@ -109,14 +109,7 @@ func getLaunchpadDoc(client *http.Client, match map[string]string, installRepoPa
return nil, err return nil, err
} }
// Write data to file _, err = com.SaveFile(absPath, fbytes)
fw, err := os.Create(absPath)
if err != nil {
return nil, err
}
_, err = fw.Write(fbytes)
fw.Close()
if err != nil { if err != nil {
return nil, err return nil, err
} }

11
doc/oschina.go

@ -51,7 +51,7 @@ func getOSCDoc(client *http.Client, match map[string]string, installRepoPath str
// zip: http://{projectRoot}/repository/archive?ref={sha} // zip: http://{projectRoot}/repository/archive?ref={sha}
// Downlaod archive. // Downlaod archive.
p, err := com.HttpGetBytes(client, expand("http://git.oschina.net/{owner}/{repo}/repository/archive?ref={sha}", match), nil) p, err := com.HttpGetBytes(client, com.Expand("http://git.oschina.net/{owner}/{repo}/repository/archive?ref={sha}", match), nil)
if err != nil { if err != nil {
return nil, errors.New("Fail to donwload OSChina repo -> " + err.Error()) return nil, errors.New("Fail to donwload OSChina repo -> " + err.Error())
} }
@ -62,7 +62,7 @@ func getOSCDoc(client *http.Client, match map[string]string, installRepoPath str
if len(suf) == 1 { if len(suf) == 1 {
suf = "" suf = ""
} }
projectPath := expand("git.oschina.net/{owner}/{repo}", match) projectPath := com.Expand("git.oschina.net/{owner}/{repo}", match)
installPath = installRepoPath + "/" + projectPath + suf installPath = installRepoPath + "/" + projectPath + suf
nod.ImportPath = projectPath nod.ImportPath = projectPath
} else { } else {
@ -83,7 +83,7 @@ func getOSCDoc(client *http.Client, match map[string]string, installRepoPath str
// Need to add root path because we cannot get from tarball. // Need to add root path because we cannot get from tarball.
dirs = append(dirs, installPath+"/") dirs = append(dirs, installPath+"/")
for _, f := range r.File { for _, f := range r.File {
fileName := f.FileInfo().Name()[nameLen+1:] fileName := f.Name[nameLen+1:]
absPath := installPath + "/" + fileName absPath := installPath + "/" + fileName
if strings.HasSuffix(absPath, "/") { if strings.HasSuffix(absPath, "/") {
@ -91,11 +91,6 @@ func getOSCDoc(client *http.Client, match map[string]string, installRepoPath str
os.MkdirAll(absPath, os.ModePerm) os.MkdirAll(absPath, os.ModePerm)
continue continue
} }
// d, _ := path.Split(absPath)
// if !checkDir(d, dirs) {
// dirs = append(dirs, d)
// os.MkdirAll(d, os.ModePerm)
// }
// Get file from archive. // Get file from archive.
rc, err := f.Open() rc, err := f.Open()

3
doc/utils.go

@ -97,7 +97,8 @@ func GetProjectPath(importPath string) (projectPath string) {
// Check project hosting. // Check project hosting.
switch { switch {
case strings.HasPrefix(importPath, "github.com"): case strings.HasPrefix(importPath, "github.com") ||
strings.HasPrefix(importPath, "git.oschina.net"):
projectPath = joinPath(importPath, 3) projectPath = joinPath(importPath, 3)
case strings.HasPrefix(importPath, "code.google.com"): case strings.HasPrefix(importPath, "code.google.com"):
projectPath = joinPath(importPath, 3) projectPath = joinPath(importPath, 3)

119
doc/vcs.go

@ -15,18 +15,13 @@
package doc package doc
import ( import (
"bytes"
"encoding/xml" "encoding/xml"
"errors" "errors"
"io" "io"
"io/ioutil"
"log"
"net/http" "net/http"
"os" "os"
"os/exec"
"path" "path"
"regexp" "regexp"
"strconv"
"strings" "strings"
"github.com/Unknwon/com" "github.com/Unknwon/com"
@ -90,117 +85,17 @@ type vcsCmd struct {
download func([]string, string, string) (string, string, error) download func([]string, string, string) (string, string, error)
} }
var vcsCmds = map[string]*vcsCmd{
"git": &vcsCmd{
schemes: []string{"http", "https", "git"},
download: downloadGit,
},
}
var lsremoteRe = regexp.MustCompile(`(?m)^([0-9a-f]{40})\s+refs/(?:tags|heads)/(.+)$`) var lsremoteRe = regexp.MustCompile(`(?m)^([0-9a-f]{40})\s+refs/(?:tags|heads)/(.+)$`)
func downloadGit(schemes []string, repo, savedEtag string) (string, string, error) {
var p []byte
var scheme string
for i := range schemes {
cmd := exec.Command("git", "ls-remote", "--heads", "--tags", schemes[i]+"://"+repo+".git")
log.Println(strings.Join(cmd.Args, " "))
var err error
p, err = cmd.Output()
if err == nil {
scheme = schemes[i]
break
}
}
if scheme == "" {
return "", "", com.NotFoundError{"VCS not found"}
}
tags := make(map[string]string)
for _, m := range lsremoteRe.FindAllSubmatch(p, -1) {
tags[string(m[2])] = string(m[1])
}
tag, commit, err := bestTag(tags, "master")
if err != nil {
return "", "", err
}
etag := scheme + "-" + commit
if etag == savedEtag {
return "", "", errNotModified
}
dir := path.Join(repoRoot, repo+".git")
p, err = ioutil.ReadFile(path.Join(dir, ".git/HEAD"))
switch {
case err != nil:
if err := os.MkdirAll(dir, 0777); err != nil {
return "", "", err
}
cmd := exec.Command("git", "clone", scheme+"://"+repo, dir)
log.Println(strings.Join(cmd.Args, " "))
if err := cmd.Run(); err != nil {
return "", "", err
}
case string(bytes.TrimRight(p, "\n")) == commit:
return tag, etag, nil
default:
cmd := exec.Command("git", "fetch")
log.Println(strings.Join(cmd.Args, " "))
cmd.Dir = dir
if err := cmd.Run(); err != nil {
return "", "", err
}
}
cmd := exec.Command("git", "checkout", "--detach", "--force", commit)
cmd.Dir = dir
if err := cmd.Run(); err != nil {
return "", "", err
}
return tag, etag, nil
}
var defaultTags = map[string]string{"git": "master", "hg": "default"} var defaultTags = map[string]string{"git": "master", "hg": "default"}
func bestTag(tags map[string]string, defaultTag string) (string, string, error) { func bestTag(tags map[string]string, defaultTag string) (string, string, error) {
if commit, ok := tags["go1"]; ok {
return "go1", commit, nil
}
if commit, ok := tags[defaultTag]; ok { if commit, ok := tags[defaultTag]; ok {
return defaultTag, commit, nil return defaultTag, commit, nil
} }
return "", "", com.NotFoundError{"Tag or branch not found."} return "", "", com.NotFoundError{"Tag or branch not found."}
} }
// expand replaces {k} in template with match[k] or subs[atoi(k)] if k is not in match.
func expand(template string, match map[string]string, subs ...string) string {
var p []byte
var i int
for {
i = strings.Index(template, "{")
if i < 0 {
break
}
p = append(p, template[:i]...)
template = template[i+1:]
i = strings.Index(template, "}")
if s, ok := match[template[:i]]; ok {
p = append(p, s...)
} else {
j, _ := strconv.Atoi(template[:i])
p = append(p, subs[j]...)
}
template = template[i+1:]
}
p = append(p, template...)
return string(p)
}
// PureDownload downloads package without version control. // PureDownload downloads package without version control.
func PureDownload(nod *Node, installRepoPath string, flags map[string]bool) ([]string, error) { func PureDownload(nod *Node, installRepoPath string, flags map[string]bool) ([]string, error) {
for _, s := range services { for _, s := range services {
@ -243,7 +138,7 @@ func getDynamic(client *http.Client, nod *Node, installRepoPath string, flags ma
} }
} }
nod.DownloadURL = expand("{repo}{dir}", match) nod.DownloadURL = com.Expand("{repo}{dir}", match)
return PureDownload(nod, installRepoPath, flags) return PureDownload(nod, installRepoPath, flags)
} }
@ -384,22 +279,14 @@ func CheckImports(absPath, importPath string, nod *Node) (importPkgs []string, e
for _, fi := range fis { for _, fi := range fis {
// Only handle files. // Only handle files.
if strings.HasSuffix(fi.Name(), ".go") { if strings.HasSuffix(fi.Name(), ".go") {
f, err := os.Open(absPath + fi.Name()) data, err := com.ReadFile(absPath + fi.Name())
if err != nil {
return nil, err
}
fbytes := make([]byte, fi.Size())
_, err = f.Read(fbytes)
f.Close()
if err != nil { if err != nil {
return nil, err return nil, err
} }
files = append(files, &source{ files = append(files, &source{
name: fi.Name(), name: fi.Name(),
data: fbytes, data: data,
}) })
} }
} }

2
gopm.go

@ -33,7 +33,7 @@ import (
// Test that go1.1 tag above is included in builds. main.go refers to this definition. // Test that go1.1 tag above is included in builds. main.go refers to this definition.
const go11tag = true const go11tag = true
const APP_VER = "0.2.5.0827" const APP_VER = "0.4.0.1012"
var ( var (
config map[string]interface{} config map[string]interface{}

Loading…
Cancel
Save