feat: Wrap commit and its references Create a wrapper around commit to add all reference to it.
Gabriel A. Giovanini mail@gabrielgio.me
Tue, 18 Feb 2025 18:10:12 +0100
4 files changed, 92 insertions(+), 16 deletions(-)
M git/git.go → git/git.go
@@ -32,6 +32,12 @@ ref *plumbing.Reference tag *object.Tag } +// CommitReference aggregate all the references for a given commit +type CommitReference struct { + commit *object.Commit + refs []*plumbing.Reference +} + // infoWrapper wraps the property of a TreeEntry so it can export fs.FileInfo // to tar WriteHeader type infoWrapper struct { @@ -104,27 +110,81 @@ } return &g, nil } -func (g *GitRepo) Commits() ([]*object.Commit, error) { +func (g *GitRepo) Commits() ([]*CommitReference, error) { ci, err := g.r.Log(&git.LogOptions{From: g.h}) if err != nil { return nil, fmt.Errorf("commits from ref: %w", err) } - commits := []*object.Commit{} + commitRefs := []*CommitReference{} ci.ForEach(func(c *object.Commit) error { - commits = append(commits, c) + commitRefs = append(commitRefs, &CommitReference{commit: c}) return nil }) - return commits, nil + // new we fetch for possible tags for each commit + iter, err := g.r.References() + if err != nil { + return nil, err + } + + if err := iter.ForEach(func(ref *plumbing.Reference) error { + for _, c := range commitRefs { + obj, err := g.r.TagObject(ref.Hash()) + switch err { + case nil: + if obj.Target == c.commit.Hash { + c.AddReference(ref) + } + case plumbing.ErrObjectNotFound: + if c.commit.Hash == ref.Hash() { + c.AddReference(ref) + } + default: + return err + } + } + return nil + }); err != nil { + return nil, err + } + + return commitRefs, nil } -func (g *GitRepo) LastCommit() (*object.Commit, error) { +func (g *GitRepo) LastCommit() (*CommitReference, error) { c, err := g.r.CommitObject(g.h) if err != nil { return nil, fmt.Errorf("last commit: %w", err) } - return c, nil + + iter, err := g.r.Tags() + if err != nil { + return nil, err + } + + commitRef := &CommitReference{commit: c} + if err := iter.ForEach(func(ref *plumbing.Reference) error { + obj, err := g.r.TagObject(ref.Hash()) + switch err { + case nil: + if obj.Target == commitRef.commit.Hash { + commitRef.AddReference(ref) + } + case plumbing.ErrObjectNotFound: + if commitRef.commit.Hash == ref.Hash() { + commitRef.AddReference(ref) + } + default: + return err + } + + return nil + }); err != nil { + return nil, err + } + + return commitRef, nil } func (g *GitRepo) FileContent(path string) (string, error) { @@ -342,3 +402,19 @@ return t.tag.Message } return "" } + +func (c *CommitReference) Commit() *object.Commit { + return c.commit +} + +func (c *CommitReference) HasReference() bool { + return len(c.refs) > 0 +} + +func (c *CommitReference) References() []*plumbing.Reference { + return c.refs +} + +func (c *CommitReference) AddReference(ref *plumbing.Reference) { + c.refs = append(c.refs, ref) +}
M routes/routes.go → routes/routes.go
@@ -63,8 +63,8 @@ infos = append(infos, info{ DisplayName: getDisplayName(name), Name: name, Desc: getDescription(path), - Idle: humanize.Time(c.Author.When), - d: c.Author.When, + Idle: humanize.Time(c.Commit().Author.When), + d: c.Commit().Author.When, }) }
M templates/log.html → templates/log.html
@@ -10,12 +10,12 @@ {{ $repo := .name }} <div class="log"> {{ range .commits }} <div> - <div><a href="/{{ $repo }}/commit/{{ .Hash.String }}" class="commit-hash">{{ slice .Hash.String 0 8 }}</a></div> - <pre>{{ .Message }}</pre> + <div><a href="/{{ $repo }}/commit/{{ .Commit.Hash.String }}" class="commit-hash">{{ slice .Commit.Hash.String 0 8 }}</a></div> + <pre>{{ .Commit.Message }}</pre> </div> <div class="commit-info"> - {{ .Author.Name }} <a href="mailto:{{ .Author.Email }}" class="commit-email">{{ .Author.Email }}</a> - <div>{{ .Author.When.Format "Mon, 02 Jan 2006 15:04:05 -0700" }}</div> + {{ .Commit.Author.Name }} <a href="mailto:{{ .Commit.Author.Email }}" class="commit-email">{{ .Commit.Author.Email }}</a> + <div>{{ .Commit.Author.When.Format "Mon, 02 Jan 2006 15:04:05 -0700" }}</div> </div> {{ end }} </div>
M templates/repo.html → templates/repo.html
@@ -11,12 +11,12 @@ {{ $repo := .name }} <div class="log"> {{ range .commits }} <div> - <div><a href="/{{ $repo }}/commit/{{ .Hash.String }}" class="commit-hash">{{ slice .Hash.String 0 8 }}</a></div> - <pre>{{ .Message }}</pre> + <div><a href="/{{ $repo }}/commit/{{ .Commit.Hash.String }}" class="commit-hash">{{ slice .Commit.Hash.String 0 8 }}</a></div> + <pre>{{ .Commit.Message }}</pre> </div> <div class="commit-info"> - {{ .Author.Name }} <a href="mailto:{{ .Author.Email }}" class="commit-email">{{ .Author.Email }}</a> - <div>{{ .Author.When.Format "Mon, 02 Jan 2006 15:04:05 -0700" }}</div> + {{ .Commit.Author.Name }} <a href="mailto:{{ .Commit.Author.Email }}" class="commit-email">{{ .Commit.Author.Email }}</a> + <div>{{ .Commit.Author.When.Format "Mon, 02 Jan 2006 15:04:05 -0700" }}</div> </div> {{ end }} </div>