Files
gico/commits/commits.go

290 lines
6.0 KiB
Go

package commits
import (
"errors"
"os"
"regexp"
"sync"
"time"
git "github.com/go-git/go-git/v5"
"github.com/go-git/go-git/v5/plumbing/object"
"github.com/taigrr/gico/types"
"github.com/taigrr/mg/parse"
)
type Repo git.Repository
func GlobalFrequencyChan(year int, authors []string) (types.YearFreq, error) {
yearLength := 365
if year%4 == 0 {
yearLength++
}
mrconf, err := parse.LoadMRConfig()
if err != nil {
return types.YearFreq{}, err
}
paths := mrconf.GetRepoPaths()
outChan := make(chan types.Commit, 10)
var wg sync.WaitGroup
for _, p := range paths {
wg.Add(1)
go func(path string) {
repo, err := OpenRepo(path)
if err != nil {
return
}
cc, err := repo.GetCommitChan()
if err != nil {
return
}
cc = FilterCChanByYear(cc, year)
cc, err = FilterCChanByAuthor(cc, authors)
if err != nil {
return
}
for c := range cc {
outChan <- c
}
wg.Done()
}(p)
}
go func() {
wg.Wait()
close(outChan)
}()
freq := YearFreqFromChan(outChan, year)
return freq, nil
}
func YearFreqFromChan(cc chan types.Commit, year int) types.YearFreq {
yearLength := 365
if year%4 == 0 {
yearLength++
}
freq := make([]int, yearLength)
data := types.NewDataSet()
for commit := range cc {
ts := commit.TimeStamp
roundedTS := ts.Round(time.Hour * 24)
wd, ok := data[roundedTS]
if !ok {
wd = types.WorkDay{}
wd.Commits = []types.Commit{}
}
wd.Commits = append(wd.Commits, commit)
wd.Count++
wd.Day = roundedTS
data[roundedTS] = wd
}
for k, v := range data {
if k.Year() != year {
continue
}
// this is equivalent to adding len(commits) to the freq total, but
// it's a stub for later when we do more here
for range v.Commits {
freq[k.YearDay()-1]++
}
}
return freq
}
func GlobalFrequency(year int, authors []string) (types.YearFreq, error) {
yearLength := 365
if year%4 == 0 {
yearLength++
}
gfreq := make(types.YearFreq, yearLength)
mrconf, err := parse.LoadMRConfig()
if err != nil {
return types.YearFreq{}, err
}
paths := mrconf.GetRepoPaths()
for _, p := range paths {
repo, err := OpenRepo(p)
if err != nil {
return types.YearFreq{}, err
}
commits, err := repo.GetCommitSet()
if err != nil {
return types.YearFreq{}, err
}
commits = commits.FilterByYear(year)
commits, err = commits.FilterByAuthorRegex(authors)
if err != nil {
return types.YearFreq{}, err
}
freq := commits.ToYearFreq()
gfreq = gfreq.Merge(freq)
}
return gfreq, nil
}
func OpenRepo(directory string) (Repo, error) {
if s, err := os.Stat(directory); err != nil {
return Repo{}, err
} else {
if !s.IsDir() {
return Repo{}, errors.New("received path to non-directory for git repo")
}
}
r, err := git.PlainOpenWithOptions(directory, &(git.PlainOpenOptions{DetectDotGit: true}))
return Repo(*r), err
}
type CommitSet struct {
Commits []types.Commit
Year int
}
func (cs CommitSet) ToYearFreq() types.YearFreq {
year := cs.Year
yearLength := 365
if year%4 == 0 {
yearLength++
}
freq := make([]int, yearLength)
data := types.NewDataSet()
for _, commit := range cs.Commits {
ts := commit.TimeStamp
roundedTS := ts.Round(time.Hour * 24)
wd, ok := data[roundedTS]
if !ok {
wd = types.WorkDay{}
wd.Commits = []types.Commit{}
}
wd.Commits = append(wd.Commits, commit)
wd.Count++
wd.Day = roundedTS
data[roundedTS] = wd
}
for k, v := range data {
if k.Year() != year {
continue
}
// this is equivalent to adding len(commits) to the freq total, but
// it's a stub for later when we do more here
for range v.Commits {
freq[k.YearDay()-1]++
}
}
return freq
}
func (cs CommitSet) FilterByAuthorRegex(authors []string) (CommitSet, error) {
regSet := [](*regexp.Regexp){}
for _, a := range authors {
r, err := regexp.Compile(a)
if err != nil {
return CommitSet{}, err
}
regSet = append(regSet, r)
}
newCS := CommitSet{Year: cs.Year}
for _, commit := range cs.Commits {
for _, r := range regSet {
if r.MatchString(commit.Author) {
newCS.Commits = append(newCS.Commits, commit)
break
}
}
}
return newCS, nil
}
func (cs CommitSet) FilterByYear(year int) CommitSet {
newCS := CommitSet{Year: year}
for _, commit := range cs.Commits {
if commit.TimeStamp.Year() == year {
newCS.Commits = append(newCS.Commits, commit)
}
}
return newCS
}
func FilterCChanByYear(in chan types.Commit, year int) chan types.Commit {
out := make(chan types.Commit, 30)
go func() {
for commit := range in {
if commit.TimeStamp.Year() == year {
out <- commit
}
}
close(out)
}()
return out
}
func FilterCChanByAuthor(in chan types.Commit, authors []string) (chan types.Commit, error) {
out := make(chan types.Commit, 30)
regSet := [](*regexp.Regexp){}
for _, a := range authors {
r, err := regexp.Compile(a)
if err != nil {
close(out)
return out, err
}
regSet = append(regSet, r)
}
go func() {
for commit := range in {
for _, r := range regSet {
if r.MatchString(commit.Author) {
out <- commit
break
}
}
}
close(out)
}()
return out, nil
}
func (repo Repo) GetCommitChan() (chan types.Commit, error) {
cc := make(chan types.Commit, 30)
r := git.Repository(repo)
ref, err := r.Head()
if err != nil {
return cc, err
}
cIter, err := r.Log(&git.LogOptions{From: ref.Hash()})
if err != nil {
return cc, err
}
go func() {
cIter.ForEach(func(c *object.Commit) error {
ts := c.Author.When
commit := types.Commit{Author: c.Author.Name, Message: c.Message, TimeStamp: ts}
cc <- commit
return nil
})
close(cc)
}()
return cc, nil
}
func (repo Repo) GetCommitSet() (CommitSet, error) {
cs := CommitSet{}
commits := []types.Commit{}
r := git.Repository(repo)
ref, err := r.Head()
if err != nil {
return cs, err
}
cIter, err := r.Log(&git.LogOptions{From: ref.Hash()})
if err != nil {
return cs, err
}
cIter.ForEach(func(c *object.Commit) error {
ts := c.Author.When
commit := types.Commit{Author: c.Author.Name, Message: c.Message, TimeStamp: ts}
commits = append(commits, commit)
return nil
})
cs.Commits = commits
return cs, nil
}