ugit/internal/git/repo.go

280 lines
5.4 KiB
Go

package git
import (
"encoding/json"
"errors"
"os"
"path/filepath"
"sort"
"strings"
"time"
"github.com/go-git/go-git/v5"
"github.com/go-git/go-git/v5/plumbing"
"github.com/go-git/go-git/v5/plumbing/object"
)
type Repo struct {
path string
Meta RepoMeta
}
func (r Repo) Name() string {
return strings.TrimSuffix(filepath.Base(r.path), ".git")
}
func NewRepo(dir, name string) (*Repo, error) {
if !strings.HasSuffix(name, ".git") {
name += ".git"
}
r := &Repo{
path: filepath.Join(dir, name),
}
_, err := os.Stat(r.path)
if err != nil {
return nil, err
}
if err := ensureJSONFile(r.metaPath()); err != nil {
return nil, err
}
fi, err := os.Open(r.metaPath())
if err != nil {
return nil, err
}
defer fi.Close()
if err := json.NewDecoder(fi).Decode(&r.Meta); err != nil {
return nil, err
}
return r, nil
}
// DefaultBranch returns the branch referenced by HEAD, setting it if needed
func (r Repo) DefaultBranch() (string, error) {
repo, err := r.Git()
if err != nil {
return "", err
}
ref, err := repo.Head()
if err != nil {
if !errors.Is(err, plumbing.ErrReferenceNotFound) {
return "", err
}
brs, err := repo.Branches()
if err != nil {
return "", err
}
defer brs.Close()
fb, err := brs.Next()
if err != nil {
return "", err
}
// Rename the default branch to the first branch available
ref = fb
sym := plumbing.NewSymbolicReference(plumbing.HEAD, fb.Name())
if err := repo.Storer.SetReference(sym); err != nil {
return "", err
}
}
return strings.TrimPrefix(ref.Name().String(), "refs/heads/"), nil
}
// Git allows access to the git repository
func (r Repo) Git() (*git.Repository, error) {
return git.PlainOpen(r.path)
}
// Commit is a git commit
type Commit struct {
SHA string
Message string
Signature string
Author string
Email string
When time.Time
}
func (c Commit) Short() string {
return c.SHA[:8]
}
func (c Commit) Summary() string {
return strings.Split(c.Message, "\n")[0]
}
func (c Commit) Details() string {
return strings.Join(strings.Split(c.Message, "\n")[1:], "\n")
}
// Commit gets a specific commit by SHA
func (r Repo) Commit(sha string) (Commit, error) {
repo, err := r.Git()
if err != nil {
return Commit{}, err
}
return commit(repo, sha)
}
// LastCommit returns the last commit of the repo
func (r Repo) LastCommit() (Commit, error) {
repo, err := r.Git()
if err != nil {
return Commit{}, err
}
head, err := repo.Head()
if err != nil {
return Commit{}, err
}
return commit(repo, head.Hash().String())
}
func commit(repo *git.Repository, sha string) (Commit, error) {
obj, err := repo.CommitObject(plumbing.NewHash(sha))
if err != nil {
return Commit{}, err
}
return Commit{
SHA: obj.Hash.String(),
Message: obj.Message,
Signature: obj.PGPSignature,
Author: obj.Author.Name,
Email: obj.Author.Email,
When: obj.Author.When,
}, nil
}
// Branches is all repo branches, default first and sorted alphabetically after that
func (r Repo) Branches() ([]string, error) {
repo, err := r.Git()
if err != nil {
return nil, err
}
def, err := r.DefaultBranch()
if err != nil {
return nil, err
}
brs, err := repo.Branches()
if err != nil {
return nil, err
}
var branches []string
if err := brs.ForEach(func(branch *plumbing.Reference) error {
branches = append(branches, branch.Name().Short())
return nil
}); err != nil {
return nil, err
}
sort.Slice(branches, func(i, j int) bool {
return branches[i] == def || branches[i] < branches[j]
})
return branches, nil
}
// Tag is a git tag, which may or may not have an annotation/signature
type Tag struct {
Name string
Annotation string
Signature string
When time.Time
}
// Tags is all repo tags, sorted by time descending
func (r Repo) Tags() ([]Tag, error) {
repo, err := r.Git()
if err != nil {
return nil, err
}
tgs, err := repo.Tags()
if err != nil {
return nil, err
}
var tags []Tag
if err := tgs.ForEach(func(tag *plumbing.Reference) error {
obj, err := repo.TagObject(tag.Hash())
switch err {
case nil:
tags = append(tags, Tag{
Name: obj.Name,
Annotation: obj.Message,
Signature: obj.PGPSignature,
When: obj.Tagger.When,
})
case plumbing.ErrObjectNotFound:
commit, err := repo.CommitObject(tag.Hash())
if err != nil {
return err
}
tags = append(tags, Tag{
Name: tag.Name().Short(),
Annotation: commit.Message,
Signature: commit.PGPSignature,
When: commit.Author.When,
})
default:
return err
}
return nil
}); err != nil {
return nil, err
}
sort.Slice(tags, func(i, j int) bool {
return tags[i].When.After(tags[j].When)
})
return tags, nil
}
// Commits returns commits from a specific hash in descending order
func (r Repo) Commits(ref string) ([]Commit, error) {
repo, err := r.Git()
if err != nil {
return nil, err
}
hash, err := repo.ResolveRevision(plumbing.Revision(ref))
if err != nil {
return nil, err
}
cmts, err := repo.Log(&git.LogOptions{
From: *hash,
})
if err != nil {
return nil, err
}
var commits []Commit
if err := cmts.ForEach(func(commit *object.Commit) error {
commits = append(commits, Commit{
SHA: commit.Hash.String(),
Message: commit.Message,
Signature: commit.PGPSignature,
Author: commit.Author.Name,
Email: commit.Author.Email,
When: commit.Author.When,
})
return nil
}); err != nil {
return nil, err
}
return commits, nil
}