package main import ( "encoding/json" "errors" "net/http" "regexp" "strings" "time" "golang.org/x/net/context" ) func init() { registerServiceHandler("github", githubServiceHandler{}) } type githubRelease struct { TagName string `json:"tag_name"` Assets []githubAsset `json:"assets"` } type githubAsset struct { Name string `json:"name"` Downloads int64 `json:"download_count"` } type githubRepo struct { StargazersCount int64 `json:"stargazers_count"` } type githubServiceHandler struct{} func (g githubServiceHandler) GetDocumentation() serviceHandlerDocumentationList { return serviceHandlerDocumentationList{ { ServiceName: "GitHub repo license", DemoPath: "/github/license/Luzifer/badge-gen", Arguments: []string{"license", "", ""}, }, { ServiceName: "GitHub latest tag", DemoPath: "/github/latest-tag/Luzifer/badge-gen", Arguments: []string{"latest-tag", "", ""}, }, { ServiceName: "GitHub latest release", DemoPath: "/github/latest-release/lastpass/lastpass-cli", Arguments: []string{"latest-release", "", ""}, }, { ServiceName: "GitHub downloads by repo", DemoPath: "/github/downloads/atom/atom", Arguments: []string{"downloads", "", ""}, }, { ServiceName: "GitHub downloads by release", DemoPath: "/github/downloads/atom/atom/latest", Arguments: []string{"downloads", "", "", ""}, }, { ServiceName: "GitHub downloads by release and asset", DemoPath: "/github/downloads/atom/atom/v1.8.0/atom-amd64.deb", Arguments: []string{"downloads", "", "", "", ""}, }, { ServiceName: "Github stars by repository", DemoPath: "/github/stars/atom/atom", Arguments: []string{"stars", "", ""}, }, } } func (githubServiceHandler) IsEnabled() bool { return true } func (g githubServiceHandler) Handle(ctx context.Context, params []string) (title, text, color string, err error) { if len(params) < 2 { err = errors.New("No service-command / parameters were given") return } switch params[0] { case "license": title, text, color, err = g.handleLicense(ctx, params[1:]) case "latest-tag": title, text, color, err = g.handleLatestTag(ctx, params[1:]) case "latest-release": title, text, color, err = g.handleLatestRelease(ctx, params[1:]) case "downloads": title, text, color, err = g.handleDownloads(ctx, params[1:]) case "stars": title, text, color, err = g.handleStargazers(ctx, params[1:]) default: err = errors.New("An unknown service command was called") } return } func (g githubServiceHandler) handleStargazers(ctx context.Context, params []string) (title, text, color string, err error) { path := strings.Join([]string{"repos", params[0], params[1]}, "/") text, err = cacheStore.Get("github_repo_stargazers", path) if err != nil { r := githubRepo{} if err = g.fetchAPI(ctx, path, nil, &r); err != nil { return } text = metricFormat(r.StargazersCount) cacheStore.Set("github_repo_stargazers", path, text, 10*time.Minute) } title = "stars" color = "brightgreen" return } func (g githubServiceHandler) handleDownloads(ctx context.Context, params []string) (title, text, color string, err error) { switch len(params) { case 2: title, text, color, err = g.handleRepoDownloads(ctx, params) case 3: params = append(params, "total") fallthrough case 4: title, text, color, err = g.handleReleaseDownloads(ctx, params) default: err = errors.New("Unsupported number of arguments") } return } func (g githubServiceHandler) handleReleaseDownloads(ctx context.Context, params []string) (title, text, color string, err error) { path := strings.Join([]string{"repos", params[0], params[1], "releases", "tags", params[2]}, "/") if params[2] == "latest" { path = strings.Join([]string{"repos", params[0], params[1], "releases", params[2]}, "/") } text, err = cacheStore.Get("github_release_downloads", path) if err != nil { r := githubRelease{} if err = g.fetchAPI(ctx, path, nil, &r); err != nil { return } var sum int64 for _, rel := range r.Assets { if params[3] == "total" || rel.Name == params[3] { sum = sum + rel.Downloads } } text = metricFormat(sum) cacheStore.Set("github_release_downloads", path, text, 10*time.Minute) } title = "downloads" color = "brightgreen" return } func (g githubServiceHandler) handleRepoDownloads(ctx context.Context, params []string) (title, text, color string, err error) { path := strings.Join([]string{"repos", params[0], params[1], "releases"}, "/") text, err = cacheStore.Get("github_repo_downloads", path) if err != nil { r := []githubRelease{} // TODO: This does not respect pagination! if err = g.fetchAPI(ctx, path, nil, &r); err != nil { return } var sum int64 for _, rel := range r { for _, rea := range rel.Assets { sum = sum + rea.Downloads } } text = metricFormat(sum) cacheStore.Set("github_repo_downloads", path, text, 10*time.Minute) } title = "downloads" color = "brightgreen" return } func (g githubServiceHandler) handleLatestRelease(ctx context.Context, params []string) (title, text, color string, err error) { path := strings.Join([]string{"repos", params[0], params[1], "releases", "latest"}, "/") text, err = cacheStore.Get("github_latest_release", path) if err != nil { r := githubRelease{} if err = g.fetchAPI(ctx, path, nil, &r); err != nil { return } text = r.TagName if text == "" { text = "None" } cacheStore.Set("github_latest_release", path, text, 10*time.Minute) } title = "release" color = "blue" if regexp.MustCompile(`^v?0\.`).MatchString(text) { color = "orange" } return } func (g githubServiceHandler) handleLatestTag(ctx context.Context, params []string) (title, text, color string, err error) { path := strings.Join([]string{"repos", params[0], params[1], "tags"}, "/") text, err = cacheStore.Get("github_latest_tag", path) if err != nil { r := []struct { Name string `json:"name"` }{} if err = g.fetchAPI(ctx, path, nil, &r); err != nil { return } if len(r) > 0 { text = r[0].Name } else { text = "None" } cacheStore.Set("github_latest_tag", path, text, 10*time.Minute) } title = "tag" color = "blue" if regexp.MustCompile(`^v?0\.`).MatchString(text) { color = "orange" } return } func (g githubServiceHandler) handleLicense(ctx context.Context, params []string) (title, text, color string, err error) { path := strings.Join([]string{"repos", params[0], params[1], "license"}, "/") text, err = cacheStore.Get("github_license", path) if err != nil { r := struct { License struct { Name string `json:"name"` } `json:"license"` }{} headers := map[string]string{ "Accept": "application/vnd.github.drax-preview+json", } if err = g.fetchAPI(ctx, path, headers, &r); err != nil { return } text = r.License.Name cacheStore.Set("github_license", path, text, 10*time.Minute) } title = "license" color = "007ec6" if text == "" { text = "None" } return } func (g githubServiceHandler) fetchAPI(ctx context.Context, path string, headers map[string]string, out interface{}) error { req, _ := http.NewRequest("GET", "https://api.github.com/"+path, nil) if headers != nil { for k, v := range headers { req.Header.Set(k, v) } } if configStore.Str("github.personal_token") != "" { req.SetBasicAuth(configStore.Str("github.username"), configStore.Str("github.personal_token")) } resp, err := http.DefaultClient.Do(req.WithContext(ctx)) if err != nil { return err } defer resp.Body.Close() return json.NewDecoder(resp.Body).Decode(out) }