package util import ( "errors" "fmt" "io" "net/http" "os" "os/exec" "path/filepath" "runtime" "time" "context" "github.com/google/go-github/github" ) // Adapted from github.com/apex/apex // UpdateAvailable contains the full repository information for the latest release of Semaphore var UpdateAvailable *github.RepositoryRelease // DoUpgrade checks for an update, and if available downloads the binary and installs it func DoUpgrade(version string) error { fmt.Printf("current release is v%s\n", version) if err := CheckUpdate(version); err != nil || UpdateAvailable == nil { return err } asset := findAsset(UpdateAvailable) if asset == nil { return errors.New("cannot find binary for your system") } // create tmp file tmpPath := filepath.Join(os.TempDir(), "semaphore-upgrade") f, err := os.OpenFile(tmpPath, os.O_CREATE|os.O_TRUNC|os.O_RDWR, 0755) //nolint: gas if err != nil { return err } // download binary fmt.Printf("downloading %s\n", *asset.BrowserDownloadURL) res, err := http.Get(*asset.BrowserDownloadURL) if err != nil { return err } defer res.Body.Close() //nolint: errcheck // copy it down _, err = io.Copy(f, res.Body) if err != nil { return err } // replace it cmdPath := FindSemaphore() if len(cmdPath) == 0 { return errors.New("Cannot find semaphore binary") } fmt.Printf("replacing %s\n", cmdPath) err = os.Rename(tmpPath, cmdPath) if err != nil { return err } fmt.Println("visit https://github.com/ansible-semaphore/semaphore/releases for the changelog") go func() { time.Sleep(time.Second * 3) os.Exit(0) }() return nil } // FindSemaphore looks in the PATH for the semaphore variable // if not found it will attempt to find the absolute path of the first // os argument, the semaphore command, and return it func FindSemaphore() string { cmdPath, _ := exec.LookPath("semaphore") //nolint: gas if len(cmdPath) == 0 { cmdPath, _ = filepath.Abs(os.Args[0]) // nolint: gas } return cmdPath } // findAsset returns the binary for this platform. func findAsset(release *github.RepositoryRelease) *github.ReleaseAsset { for _, asset := range release.Assets { if *asset.Name == fmt.Sprintf("semaphore_%s_%s", runtime.GOOS, runtime.GOARCH) { return &asset } } return nil } // CheckUpdate uses the github client to check for new tags in the semaphore repo func CheckUpdate(version string) error { // fetch releases gh := github.NewClient(nil) releases, _, err := gh.Repositories.ListReleases(context.TODO(), "ansible-semaphore", "semaphore", nil) if err != nil { return err } UpdateAvailable = nil if (*releases[0].TagName)[1:] != version { UpdateAvailable = releases[0] } return nil }