mirror of
https://github.com/tiennm99/ghstats.git
synced 2026-05-25 09:37:23 +00:00
40c311d304
Align card set with github-profile-summary-cards' 5-card layout: 0-profile-details.svg (unchanged) 1-repos-per-language.svg (new) owned repos grouped by primary language 2-most-commit-language.svg (new) last-year commits attributed to each repo's primary language 3-stats.svg (renumbered) 4-productive-time.svg (renumbered) - FetchProductive now fills p.CommitsByLanguage from the same commit history it uses for the heatmap, so no extra API calls are introduced. - TopRepos carries primary language so productive-time can aggregate by lang. - LangStat.Bytes renamed to Value (repo count or commit count, context-dependent). - Shared bar+legend renderer extracted to language_bar.go. - Ignore generated output/ directory.
95 lines
2.4 KiB
Go
95 lines
2.4 KiB
Go
package github
|
|
|
|
import (
|
|
"time"
|
|
)
|
|
|
|
// productiveGQL is the response shape for commitHistoryQuery.
|
|
type productiveGQL struct {
|
|
Repository *struct {
|
|
DefaultBranchRef *struct {
|
|
Target *struct {
|
|
History struct {
|
|
PageInfo struct {
|
|
HasNextPage bool `json:"hasNextPage"`
|
|
EndCursor string `json:"endCursor"`
|
|
} `json:"pageInfo"`
|
|
Nodes []struct {
|
|
CommittedDate string `json:"committedDate"`
|
|
} `json:"nodes"`
|
|
} `json:"history"`
|
|
} `json:"target"`
|
|
} `json:"defaultBranchRef"`
|
|
} `json:"repository"`
|
|
}
|
|
|
|
// FetchProductive fills p.Productive with a [7][24] commit histogram over the
|
|
// last year and p.CommitsByLanguage with commit counts attributed to each
|
|
// repo's primary language. Commits are gathered from the given repos (usually
|
|
// p.TopRepos[:N]); each repo is sampled up to maxPerRepo commits to keep the
|
|
// cost bounded.
|
|
//
|
|
// The timezone loc is applied to CommittedDate so the heatmap reflects when
|
|
// the user actually commits, not UTC.
|
|
func (c *Client) FetchProductive(p *Profile, repos []RepoInfo, loc *time.Location, maxPerRepo int) error {
|
|
if loc == nil {
|
|
loc = time.UTC
|
|
}
|
|
since := time.Now().AddDate(-1, 0, 0).UTC().Format(time.RFC3339)
|
|
|
|
commitsByLang := map[string]int64{}
|
|
langColor := map[string]string{}
|
|
|
|
for _, repo := range repos {
|
|
var cursor *string
|
|
seen := 0
|
|
for {
|
|
if seen >= maxPerRepo {
|
|
break
|
|
}
|
|
vars := map[string]any{
|
|
"login": p.Login,
|
|
"repo": repo.Name,
|
|
"userId": p.ID,
|
|
"since": since,
|
|
}
|
|
if cursor != nil {
|
|
vars["after"] = *cursor
|
|
}
|
|
|
|
var resp productiveGQL
|
|
if err := c.query(commitHistoryQuery, vars, &resp); err != nil {
|
|
return err
|
|
}
|
|
if resp.Repository == nil || resp.Repository.DefaultBranchRef == nil ||
|
|
resp.Repository.DefaultBranchRef.Target == nil {
|
|
break
|
|
}
|
|
h := resp.Repository.DefaultBranchRef.Target.History
|
|
for _, n := range h.Nodes {
|
|
t, err := time.Parse(time.RFC3339, n.CommittedDate)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
tl := t.In(loc)
|
|
p.Productive[int(tl.Weekday())][tl.Hour()]++
|
|
if repo.PrimaryLanguage != "" {
|
|
commitsByLang[repo.PrimaryLanguage]++
|
|
if _, ok := langColor[repo.PrimaryLanguage]; !ok {
|
|
langColor[repo.PrimaryLanguage] = repo.PrimaryColor
|
|
}
|
|
}
|
|
seen++
|
|
}
|
|
if !h.PageInfo.HasNextPage {
|
|
break
|
|
}
|
|
end := h.PageInfo.EndCursor
|
|
cursor = &end
|
|
}
|
|
}
|
|
|
|
p.CommitsByLanguage = sortLangStats(commitsByLang, langColor)
|
|
return nil
|
|
}
|