2020-11-07 16:52:09 +00:00
|
|
|
package git
|
|
|
|
|
|
|
|
import (
|
2020-12-23 11:05:36 +00:00
|
|
|
"fmt"
|
|
|
|
|
2020-11-07 16:52:09 +00:00
|
|
|
"github.com/go-git/go-git/v5"
|
|
|
|
"github.com/go-git/go-git/v5/plumbing"
|
|
|
|
"github.com/go-git/go-git/v5/plumbing/object"
|
|
|
|
"github.com/go-git/go-git/v5/plumbing/storer"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
head = "HEAD"
|
|
|
|
|
|
|
|
defaultCommitCount = 10
|
|
|
|
)
|
|
|
|
|
2021-03-29 13:57:06 +00:00
|
|
|
// Repository is an abstraction for git-repository
|
2020-11-07 16:52:09 +00:00
|
|
|
type Repository interface {
|
2021-01-05 09:15:31 +00:00
|
|
|
Log(fromRev, toRev string) ([]Commit, error)
|
2020-11-07 16:52:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type repo struct {
|
|
|
|
r *git.Repository
|
|
|
|
}
|
|
|
|
|
|
|
|
type stopFn func(*object.Commit) error
|
|
|
|
|
2021-03-29 13:57:06 +00:00
|
|
|
// NewRepository return Repository from path
|
2020-11-07 16:52:09 +00:00
|
|
|
func NewRepository(path string) (Repository, error) {
|
|
|
|
r, err := git.PlainOpen(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &repo{
|
|
|
|
r: r,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2021-03-29 13:57:06 +00:00
|
|
|
// Log return all commits between <from revision> and <to revision>
|
2021-01-05 09:15:31 +00:00
|
|
|
func (r *repo) Log(fromRev, toRev string) ([]Commit, error) {
|
2020-11-10 16:03:28 +00:00
|
|
|
if fromRev == "" {
|
|
|
|
fromRev = head
|
|
|
|
}
|
|
|
|
|
|
|
|
fromHash, err := r.r.ResolveRevision(plumbing.Revision(fromRev))
|
|
|
|
if err != nil {
|
2020-12-23 11:05:36 +00:00
|
|
|
return nil, fmt.Errorf("failed to resolve %s: %w", fromRev, err)
|
2020-11-10 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if toRev == "" {
|
2021-01-05 09:20:32 +00:00
|
|
|
return r.logWithStopFn(fromHash, nil, nil)
|
2020-11-10 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
toHash, err := r.r.ResolveRevision(plumbing.Revision(toRev))
|
|
|
|
if err != nil {
|
2020-12-23 11:05:36 +00:00
|
|
|
return nil, fmt.Errorf("failed to resolve %s: %w", toRev, err)
|
2020-11-10 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
2021-01-05 09:20:32 +00:00
|
|
|
return r.logWithStopFn(fromHash, nil, stopAtHash(toHash))
|
2020-11-07 16:52:09 +00:00
|
|
|
}
|
|
|
|
|
2021-08-20 19:01:19 +00:00
|
|
|
func (r *repo) logWithStopFn(fromHash *plumbing.Hash, beginFn, endFn stopFn) ([]Commit, error) {
|
2020-11-10 16:03:28 +00:00
|
|
|
cIter, err := r.r.Log(&git.LogOptions{
|
|
|
|
From: *fromHash,
|
|
|
|
})
|
|
|
|
if err != nil {
|
2020-12-23 11:05:36 +00:00
|
|
|
return nil, fmt.Errorf("failed to git log: %w", err)
|
2020-11-10 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
commits := make([]Commit, 0, defaultCommitCount)
|
|
|
|
|
2021-08-20 19:01:19 +00:00
|
|
|
if err := cIter.ForEach(newIterFn(&commits, beginFn, endFn)); err != nil {
|
2020-12-23 11:05:36 +00:00
|
|
|
return nil, fmt.Errorf("failed to iterate each git log: %w", err)
|
2020-11-10 16:03:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return commits, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func stopAtHash(hash *plumbing.Hash) stopFn {
|
|
|
|
return func(c *object.Commit) error {
|
|
|
|
if c.Hash == *hash {
|
|
|
|
return storer.ErrStop
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
2021-08-20 19:01:19 +00:00
|
|
|
|
|
|
|
func newIterFn(commits *[]Commit, beginFn, endFn stopFn) func(c *object.Commit) error {
|
|
|
|
if beginFn == nil && endFn == nil {
|
|
|
|
return func(c *object.Commit) error {
|
|
|
|
commit := newCommit(c)
|
|
|
|
*commits = append(*commits, commit)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if beginFn == nil {
|
|
|
|
return func(c *object.Commit) error {
|
|
|
|
commit := newCommit(c)
|
|
|
|
*commits = append(*commits, commit)
|
|
|
|
|
|
|
|
if err := endFn(c); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if endFn == nil {
|
|
|
|
return func(c *object.Commit) error {
|
|
|
|
if err := beginFn(c); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
commit := newCommit(c)
|
|
|
|
*commits = append(*commits, commit)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return func(c *object.Commit) error {
|
|
|
|
if err := beginFn(c); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
commit := newCommit(c)
|
|
|
|
*commits = append(*commits, commit)
|
|
|
|
|
|
|
|
if err := endFn(c); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|