Browse Source

Merge pull request #15 from gpmgo/dev

Dev
pull/103/head
无闻 11 years ago
parent
commit
bf08e28098
  1. 5
      README.md
  2. 26
      cmd/bin.go
  3. 3
      cmd/cmd.go
  4. 31
      cmd/cmd_test.go
  5. 7
      cmd/gen.go
  6. 12
      cmd/get.go
  7. 18
      cmd/gopath.go
  8. 5
      cmd/install.go
  9. 2
      doc/conf.go
  10. 127
      doc/utils.go
  11. 71
      doc/utils_test.go
  12. 8
      doc/vcs.go
  13. 7
      gopm.go

5
README.md

@ -14,11 +14,14 @@ Please see **[Documentation](https://github.com/gpmgo/docs)** before you ever st
# Commands # Commands
``` ```
NAME:
gopm - Go Package Manager
USAGE: USAGE:
gopm [global options] command [command options] [arguments...] gopm [global options] command [command options] [arguments...]
VERSION: VERSION:
0.6.3.0311 0.6.5.0320
COMMANDS: COMMANDS:
get fetch remote package(s) and dependencies to local repository get fetch remote package(s) and dependencies to local repository

26
cmd/bin.go

@ -1,4 +1,4 @@
// Copyright 2013 gopm authors. // Copyright 2013-2014 gopm authors.
// //
// Licensed under the Apache License, Version 2.0 (the "License"): you may // 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 // not use this file except in compliance with the License. You may obtain
@ -110,8 +110,7 @@ func runBin(ctx *cli.Context) {
// Change to repository path. // Change to repository path.
log.Log("Changing work directory to %s", repoPath) log.Log("Changing work directory to %s", repoPath)
err = os.Chdir(repoPath) if err = os.Chdir(repoPath); err != nil {
if err != nil {
log.Error("bin", "Fail to change work directory:") log.Error("bin", "Fail to change work directory:")
log.Fatal("", "\t"+err.Error()) log.Fatal("", "\t"+err.Error())
} }
@ -123,6 +122,7 @@ func runBin(ctx *cli.Context) {
os.RemoveAll(path.Join(repoPath, doc.VENDOR)) os.RemoveAll(path.Join(repoPath, doc.VENDOR))
}() }()
includes := make([]string, 0, 3)
// Check if previous steps were successful. // Check if previous steps were successful.
if com.IsFile(doc.GOPM_FILE_NAME) { if com.IsFile(doc.GOPM_FILE_NAME) {
log.Trace("Loading gopmfile...") log.Trace("Loading gopmfile...")
@ -133,6 +133,8 @@ func runBin(ctx *cli.Context) {
if err == nil { if err == nil {
log.Log("Target name: %s", pkgName) log.Log("Target name: %s", pkgName)
} }
includes = strings.Split(gf.MustValue("res", "include"), "|")
} }
if len(pkgName) == 0 { if len(pkgName) == 0 {
@ -157,20 +159,30 @@ func runBin(ctx *cli.Context) {
} }
if com.IsExist(movePath + "/" + binName) { if com.IsExist(movePath + "/" + binName) {
err = os.Remove(movePath + "/" + binName) if err = os.Remove(movePath + "/" + binName); err != nil {
if err != nil {
log.Warn("Cannot remove binary in work directory:") log.Warn("Cannot remove binary in work directory:")
log.Warn("\t %s", err) log.Warn("\t %s", err)
} }
} }
err = os.Rename(binName, movePath+"/"+binName) if err = os.Rename(binName, movePath+"/"+binName); err != nil {
if err != nil {
log.Error("bin", "Fail to move binary:") log.Error("bin", "Fail to move binary:")
log.Fatal("", "\t"+err.Error()) log.Fatal("", "\t"+err.Error())
} }
os.Chmod(movePath+"/"+binName, os.ModePerm) os.Chmod(movePath+"/"+binName, os.ModePerm)
if len(includes) > 0 {
log.Log("Copying resources to %s", movePath)
for _, include := range includes {
if com.IsDir(include) {
if err = com.CopyDir(include, filepath.Join(movePath, include)); err != nil {
log.Error("bin", "Fail to copy following resource:")
log.Error("", "\t"+include)
}
}
}
}
log.Log("Changing work directory back to %s", wd) log.Log("Changing work directory back to %s", wd)
os.Chdir(wd) os.Chdir(wd)

3
cmd/cmd.go

@ -83,5 +83,6 @@ func versionSuffix(value string) string {
} }
func isSubpackage(rootPath, targetPath string) bool { func isSubpackage(rootPath, targetPath string) bool {
return strings.HasSuffix(workDir, rootPath) || strings.HasPrefix(rootPath, targetPath) return strings.HasSuffix(strings.Replace(workDir, "\\", "/", -1), rootPath) ||
strings.HasPrefix(rootPath, targetPath)
} }

31
cmd/cmd_test.go

@ -0,0 +1,31 @@
// Copyright 2013-2014 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 (
"testing"
. "github.com/smartystreets/goconvey/convey"
)
func Test_parseTarget(t *testing.T) {
Convey("Target is empty", t, func() {
So(parseTarget(""), ShouldEqual, ".")
})
Convey("Target is not empty", t, func() {
So(parseTarget("github.com/gpmgo/gopm"), ShouldEqual, "github.com/gpmgo/gopm")
})
}

7
cmd/gen.go

@ -1,4 +1,4 @@
// Copyright 2013 gopm authors. // Copyright 2013-2014 gopm authors.
// //
// Licensed under the Apache License, Version 2.0 (the "License"): you may // 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 // not use this file except in compliance with the License. You may obtain
@ -58,12 +58,11 @@ func runGen(ctx *cli.Context) {
targetPath := parseTarget(gf.MustValue("target", "path")) targetPath := parseTarget(gf.MustValue("target", "path"))
// Get and set dependencies. // Get and set dependencies.
imports := doc.GetAllImports([]string{workDir}, targetPath, ctx.Bool("example")) imports := doc.GetAllImports([]string{workDir}, targetPath, ctx.Bool("example"), false)
log.Log("%v", imports)
for _, p := range imports { for _, p := range imports {
p = doc.GetProjectPath(p) p = doc.GetProjectPath(p)
// Skip subpackage(s) of current project. // Skip subpackage(s) of current project.
if strings.HasSuffix(workDir, p) || strings.HasPrefix(p, targetPath) { if isSubpackage(p, targetPath) {
continue continue
} }

12
cmd/get.go

@ -106,8 +106,11 @@ func runGet(ctx *cli.Context) {
switch len(ctx.Args()) { switch len(ctx.Args()) {
case 0: case 0:
getByGopmfile(ctx) getByGopmfile(ctx)
default: case 1:
getByPath(ctx) getByPath(ctx)
default:
log.Error("get", "too many arguments")
log.Help("Try 'gopm help get' to get more information")
} }
} }
@ -120,10 +123,11 @@ func getByGopmfile(ctx *cli.Context) {
targetPath := parseTarget(gf.MustValue("target", "path")) targetPath := parseTarget(gf.MustValue("target", "path"))
// Get dependencies. // Get dependencies.
imports := doc.GetAllImports([]string{workDir}, targetPath, ctx.Bool("example")) imports := doc.GetAllImports([]string{workDir}, targetPath, ctx.Bool("example"), false)
nodes := make([]*doc.Node, 0, len(imports)) nodes := make([]*doc.Node, 0, len(imports))
for _, p := range imports { for _, p := range imports {
// TODO: DOING TEST CASES!!!
p = doc.GetProjectPath(p) p = doc.GetProjectPath(p)
// Skip subpackage(s) of current project. // Skip subpackage(s) of current project.
if isSubpackage(p, targetPath) { if isSubpackage(p, targetPath) {
@ -312,12 +316,12 @@ func downloadPackage(ctx *cli.Context, nod *doc.Node) (*doc.Node, []string) {
vcs := getVcsName(gopathDir) vcs := getVcsName(gopathDir)
if ctx.Bool("update") && ctx.Bool("gopath") && len(vcs) > 0 { if ctx.Bool("update") && ctx.Bool("gopath") && len(vcs) > 0 {
err = updateByVcs(vcs, gopathDir) err = updateByVcs(vcs, gopathDir)
imports = doc.GetAllImports([]string{gopathDir}, nod.RootPath, false) imports = doc.GetAllImports([]string{gopathDir}, nod.RootPath, false, false)
} else { } else {
// If package has revision and exist, then just check dependencies. // If package has revision and exist, then just check dependencies.
if nod.IsGetDepsOnly { if nod.IsGetDepsOnly {
return nod, doc.GetAllImports([]string{path.Join(installRepoPath, nod.RootPath) + versionSuffix(nod.Value)}, return nod, doc.GetAllImports([]string{path.Join(installRepoPath, nod.RootPath) + versionSuffix(nod.Value)},
nod.RootPath, ctx.Bool("example")) nod.RootPath, ctx.Bool("example"), false)
} }
nod.Revision = doc.LocalNodes.MustValue(nod.RootPath, "value") nod.Revision = doc.LocalNodes.MustValue(nod.RootPath, "value")
imports, err = doc.PureDownload(nod, installRepoPath, ctx) //CmdGet.Flags) imports, err = doc.PureDownload(nod, installRepoPath, ctx) //CmdGet.Flags)

18
cmd/gopath.go

@ -1,3 +1,17 @@
// Copyright 2013-2014 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 package cmd
import ( import (
@ -35,7 +49,7 @@ func getGopmPkgs(dirPath string, isTest bool) (pkgs map[string]*doc.Pkg, err err
} }
} }
imports := doc.GetAllImports([]string{dirPath}, ".", false) imports := doc.GetAllImports([]string{dirPath}, ".", false, false)
pkgs = make(map[string]*doc.Pkg) pkgs = make(map[string]*doc.Pkg)
for _, name := range imports { for _, name := range imports {
if name == "C" { if name == "C" {
@ -257,7 +271,7 @@ func genNewGoPath(ctx *cli.Context, isTest bool) {
continue continue
} }
if !isExistP && ((len(pkg.Value) > 0 || ctx.Bool("remote")) || if !isExistP && (len(pkg.Value) > 0 || ctx.Bool("remote") ||
!com.IsDir(filepath.Join(installGopath, pkg.ImportPath))) { !com.IsDir(filepath.Join(installGopath, pkg.ImportPath))) {
log.Log("Linking %s", name+suf) log.Log("Linking %s", name+suf)
err = autoLink(oldPath, newPath) err = autoLink(oldPath, newPath)

5
cmd/install.go

@ -1,4 +1,4 @@
// Copyright 2013 gopm authors. // Copyright 2013-2014 gopm authors.
// //
// Licensed under the Apache License, Version 2.0 (the "License"): you may // 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 // not use this file except in compliance with the License. You may obtain
@ -83,8 +83,7 @@ func runInstall(ctx *cli.Context) {
var installRepos []string var installRepos []string
if ctx.Bool("pkg") { if ctx.Bool("pkg") {
curPath, _ := filepath.Abs(".") curPath, _ := filepath.Abs(".")
installRepos = doc.GetAllImports([]string{curPath}, installRepos = doc.GetAllImports([]string{curPath}, ".", ctx.Bool("example"), false)
".", ctx.Bool("example"))
} else { } else {
if len(target) == 0 { if len(target) == 0 {
target = pkgName target = pkgName

2
doc/conf.go

@ -59,7 +59,7 @@ func init() {
} }
} }
Cfg, err = goconfig.LoadConfigFile(cfgPath) Cfg, err = goconfig.LoadConfigFile(cfgPath)
if _, err = os.Create(cfgPath); err != nil { if err != nil {
log.Error("", "Fail to load gopm config file") log.Error("", "Fail to load gopm config file")
log.Fatal("", err.Error()) log.Fatal("", err.Error())
} }

127
doc/utils.go

@ -1,4 +1,4 @@
// Copyright 2013 gopm authors. // Copyright 2013-2014 gopm authors.
// //
// Licensed under the Apache License, Version 2.0 (the "License"): you may // 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 // not use this file except in compliance with the License. You may obtain
@ -15,11 +15,16 @@
package doc package doc
import ( import (
"bytes"
"go/build" "go/build"
"io"
"io/ioutil"
"os" "os"
"path" "path"
"path/filepath"
"regexp" "regexp"
"strings" "strings"
"time"
"github.com/Unknwon/com" "github.com/Unknwon/com"
@ -29,26 +34,105 @@ import (
const VENDOR = ".vendor" const VENDOR = ".vendor"
// GetDirsInfo returns os.FileInfo of all sub-directories in root path. // GetDirsInfo returns os.FileInfo of all sub-directories in root path.
func GetDirsInfo(rootPath string) []os.FileInfo { func GetDirsInfo(rootPath string) ([]os.FileInfo, error) {
if !com.IsDir(rootPath) {
log.Warn("Directory %s does not exist", rootPath)
return []os.FileInfo{}, nil
}
rootDir, err := os.Open(rootPath) rootDir, err := os.Open(rootPath)
if err != nil { if err != nil {
log.Error("", "Fail to open directory") return nil, err
log.Fatal("", err.Error())
} }
defer rootDir.Close() defer rootDir.Close()
dirs, err := rootDir.Readdir(0) dirs, err := rootDir.Readdir(0)
if err != nil { if err != nil {
log.Error("", "Fail to read directory") return nil, err
log.Fatal("", err.Error()) }
return dirs, nil
}
// A Source describles a Source code file.
type Source struct {
SrcName string
SrcData []byte
}
func (s *Source) Name() string { return s.SrcName }
func (s *Source) Size() int64 { return int64(len(s.SrcData)) }
func (s *Source) Mode() os.FileMode { return 0 }
func (s *Source) ModTime() time.Time { return time.Time{} }
func (s *Source) IsDir() bool { return false }
func (s *Source) Sys() interface{} { return nil }
func (s *Source) Data() []byte { return s.SrcData }
type Context struct {
build.Context
importPath string
srcFiles map[string]*Source
}
func (ctx *Context) readDir(dir string) ([]os.FileInfo, error) {
fis := make([]os.FileInfo, 0, len(ctx.srcFiles))
for _, src := range ctx.srcFiles {
fis = append(fis, src)
} }
return fis, nil
}
return dirs func (ctx *Context) openFile(path string) (r io.ReadCloser, err error) {
if src, ok := ctx.srcFiles[filepath.Base(path)]; ok {
return ioutil.NopCloser(bytes.NewReader(src.Data())), nil
}
return nil, os.ErrNotExist
} }
// GetImports returns package denpendencies. // GetImports returns package denpendencies.
func GetImports(absPath, importPath string, example bool) []string { func GetImports(absPath, importPath string, example, test bool) []string {
pkg, err := build.ImportDir(absPath, build.AllowBinary) fis, err := GetDirsInfo(absPath)
if err != nil {
log.Error("", "Fail to get directory's information")
log.Fatal("", err.Error())
}
absPath += "/"
ctx := new(Context)
ctx.importPath = importPath
ctx.srcFiles = make(map[string]*Source)
ctx.Context = build.Default
ctx.JoinPath = path.Join
ctx.IsAbsPath = path.IsAbs
ctx.ReadDir = ctx.readDir
ctx.OpenFile = ctx.openFile
// TODO: Load too much, need to make sure which is imported which are not.
dirs := make([]string, 0, 10)
for _, fi := range fis {
if strings.Contains(fi.Name(), VENDOR) {
continue
}
if fi.IsDir() {
dirs = append(dirs, absPath+fi.Name())
continue
} else if !test && strings.HasSuffix(fi.Name(), "_test.go") {
continue
} else if !strings.HasSuffix(fi.Name(), ".go") || strings.HasPrefix(fi.Name(), ".") ||
strings.HasPrefix(fi.Name(), "_") {
continue
}
src := &Source{SrcName: fi.Name()}
src.SrcData, err = ioutil.ReadFile(absPath + fi.Name())
if err != nil {
log.Error("", "Fail to read file")
log.Fatal("", err.Error())
}
ctx.srcFiles[fi.Name()] = src
}
pkg, err := ctx.ImportDir(absPath, build.AllowBinary)
if err != nil { if err != nil {
if _, ok := err.(*build.NoGoError); !ok { if _, ok := err.(*build.NoGoError); !ok {
log.Error("", "Fail to get imports") log.Error("", "Fail to get imports")
@ -56,9 +140,6 @@ func GetImports(absPath, importPath string, example bool) []string {
} }
} }
fis := GetDirsInfo(absPath)
absPath += "/"
imports := make([]string, 0, len(pkg.Imports)) imports := make([]string, 0, len(pkg.Imports))
for _, p := range pkg.Imports { for _, p := range pkg.Imports {
if !IsGoRepoPath(p) && !strings.HasPrefix(p, importPath) { if !IsGoRepoPath(p) && !strings.HasPrefix(p, importPath) {
@ -66,31 +147,25 @@ func GetImports(absPath, importPath string, example bool) []string {
} }
} }
// TODO: Load too much
dirs := make([]string, 0, len(imports))
for _, fi := range fis {
if fi.IsDir() && !strings.Contains(fi.Name(), VENDOR) {
dirs = append(dirs, absPath+fi.Name())
}
}
if len(dirs) > 0 { if len(dirs) > 0 {
imports = append(imports, GetAllImports(dirs, importPath, example)...) imports = append(imports, GetAllImports(dirs, importPath, example, test)...)
} }
return imports return imports
} }
// isVcsPath returns true if the directory was created by VCS.
func isVcsPath(dirPath string) bool { func isVcsPath(dirPath string) bool {
return strings.Contains(dirPath, "/.git") || return strings.Contains(dirPath, "/.git") ||
strings.Contains(dirPath, "/.hg") || strings.Contains(dirPath, "/.hg") ||
strings.Contains(dirPath, "/.svn") strings.Contains(dirPath, "/.svn")
} }
func GetAllImports(dirs []string, importPath string, example bool) (imports []string) { // GetAllImports returns all imports in given directory and all sub-directories.
func GetAllImports(dirs []string, importPath string, example, test bool) (imports []string) {
for _, d := range dirs { for _, d := range dirs {
if !isVcsPath(d) && if !isVcsPath(d) &&
!(!example && strings.Contains(d, "example")) { !(!example && strings.Contains(d, "example")) {
imports = append(imports, GetImports(d, importPath, example)...) imports = append(imports, GetImports(d, importPath, example, test)...)
} }
} }
return imports return imports
@ -116,7 +191,11 @@ func CheckIsExistWithVCS(path string) bool {
} }
// Check if only has VCS folder. // Check if only has VCS folder.
dirs := GetDirsInfo(path) dirs, err := GetDirsInfo(path)
if err != nil {
log.Error("", "Fail to get directory's information")
log.Fatal("", err.Error())
}
if len(dirs) > 1 { if len(dirs) > 1 {
return true return true

71
doc/utils_test.go

@ -0,0 +1,71 @@
// Copyright 2013-2014 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 doc
import (
"testing"
. "github.com/smartystreets/goconvey/convey"
)
var VcsTestPairs = map[string]bool{
"/.hg": true,
"/.git": true,
"/.svn": true,
"/.vendor": false,
}
func Test_isVcsPath(t *testing.T) {
Convey("Test if the path is belonging to VCS", t, func() {
for name, expect := range VcsTestPairs {
So(isVcsPath(name), ShouldEqual, expect)
}
})
}
func TestGetDirsInfo(t *testing.T) {
Convey("Get directory's information that exist", t, func() {
dis, err := GetDirsInfo(".")
So(err, ShouldBeNil)
So(len(dis), ShouldEqual, 13)
})
Convey("Get directory's information does not exist", t, func() {
dis, err := GetDirsInfo("./404")
So(err, ShouldBeNil)
So(len(dis), ShouldEqual, 0)
})
}
var GoStdTestPairs = map[string]bool{
"net/http": true,
"fmt": true,
"github.com/gpmgo/gopm": false,
"github.com/Unknwon/com": false,
}
func TestIsGoRepoPath(t *testing.T) {
Convey("Test if the path is belonging to Go STD", t, func() {
for name, expect := range GoStdTestPairs {
So(IsGoRepoPath(name), ShouldEqual, expect)
}
})
}
func TestGetImports(t *testing.T) {
Convey("Get package that are imported", t, func() {
So(len(GetImports(".", "github.com/gpmgo/gopm/docs", false)), ShouldEqual, 4)
})
}

8
doc/vcs.go

@ -1,4 +1,4 @@
// Copyright 2013 gopm authors. // Copyright 2013-2014 gopm authors.
// //
// Licensed under the Apache License, Version 2.0 (the "License"): you may // 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 // not use this file except in compliance with the License. You may obtain
@ -245,7 +245,11 @@ metaScan:
} }
func getImports(rootPath string, match map[string]string, cmdFlags map[string]bool, nod *Node) (imports []string) { func getImports(rootPath string, match map[string]string, cmdFlags map[string]bool, nod *Node) (imports []string) {
dirs := GetDirsInfo(rootPath) dirs, err := GetDirsInfo(rootPath)
if err != nil {
log.Error("", "Fail to get directory's information")
log.Fatal("", err.Error())
}
for _, d := range dirs { for _, d := range dirs {
if d.IsDir() && !(!cmdFlags["-e"] && strings.Contains(d.Name(), "example")) { if d.IsDir() && !(!cmdFlags["-e"] && strings.Contains(d.Name(), "example")) {

7
gopm.go

@ -29,9 +29,10 @@ 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.6.3.0312" const APP_VER = "0.6.5.0320"
// //cmd.CmdSearch, // cmd.CmdTest,
// cmd.CmdSearch,
// cmdClean, // cmdClean,
// cmdDoc, // cmdDoc,
// cmdEnv, // cmdEnv,
@ -39,7 +40,6 @@ const APP_VER = "0.6.3.0312"
// cmdList, // cmdList,
// cmdTool, // cmdTool,
// cmdVet, // cmdVet,
// }
func init() { func init() {
runtime.GOMAXPROCS(runtime.NumCPU()) runtime.GOMAXPROCS(runtime.NumCPU())
@ -59,7 +59,6 @@ func main() {
cmd.CmdInstall, cmd.CmdInstall,
cmd.CmdUpdate, cmd.CmdUpdate,
cmd.CmdConfig, cmd.CmdConfig,
//cmd.CmdTest,
} }
app.Flags = append(app.Flags, []cli.Flag{ app.Flags = append(app.Flags, []cli.Flag{
cli.BoolFlag{"noterm", "disable color output"}, cli.BoolFlag{"noterm", "disable color output"},

Loading…
Cancel
Save