2019-02-19 00:18:30 +02:00
|
|
|
package git
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/fatih/color"
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands"
|
2019-02-20 13:52:17 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/i18n"
|
2019-02-19 00:18:30 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
|
|
|
// 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
|
|
|
|
|
|
|
|
// if we find out we need to use one of these functions in the git.go file, we
|
|
|
|
// can just pull them out of here and put them there and then call them from in here
|
|
|
|
|
|
|
|
// CommitListBuilder returns a list of Branch objects for the current repo
|
|
|
|
type CommitListBuilder struct {
|
2019-02-24 08:34:19 +02:00
|
|
|
Log *logrus.Entry
|
|
|
|
GitCommand *commands.GitCommand
|
|
|
|
OSCommand *commands.OSCommand
|
|
|
|
Tr *i18n.Localizer
|
|
|
|
CherryPickedCommits []*commands.Commit
|
2019-02-19 00:18:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewCommitListBuilder builds a new commit list builder
|
2019-02-24 08:34:19 +02:00
|
|
|
func NewCommitListBuilder(log *logrus.Entry, gitCommand *commands.GitCommand, osCommand *commands.OSCommand, tr *i18n.Localizer, cherryPickedCommits []*commands.Commit) (*CommitListBuilder, error) {
|
2019-02-19 00:18:30 +02:00
|
|
|
return &CommitListBuilder{
|
2019-02-24 08:34:19 +02:00
|
|
|
Log: log,
|
|
|
|
GitCommand: gitCommand,
|
|
|
|
OSCommand: osCommand,
|
|
|
|
Tr: tr,
|
|
|
|
CherryPickedCommits: cherryPickedCommits,
|
2019-02-19 00:18:30 +02:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetCommits obtains the commits of the current branch
|
|
|
|
func (c *CommitListBuilder) GetCommits() ([]*commands.Commit, error) {
|
|
|
|
commits := []*commands.Commit{}
|
2019-02-19 14:36:29 +02:00
|
|
|
var rebasingCommits []*commands.Commit
|
2019-02-24 02:03:14 +02:00
|
|
|
isRebasing, err := c.GitCommand.IsInRebasingState()
|
2019-02-19 00:18:30 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2019-02-24 02:03:14 +02:00
|
|
|
if isRebasing {
|
2019-02-19 14:36:29 +02:00
|
|
|
// here we want to also prepend the commits that we're in the process of rebasing
|
2019-02-24 02:03:14 +02:00
|
|
|
rebasingCommits, err = c.getRebasingCommits()
|
2019-02-19 14:36:29 +02:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if len(rebasingCommits) > 0 {
|
|
|
|
commits = append(commits, rebasingCommits...)
|
|
|
|
}
|
2019-02-19 00:18:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
unpushedCommits := c.getUnpushedCommits()
|
|
|
|
log := c.getLog()
|
|
|
|
|
|
|
|
// now we can split it up and turn it into commits
|
|
|
|
for _, line := range utils.SplitLines(log) {
|
|
|
|
splitLine := strings.Split(line, " ")
|
|
|
|
sha := splitLine[0]
|
|
|
|
_, unpushed := unpushedCommits[sha]
|
|
|
|
status := map[bool]string{true: "unpushed", false: "pushed"}[unpushed]
|
|
|
|
commits = append(commits, &commands.Commit{
|
|
|
|
Sha: sha,
|
|
|
|
Name: strings.Join(splitLine[1:], " "),
|
|
|
|
Status: status,
|
|
|
|
DisplayString: strings.Join(splitLine, " "),
|
|
|
|
})
|
|
|
|
}
|
2019-02-24 02:03:14 +02:00
|
|
|
if isRebasing {
|
2019-02-19 00:18:30 +02:00
|
|
|
currentCommit := commits[len(rebasingCommits)]
|
|
|
|
blue := color.New(color.FgYellow)
|
2019-02-20 13:52:17 +02:00
|
|
|
youAreHere := blue.Sprintf("<-- %s ---", c.Tr.SLocalize("YouAreHere"))
|
2019-02-19 00:18:30 +02:00
|
|
|
currentCommit.Name = fmt.Sprintf("%s %s", youAreHere, currentCommit.Name)
|
|
|
|
}
|
2019-02-24 04:51:52 +02:00
|
|
|
|
|
|
|
commits, err = c.setCommitMergedStatuses(commits)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
commits, err = c.setCommitCherryPickStatuses(commits)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return commits, nil
|
2019-02-19 00:18:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
|
|
|
|
2019-02-24 02:03:14 +02:00
|
|
|
// getRebasingCommits takes our git-rebase-todo and our git-rebase-todo.backup files
|
2019-02-19 00:18:30 +02:00
|
|
|
// and extracts out the sha and names of commits that we still have to go
|
|
|
|
// in the rebase:
|
2019-02-24 02:03:14 +02:00
|
|
|
func (c *CommitListBuilder) getRebasingCommits() ([]*commands.Commit, error) {
|
2019-02-19 00:18:30 +02:00
|
|
|
bytesContent, err := ioutil.ReadFile(".git/rebase-merge/git-rebase-todo")
|
2019-02-19 14:36:29 +02:00
|
|
|
if err != nil {
|
|
|
|
c.Log.Info(fmt.Sprintf("error occured reading git-rebase-todo: %s", err.Error()))
|
|
|
|
// we assume an error means the file doesn't exist so we just return
|
|
|
|
return nil, nil
|
2019-02-19 00:18:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
commits := []*commands.Commit{}
|
2019-02-19 14:36:29 +02:00
|
|
|
lines := strings.Split(string(bytesContent), "\n")
|
|
|
|
for _, line := range lines {
|
2019-02-24 04:51:52 +02:00
|
|
|
if line == "" || line == "noop" {
|
2019-02-19 14:36:29 +02:00
|
|
|
return commits, nil
|
2019-02-19 00:18:30 +02:00
|
|
|
}
|
2019-02-19 14:36:29 +02:00
|
|
|
splitLine := strings.Split(line, " ")
|
2019-02-24 08:59:47 +02:00
|
|
|
commits = append([]*commands.Commit{{
|
2019-02-19 14:36:29 +02:00
|
|
|
Sha: splitLine[1][0:7],
|
|
|
|
Name: strings.Join(splitLine[2:], " "),
|
|
|
|
Status: "rebasing",
|
|
|
|
Action: splitLine[0],
|
|
|
|
}}, commits...)
|
2019-02-19 00:18:30 +02:00
|
|
|
}
|
|
|
|
|
2019-02-19 14:36:29 +02:00
|
|
|
return nil, nil
|
2019-02-19 00:18:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *CommitListBuilder) setCommitMergedStatuses(commits []*commands.Commit) ([]*commands.Commit, error) {
|
|
|
|
ancestor, err := c.getMergeBase()
|
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2019-02-24 04:51:52 +02:00
|
|
|
func (c *CommitListBuilder) setCommitCherryPickStatuses(commits []*commands.Commit) ([]*commands.Commit, error) {
|
|
|
|
for _, commit := range commits {
|
2019-02-24 08:34:19 +02:00
|
|
|
for _, cherryPickedCommit := range c.CherryPickedCommits {
|
|
|
|
if commit.Sha == cherryPickedCommit.Sha {
|
2019-02-24 04:51:52 +02:00
|
|
|
commit.Copied = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return commits, nil
|
|
|
|
}
|
|
|
|
|
2019-02-19 00:18:30 +02:00
|
|
|
func (c *CommitListBuilder) getMergeBase() (string, error) {
|
|
|
|
currentBranch, err := c.GitCommand.CurrentBranchName()
|
|
|
|
if err != nil {
|
|
|
|
return "", err
|
|
|
|
}
|
|
|
|
|
|
|
|
baseBranch := "master"
|
|
|
|
if strings.HasPrefix(currentBranch, "feature/") {
|
|
|
|
baseBranch = "develop"
|
|
|
|
}
|
|
|
|
|
|
|
|
output, err := c.OSCommand.RunCommandWithOutput(fmt.Sprintf("git merge-base HEAD %s", baseBranch))
|
|
|
|
if err != nil {
|
|
|
|
// swallowing error because it's not a big deal; probably because there are no commits yet
|
|
|
|
}
|
|
|
|
return output, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// getUnpushedCommits Returns the sha's of the commits that have not yet been pushed
|
|
|
|
// to the remote branch of the current branch, a map is returned to ease look up
|
|
|
|
func (c *CommitListBuilder) getUnpushedCommits() map[string]bool {
|
|
|
|
pushables := map[string]bool{}
|
|
|
|
o, err := c.OSCommand.RunCommandWithOutput("git rev-list @{u}..HEAD --abbrev-commit")
|
|
|
|
if err != nil {
|
|
|
|
return pushables
|
|
|
|
}
|
|
|
|
for _, p := range utils.SplitLines(o) {
|
|
|
|
pushables[p] = true
|
|
|
|
}
|
|
|
|
|
|
|
|
return pushables
|
|
|
|
}
|
|
|
|
|
|
|
|
// getLog gets the git log (currently limited to 30 commits for performance
|
|
|
|
// until we work out lazy loading
|
|
|
|
func (c *CommitListBuilder) getLog() string {
|
|
|
|
// currently limiting to 30 for performance reasons
|
|
|
|
// TODO: add lazyloading when you scroll down
|
|
|
|
result, err := c.OSCommand.RunCommandWithOutput("git log --oneline -30")
|
|
|
|
if err != nil {
|
|
|
|
// assume if there is an error there are no commits yet for this branch
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|