2017-12-26 02:45:08 +00:00
|
|
|
package selfupdate
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"github.com/blang/semver"
|
|
|
|
"github.com/google/go-github/github"
|
|
|
|
"golang.org/x/oauth2"
|
|
|
|
"net/http"
|
|
|
|
"os"
|
2017-12-26 03:02:00 +00:00
|
|
|
"regexp"
|
2017-12-26 09:45:52 +00:00
|
|
|
"runtime"
|
2017-12-26 02:45:08 +00:00
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
2017-12-26 03:02:00 +00:00
|
|
|
var reVersion = regexp.MustCompile(`\d+\.\d+\.\d+`)
|
|
|
|
|
2017-12-26 02:45:08 +00:00
|
|
|
// ReleaseDetector is responsible for detecting the latest release using GitHub Releases API.
|
|
|
|
type ReleaseDetector struct {
|
2017-12-26 03:02:00 +00:00
|
|
|
api *github.Client
|
|
|
|
apiCtx context.Context
|
2017-12-26 02:45:08 +00:00
|
|
|
}
|
|
|
|
|
2017-12-26 09:45:52 +00:00
|
|
|
func findSuitableReleaseAndAsset(rels []*github.RepositoryRelease) (*github.RepositoryRelease, *github.ReleaseAsset, bool) {
|
|
|
|
// Generate candidates
|
|
|
|
cs := make([]string, 0, 8)
|
|
|
|
for _, sep := range []rune{'_', '-'} {
|
2017-12-31 03:10:34 +00:00
|
|
|
for _, ext := range []string{".zip", ".tar.gz", ".gzip", ".gz", ".tar.xz", ".xz", ""} {
|
2017-12-27 04:08:16 +00:00
|
|
|
suffix := fmt.Sprintf("%s%c%s%s", runtime.GOOS, sep, runtime.GOARCH, ext)
|
2017-12-26 09:45:52 +00:00
|
|
|
cs = append(cs, suffix)
|
|
|
|
if runtime.GOOS == "windows" {
|
2017-12-27 04:08:16 +00:00
|
|
|
suffix = fmt.Sprintf("%s%c%s.exe%s", runtime.GOOS, sep, runtime.GOARCH, ext)
|
2017-12-26 09:45:52 +00:00
|
|
|
cs = append(cs, suffix)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, rel := range rels {
|
|
|
|
if rel.GetDraft() {
|
|
|
|
log.Println("Skip draft version", rel.GetTagName())
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if rel.GetPrerelease() {
|
|
|
|
log.Println("Skip pre-release version", rel.GetTagName())
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if !reVersion.MatchString(rel.GetTagName()) {
|
|
|
|
log.Println("Skip version not adopting semver", rel.GetTagName())
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
for _, asset := range rel.Assets {
|
|
|
|
name := asset.GetName()
|
|
|
|
for _, c := range cs {
|
|
|
|
if strings.HasSuffix(name, c) {
|
|
|
|
return rel, &asset, true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Println("Could no find any release for", runtime.GOOS, "and", runtime.GOARCH)
|
|
|
|
return nil, nil, false
|
|
|
|
}
|
|
|
|
|
2017-12-26 02:45:08 +00:00
|
|
|
// NewDetector crates a new detector instance. It initializes GitHub API client.
|
2017-12-26 03:02:00 +00:00
|
|
|
func NewDetector() *ReleaseDetector {
|
2017-12-26 02:45:08 +00:00
|
|
|
token := os.Getenv("GITHUB_TOKEN")
|
|
|
|
ctx := context.Background()
|
|
|
|
|
|
|
|
var auth *http.Client
|
|
|
|
if token != "" {
|
|
|
|
src := oauth2.StaticTokenSource(&oauth2.Token{AccessToken: token})
|
|
|
|
auth = oauth2.NewClient(ctx, src)
|
|
|
|
}
|
|
|
|
|
|
|
|
client := github.NewClient(auth)
|
2017-12-26 03:02:00 +00:00
|
|
|
return &ReleaseDetector{client, ctx}
|
2017-12-26 02:45:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// DetectLatest tries to get the latest version of the repository on GitHub. 'slug' means 'owner/name' formatted string.
|
2017-12-26 09:45:52 +00:00
|
|
|
func (d *ReleaseDetector) DetectLatest(slug string) (release *Release, found bool, err error) {
|
2017-12-26 02:45:08 +00:00
|
|
|
repo := strings.Split(slug, "/")
|
|
|
|
if len(repo) != 2 || repo[0] == "" || repo[1] == "" {
|
|
|
|
err = fmt.Errorf("Invalid slug format. It should be 'owner/name': %s", slug)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-12-26 09:45:52 +00:00
|
|
|
rels, res, err := d.api.Repositories.ListReleases(d.apiCtx, repo[0], repo[1], nil)
|
2017-12-26 02:45:08 +00:00
|
|
|
if err != nil {
|
2017-12-26 03:02:00 +00:00
|
|
|
log.Println("API returned an error response:", err)
|
2017-12-29 09:14:43 +00:00
|
|
|
if res != nil && res.StatusCode == 404 {
|
2017-12-26 02:45:08 +00:00
|
|
|
// 404 means repository not found or release not found. It's not an error here.
|
|
|
|
found = false
|
|
|
|
err = nil
|
|
|
|
log.Println("API returned 404. Repository or release not found")
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-12-26 09:45:52 +00:00
|
|
|
rel, asset, found := findSuitableReleaseAndAsset(rels)
|
|
|
|
if !found {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-12-26 02:45:08 +00:00
|
|
|
tag := rel.GetTagName()
|
2017-12-26 09:45:52 +00:00
|
|
|
url := asset.GetBrowserDownloadURL()
|
|
|
|
log.Println("Successfully fetched the latest release. tag:", tag, ", name:", rel.GetName(), ", URL:", rel.GetURL(), ", Asset:", url)
|
2017-12-26 02:45:08 +00:00
|
|
|
|
2017-12-26 03:02:00 +00:00
|
|
|
// Strip version prefix
|
|
|
|
if indices := reVersion.FindStringIndex(tag); indices != nil && indices[0] > 0 {
|
2017-12-26 09:45:52 +00:00
|
|
|
log.Println("Strip prefix of version:", tag[:indices[0]])
|
2017-12-26 03:02:00 +00:00
|
|
|
tag = tag[indices[0]:]
|
|
|
|
}
|
|
|
|
|
2017-12-31 08:30:54 +00:00
|
|
|
publishedAt := rel.GetPublishedAt().Time
|
2017-12-27 03:16:26 +00:00
|
|
|
release = &Release{
|
2017-12-31 08:30:54 +00:00
|
|
|
AssetURL: url,
|
|
|
|
AssetByteSize: asset.GetSize(),
|
|
|
|
URL: rel.GetHTMLURL(),
|
|
|
|
ReleaseNotes: rel.GetBody(),
|
|
|
|
Name: rel.GetName(),
|
|
|
|
PublishedAt: &publishedAt,
|
2017-12-27 03:16:26 +00:00
|
|
|
}
|
|
|
|
|
2017-12-26 09:45:52 +00:00
|
|
|
release.Version, err = semver.Make(tag)
|
2017-12-26 02:45:08 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2017-12-26 03:02:00 +00:00
|
|
|
// DetectLatest detects the latest release of the slug (owner/repo).
|
2017-12-26 09:45:52 +00:00
|
|
|
func DetectLatest(slug string) (*Release, bool, error) {
|
2017-12-26 03:02:00 +00:00
|
|
|
return NewDetector().DetectLatest(slug)
|
2017-12-26 02:45:08 +00:00
|
|
|
}
|