-
-
Notifications
You must be signed in to change notification settings - Fork 16
/
git.go
68 lines (57 loc) · 1.88 KB
/
git.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
package chglog
import (
"errors"
"fmt"
"strings"
"github.com/go-git/go-git/v5"
"github.com/go-git/go-git/v5/plumbing"
"github.com/go-git/go-git/v5/plumbing/object"
)
var errReachedToCommit = errors.New("reached to commit")
// GitRepo open a GitRepo to use to build the changelog from.
func GitRepo(gitPath string, detectDotGit bool) (*git.Repository, error) {
return git.PlainOpenWithOptions(gitPath, &git.PlainOpenOptions{
DetectDotGit: detectDotGit,
})
}
// GitHashFotTag return the git sha for a particular tag.
func GitHashFotTag(gitRepo *git.Repository, tagName string) (hash plumbing.Hash, err error) {
var ref *plumbing.Reference
ref, err = gitRepo.Tag(tagName)
if errors.Is(err, git.ErrTagNotFound) && !strings.HasPrefix(tagName, "v") {
ref, err = gitRepo.Tag("v" + tagName)
}
if err != nil {
return plumbing.ZeroHash, fmt.Errorf("error getting commit for tag %s: %w", tagName, err)
}
return ref.Hash(), nil
}
// CommitsBetween return the list of commits between two commits.
func CommitsBetween(gitRepo *git.Repository, start, end plumbing.Hash, excludeMergeCommits bool) (commits []*object.Commit, err error) {
var commitIter object.CommitIter
if commitIter, err = gitRepo.Log(&git.LogOptions{
From: start,
Order: git.LogOrderCommitterTime,
}); err != nil {
return nil, fmt.Errorf("error getting commits between %v & %v: %w", start, end, err)
}
defer commitIter.Close()
err = commitIter.ForEach(func(c *object.Commit) error {
// If no previous tag is found then from and to are equal
if end == start {
return nil
}
if c.Hash == end {
return errReachedToCommit
}
if excludeMergeCommits && len(c.ParentHashes) > 1 {
return nil
}
commits = append(commits, c)
return nil
})
if err != nil && !errors.Is(err, errReachedToCommit) {
return nil, fmt.Errorf("error getting commits between %v & %v: %w", start, end, err)
}
return commits, nil
}