// 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 ( "fmt" "os" "os/exec" "path" "path/filepath" "runtime" "strings" "github.com/Unknwon/com" "github.com/codegangsta/cli" "github.com/gpmgo/gopm/doc" "github.com/gpmgo/gopm/log" ) var CmdUpdate = cli.Command{ Name: "update", Usage: "update self", Description: `Command bin downloads and links dependencies according to gopmfile, and build executable binary to work directory gopm update Can only specify one each time, and only works for projects that contains main package`, Action: runUpdate, } func exePath() string { file, _ := exec.LookPath(os.Args[0]) path, _ := filepath.Abs(file) return path } func runUpdate(ctx *cli.Context) { doc.LoadPkgNameList(doc.HomeDir + "/data/pkgname.list") installRepoPath = doc.HomeDir + "/repos" // Check arguments. num := 0 if len(ctx.Args()) != num { log.Error("Update", "Fail to start command") log.Fatal("", "Invalid argument number") } // Parse package version. info := "github.com/gpmgo/gopm" pkgPath := info ver := "" var err error if i := strings.Index(info, "@"); i > -1 { pkgPath = info[:i] _, ver, err = validPath(info[i+1:]) if err != nil { log.Error("Update", "Fail to parse version") log.Fatal("", err.Error()) } } // Check package name. if !strings.Contains(pkgPath, "/") { name, ok := doc.PackageNameList[pkgPath] if !ok { log.Error("Update", "Invalid package name: "+pkgPath) log.Fatal("", "No match in the package name list") } pkgPath = name } // Get code. stdout, _, _ := com.ExecCmd("gopm", "get", info) if len(stdout) > 0 { fmt.Print(stdout) } // Check if previous steps were successful. repoPath := installRepoPath + "/" + pkgPath if len(ver) > 0 { repoPath += "." + ver } if !com.IsDir(repoPath) { log.Error("Bin", "Fail to continue command") log.Fatal("", "Previous steps weren't successful") } wd, err := os.Getwd() if err != nil { log.Error("Bin", "Fail to get work directory") log.Fatal("", err.Error()) } // Change to repository path. log.Log("Changing work directory to %s", repoPath) err = os.Chdir(repoPath) if err != nil { log.Error("Bin", "Fail to change work directory") log.Fatal("", err.Error()) } // Build application. stdout, _, _ = com.ExecCmd("gopm", "build") if len(stdout) > 0 { fmt.Print(stdout) } defer func() { // Clean files. os.RemoveAll(path.Join(repoPath, doc.VENDOR)) }() // Check if previous steps were successful. if com.IsFile(doc.GopmFileName) { log.Trace("Loading gopmfile...") gf := doc.NewGopmfile(".") var err error pkgName, err = gf.GetValue("target", "path") if err == nil { log.Log("Target name: %s", pkgName) } } if len(pkgName) == 0 { _, pkgName = filepath.Split(pkgPath) } binName := path.Base(pkgName) if runtime.GOOS == "windows" { binName += ".exe" } binPath := path.Join(doc.VENDOR, "src", pkgPath, binName) if !com.IsFile(binPath) { log.Error("Update", "Fail to continue command") log.Fatal("", "Previous steps weren't successful or the project does not contain main package") } movePath := exePath() fmt.Print(movePath) // Move binary to given directory. if runtime.GOOS != "windows" { err = os.Rename(binPath, movePath) if err != nil { log.Error("Update", "Fail to move binary") log.Fatal("", err.Error()) } os.Chmod(movePath+"/"+binName, os.ModePerm) } else { batPath := filepath.Join(wd, "a.bat") f, err := os.Create(batPath) if err != nil { log.Error("Update", "Fail to generate bat file") log.Fatal("", err.Error()) } f.WriteString(fmt.Sprintf(`ping -n 1 127.0.0.1>nul\ncopy "%v" "%v"\ndel "%v"`, binPath, movePath, binPath)) f.Close() attr := &os.ProcAttr{ Dir: wd, Env: os.Environ(), //Files: []*os.File{nil, nil, nil}, Files: []*os.File{os.Stdin, os.Stdout, os.Stderr}, } _, err = os.StartProcess(batPath, []string{"a.bat"}, attr) if err != nil { log.Error("Update", "Fail to start bat process") log.Fatal("", err.Error()) } } log.Log("Changing work directory back to %s", wd) os.Chdir(wd) log.Success("SUCC", "Update", "Command execute successfully!") }