1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-01-24 05:36:19 +02:00
lazygit/pkg/commands/git_commands/commit_loader.go

455 lines
13 KiB
Go
Raw Normal View History

package git_commands
import (
"bytes"
"fmt"
"os"
"path/filepath"
"regexp"
"strconv"
"strings"
"github.com/fsmiamoto/git-todo-parser/todo"
2022-03-19 16:34:46 +11:00
"github.com/jesseduffield/generics/slices"
"github.com/jesseduffield/lazygit/pkg/commands/models"
"github.com/jesseduffield/lazygit/pkg/commands/oscommands"
2021-12-30 13:35:10 +11:00
"github.com/jesseduffield/lazygit/pkg/commands/types/enums"
2021-12-29 14:33:38 +11:00
"github.com/jesseduffield/lazygit/pkg/common"
"github.com/jesseduffield/lazygit/pkg/gui/style"
)
// context:
// here we get the commits from git log but format them to show whether they're
// unpushed/pushed/merged into the base branch or not, or if they're yet to
// be processed as part of a rebase (these won't appear in git log but we
// grab them from the rebase-related files in the .git directory to show them
2021-12-30 13:35:10 +11:00
// CommitLoader returns a list of Commit objects for the current repo
type CommitLoader struct {
2021-12-29 14:33:38 +11:00
*common.Common
cmd oscommands.ICmdObjBuilder
getCurrentBranchInfo func() (BranchInfo, error)
2021-12-30 13:35:10 +11:00
getRebaseMode func() (enums.RebaseMode, error)
2021-12-29 14:33:38 +11:00
readFile func(filename string) ([]byte, error)
2021-12-30 11:22:29 +11:00
walkFiles func(root string, fn filepath.WalkFunc) error
2021-12-29 14:33:38 +11:00
dotGitDir string
}
2021-12-30 13:44:41 +11:00
// making our dependencies explicit for the sake of easier testing
2021-12-30 13:35:10 +11:00
func NewCommitLoader(
2021-12-30 13:44:41 +11:00
cmn *common.Common,
2022-01-02 10:34:33 +11:00
cmd oscommands.ICmdObjBuilder,
dotGitDir string,
getCurrentBranchInfo func() (BranchInfo, error),
2022-01-02 10:34:33 +11:00
getRebaseMode func() (enums.RebaseMode, error),
2021-12-30 13:35:10 +11:00
) *CommitLoader {
return &CommitLoader{
2021-12-30 13:44:41 +11:00
Common: cmn,
2022-01-02 10:34:33 +11:00
cmd: cmd,
getCurrentBranchInfo: getCurrentBranchInfo,
2022-01-02 10:34:33 +11:00
getRebaseMode: getRebaseMode,
2022-09-13 18:11:03 +08:00
readFile: os.ReadFile,
2021-12-30 13:44:41 +11:00
walkFiles: filepath.Walk,
2022-01-02 10:34:33 +11:00
dotGitDir: dotGitDir,
2020-08-22 08:49:02 +10:00
}
}
type GetCommitsOptions struct {
2020-08-22 08:49:02 +10:00
Limit bool
FilterPath string
IncludeRebaseCommits bool
RefName string // e.g. "HEAD" or "my_branch"
2021-11-02 21:16:00 +11:00
// determines if we show the whole git graph i.e. pass the '--all' flag
All bool
}
// GetCommits obtains the commits of the current branch
2021-12-30 13:35:10 +11:00
func (self *CommitLoader) GetCommits(opts GetCommitsOptions) ([]*models.Commit, error) {
2020-09-29 18:36:54 +10:00
commits := []*models.Commit{}
var rebasingCommits []*models.Commit
2021-12-30 11:22:29 +11:00
rebaseMode, err := self.getRebaseMode()
if err != nil {
return nil, err
}
2020-08-22 08:49:02 +10:00
if opts.IncludeRebaseCommits && opts.FilterPath == "" {
2020-08-22 08:49:02 +10:00
var err error
2021-12-30 11:22:29 +11:00
rebasingCommits, err = self.MergeRebasingCommits(commits)
if err != nil {
return nil, err
}
commits = append(commits, rebasingCommits...)
}
2020-08-27 17:05:07 +10:00
passedFirstPushedCommit := false
2021-12-30 11:22:29 +11:00
firstPushedCommit, err := self.getFirstPushedCommit(opts.RefName)
if err != nil {
2020-08-27 17:05:07 +10:00
// must have no upstream branch so we'll consider everything as pushed
passedFirstPushedCommit = true
}
2021-12-30 11:22:29 +11:00
err = self.getLogCmd(opts).RunAndProcessLines(func(line string) (bool, error) {
commit := self.extractCommitFromLine(line)
if commit.Sha == firstPushedCommit {
passedFirstPushedCommit = true
}
commit.Status = map[bool]string{true: "unpushed", false: "pushed"}[!passedFirstPushedCommit]
commits = append(commits, commit)
return false, nil
})
if err != nil {
return nil, err
}
2021-12-30 13:11:58 +11:00
if len(commits) == 0 {
return commits, nil
}
2021-12-30 13:35:10 +11:00
if rebaseMode != enums.REBASE_MODE_NONE {
currentCommit := commits[len(rebasingCommits)]
2021-12-30 11:22:29 +11:00
youAreHere := style.FgYellow.Sprintf("<-- %s ---", self.Tr.YouAreHere)
currentCommit.Name = fmt.Sprintf("%s %s", youAreHere, currentCommit.Name)
}
2019-02-24 13:51:52 +11:00
2021-12-30 11:22:29 +11:00
commits, err = self.setCommitMergedStatuses(opts.RefName, commits)
2019-02-24 13:51:52 +11:00
if err != nil {
return nil, err
}
return commits, nil
}
2021-12-30 13:35:10 +11:00
func (self *CommitLoader) MergeRebasingCommits(commits []*models.Commit) ([]*models.Commit, error) {
2021-12-30 13:11:58 +11:00
// chances are we have as many commits as last time so we'll set the capacity to be the old length
result := make([]*models.Commit, 0, len(commits))
for i, commit := range commits {
if commit.Status != "rebasing" { // removing the existing rebase commits so we can add the refreshed ones
result = append(result, commits[i:]...)
break
}
}
rebaseMode, err := self.getRebaseMode()
if err != nil {
return nil, err
}
2021-12-30 13:35:10 +11:00
if rebaseMode == enums.REBASE_MODE_NONE {
2021-12-30 13:11:58 +11:00
// not in rebase mode so return original commits
return result, nil
}
rebasingCommits, err := self.getHydratedRebasingCommits(rebaseMode)
if err != nil {
return nil, err
}
if len(rebasingCommits) > 0 {
result = append(rebasingCommits, result...)
}
return result, nil
}
// extractCommitFromLine takes a line from a git log and extracts the sha, message, date, and tag if present
// then puts them into a commit object
// example input:
// 8ad01fe32fcc20f07bc6693f87aa4977c327f1e1|10 hours ago|Jesse Duffield| (HEAD -> master, tag: v0.15.2)|refresh commits when adding a tag
2021-12-30 13:35:10 +11:00
func (self *CommitLoader) extractCommitFromLine(line string) *models.Commit {
2022-05-08 14:23:29 +10:00
split := strings.SplitN(line, "\x00", 7)
2021-12-30 13:11:58 +11:00
sha := split[0]
unixTimestamp := split[1]
2022-05-08 14:23:29 +10:00
authorName := split[2]
authorEmail := split[3]
extraInfo := strings.TrimSpace(split[4])
parentHashes := split[5]
message := split[6]
2021-12-30 13:11:58 +11:00
tags := []string{}
if extraInfo != "" {
re := regexp.MustCompile(`tag: ([^,\)]+)`)
tagMatch := re.FindStringSubmatch(extraInfo)
if len(tagMatch) > 1 {
tags = append(tags, tagMatch[1])
}
}
unitTimestampInt, _ := strconv.Atoi(unixTimestamp)
2022-03-26 22:03:32 +09:00
parents := []string{}
if len(parentHashes) > 0 {
parents = strings.Split(parentHashes, " ")
}
2021-12-30 13:11:58 +11:00
return &models.Commit{
Sha: sha,
Name: message,
Tags: tags,
ExtraInfo: extraInfo,
UnixTimestamp: int64(unitTimestampInt),
2022-05-08 14:23:29 +10:00
AuthorName: authorName,
AuthorEmail: authorEmail,
2022-03-26 22:03:32 +09:00
Parents: parents,
2021-12-30 13:11:58 +11:00
}
}
2021-12-30 13:35:10 +11:00
func (self *CommitLoader) getHydratedRebasingCommits(rebaseMode enums.RebaseMode) ([]*models.Commit, error) {
2021-12-30 11:22:29 +11:00
commits, err := self.getRebasingCommits(rebaseMode)
2021-10-30 17:42:52 +11:00
if err != nil {
return nil, err
}
2021-10-30 18:06:39 +11:00
if len(commits) == 0 {
return nil, nil
}
2022-03-19 16:34:46 +11:00
commitShas := slices.Map(commits, func(commit *models.Commit) string {
return commit.Sha
})
2021-10-30 17:42:52 +11:00
// note that we're not filtering these as we do non-rebasing commits just because
// I suspect that will cause some damage
2021-12-30 11:22:29 +11:00
cmdObj := self.cmd.New(
2021-10-30 17:42:52 +11:00
fmt.Sprintf(
"git -c log.showSignature=false show %s --no-patch --oneline %s --abbrev=%d",
2021-10-30 17:42:52 +11:00
strings.Join(commitShas, " "),
prettyFormat,
20,
),
2022-01-05 11:57:32 +11:00
).DontLog()
2021-10-30 17:42:52 +11:00
hydratedCommits := make([]*models.Commit, 0, len(commits))
i := 0
2021-12-30 11:22:29 +11:00
err = cmdObj.RunAndProcessLines(func(line string) (bool, error) {
commit := self.extractCommitFromLine(line)
matchingCommit := commits[i]
commit.Action = matchingCommit.Action
commit.Status = matchingCommit.Status
hydratedCommits = append(hydratedCommits, commit)
i++
2021-10-30 17:42:52 +11:00
return false, nil
})
if err != nil {
return nil, err
}
return hydratedCommits, nil
}
// getRebasingCommits obtains the commits that we're in the process of rebasing
2021-12-30 13:35:10 +11:00
func (self *CommitLoader) getRebasingCommits(rebaseMode enums.RebaseMode) ([]*models.Commit, error) {
switch rebaseMode {
2021-12-30 13:35:10 +11:00
case enums.REBASE_MODE_MERGING:
2021-12-30 11:22:29 +11:00
return self.getNormalRebasingCommits()
2021-12-30 13:35:10 +11:00
case enums.REBASE_MODE_INTERACTIVE:
2021-12-30 11:22:29 +11:00
return self.getInteractiveRebasingCommits()
default:
return nil, nil
}
}
2021-12-30 13:35:10 +11:00
func (self *CommitLoader) getNormalRebasingCommits() ([]*models.Commit, error) {
rewrittenCount := 0
2021-12-30 11:22:29 +11:00
bytesContent, err := self.readFile(filepath.Join(self.dotGitDir, "rebase-apply/rewritten"))
if err == nil {
content := string(bytesContent)
rewrittenCount = len(strings.Split(content, "\n"))
}
// we know we're rebasing, so lets get all the files whose names have numbers
2020-09-29 18:36:54 +10:00
commits := []*models.Commit{}
2021-12-30 11:22:29 +11:00
err = self.walkFiles(filepath.Join(self.dotGitDir, "rebase-apply"), func(path string, f os.FileInfo, err error) error {
if rewrittenCount > 0 {
rewrittenCount--
return nil
}
if err != nil {
return err
}
re := regexp.MustCompile(`^\d+$`)
if !re.MatchString(f.Name()) {
return nil
}
2021-12-30 11:22:29 +11:00
bytesContent, err := self.readFile(path)
if err != nil {
return err
}
content := string(bytesContent)
2022-01-08 15:46:35 +11:00
commit := self.commitFromPatch(content)
2020-09-29 18:36:54 +10:00
commits = append([]*models.Commit{commit}, commits...)
return nil
})
if err != nil {
return nil, err
}
return commits, nil
}
// git-rebase-todo example:
// pick ac446ae94ee560bdb8d1d057278657b251aaef17 ac446ae
// pick afb893148791a2fbd8091aeb81deba4930c73031 afb8931
// git-rebase-todo.backup example:
// pick 49cbba374296938ea86bbd4bf4fee2f6ba5cccf6 third commit on master
// pick ac446ae94ee560bdb8d1d057278657b251aaef17 blah commit on master
// pick afb893148791a2fbd8091aeb81deba4930c73031 fourth commit on master
// getInteractiveRebasingCommits takes our git-rebase-todo and our git-rebase-todo.backup files
// and extracts out the sha and names of commits that we still have to go
// in the rebase:
2021-12-30 13:35:10 +11:00
func (self *CommitLoader) getInteractiveRebasingCommits() ([]*models.Commit, error) {
2021-12-30 11:22:29 +11:00
bytesContent, err := self.readFile(filepath.Join(self.dotGitDir, "rebase-merge/git-rebase-todo"))
if err != nil {
2021-12-30 11:22:29 +11:00
self.Log.Error(fmt.Sprintf("error occurred reading git-rebase-todo: %s", err.Error()))
// we assume an error means the file doesn't exist so we just return
return nil, nil
}
2020-09-29 18:36:54 +10:00
commits := []*models.Commit{}
todos, err := todo.Parse(bytes.NewBuffer(bytesContent))
if err != nil {
self.Log.Error(fmt.Sprintf("error occurred while parsing git-rebase-todo file: %s", err.Error()))
return nil, nil
}
for _, t := range todos {
if t.Commit == "" {
// Command does not have a commit associated, skip
2020-04-22 11:15:41 +10:00
continue
}
commits = slices.Prepend(commits, &models.Commit{
Sha: t.Commit,
Name: t.Msg,
Status: "rebasing",
Action: t.Command.String(),
})
}
2020-04-22 11:21:20 +10:00
return commits, nil
}
// assuming the file starts like this:
// From e93d4193e6dd45ca9cf3a5a273d7ba6cd8b8fb20 Mon Sep 17 00:00:00 2001
// From: Lazygit Tester <test@example.com>
// Date: Wed, 5 Dec 2018 21:03:23 +1100
// Subject: second commit on master
2022-01-08 15:46:35 +11:00
func (self *CommitLoader) commitFromPatch(content string) *models.Commit {
lines := strings.Split(content, "\n")
sha := strings.Split(lines[0], " ")[1]
name := strings.TrimPrefix(lines[3], "Subject: ")
2020-09-29 18:36:54 +10:00
return &models.Commit{
Sha: sha,
Name: name,
Status: "rebasing",
2022-01-08 15:46:35 +11:00
}
}
2021-12-30 13:35:10 +11:00
func (self *CommitLoader) setCommitMergedStatuses(refName string, commits []*models.Commit) ([]*models.Commit, error) {
2021-12-30 11:22:29 +11:00
ancestor, err := self.getMergeBase(refName)
if err != nil {
return nil, err
}
if ancestor == "" {
return commits, nil
}
passedAncestor := false
for i, commit := range commits {
if strings.HasPrefix(ancestor, commit.Sha) {
passedAncestor = true
}
if commit.Status != "pushed" {
continue
}
if passedAncestor {
commits[i].Status = "merged"
}
}
return commits, nil
}
2021-12-30 13:35:10 +11:00
func (self *CommitLoader) getMergeBase(refName string) (string, error) {
info, err := self.getCurrentBranchInfo()
if err != nil {
return "", err
}
baseBranch := "master"
if strings.HasPrefix(info.RefName, "feature/") {
baseBranch = "develop"
}
// swallowing error because it's not a big deal; probably because there are no commits yet
2022-01-05 11:57:32 +11:00
output, _ := self.cmd.New(fmt.Sprintf("git merge-base %s %s", self.cmd.Quote(refName), self.cmd.Quote(baseBranch))).DontLog().RunWithOutput()
return ignoringWarnings(output), nil
}
func ignoringWarnings(commandOutput string) string {
trimmedOutput := strings.TrimSpace(commandOutput)
split := strings.Split(trimmedOutput, "\n")
// need to get last line in case the first line is a warning about how the error is ambiguous.
// At some point we should find a way to make it unambiguous
lastLine := split[len(split)-1]
return lastLine
}
// getFirstPushedCommit returns the first commit SHA which has been pushed to the ref's upstream.
// all commits above this are deemed unpushed and marked as such.
2021-12-30 13:35:10 +11:00
func (self *CommitLoader) getFirstPushedCommit(refName string) (string, error) {
2021-12-30 11:22:29 +11:00
output, err := self.cmd.
New(
fmt.Sprintf("git merge-base %s %s@{u}",
self.cmd.Quote(refName),
self.cmd.Quote(strings.TrimPrefix(refName, "refs/heads/"))),
2021-12-30 11:22:29 +11:00
).
2022-01-05 11:57:32 +11:00
DontLog().
2021-12-30 11:22:29 +11:00
RunWithOutput()
if err != nil {
return "", err
}
return ignoringWarnings(output), nil
}
2020-01-11 18:23:35 +11:00
// getLog gets the git log.
2021-12-30 13:35:10 +11:00
func (self *CommitLoader) getLogCmd(opts GetCommitsOptions) oscommands.ICmdObj {
2020-01-11 18:23:35 +11:00
limitFlag := ""
2020-08-22 08:49:02 +10:00
if opts.Limit {
2021-12-30 13:11:58 +11:00
limitFlag = " -300"
2020-01-11 18:23:35 +11:00
}
2020-03-29 10:11:15 +11:00
filterFlag := ""
2020-08-22 08:49:02 +10:00
if opts.FilterPath != "" {
2021-12-30 11:22:29 +11:00
filterFlag = fmt.Sprintf(" --follow -- %s", self.cmd.Quote(opts.FilterPath))
}
2021-12-30 11:22:29 +11:00
config := self.UserConfig.Git.Log
2021-11-02 20:05:23 +11:00
orderFlag := ""
if config.Order != "default" {
orderFlag = " --" + config.Order
}
2021-11-02 21:16:00 +11:00
allFlag := ""
if opts.All {
allFlag = " --all"
}
2021-11-02 20:05:23 +11:00
2021-12-30 11:22:29 +11:00
return self.cmd.New(
fmt.Sprintf(
"git -c log.showSignature=false log %s%s%s --oneline %s%s --abbrev=%d%s",
2021-12-30 11:22:29 +11:00
self.cmd.Quote(opts.RefName),
2021-11-02 20:05:23 +11:00
orderFlag,
2021-11-02 21:16:00 +11:00
allFlag,
2021-10-30 17:42:52 +11:00
prettyFormat,
limitFlag,
40,
filterFlag,
),
2022-01-05 11:57:32 +11:00
).DontLog()
}
2021-10-30 17:42:52 +11:00
2023-01-04 22:13:11 +09:00
const prettyFormat = `--pretty=format:"%H%x00%at%x00%aN%x00%ae%x00%d%x00%p%x00%s"`