1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-01-26 05:37:18 +02:00
lazygit/pkg/commands/loading_branches.go

168 lines
4.9 KiB
Go
Raw Normal View History

package commands
import (
"regexp"
"strings"
"github.com/jesseduffield/lazygit/pkg/commands/models"
2018-08-13 20:26:02 +10:00
"github.com/jesseduffield/lazygit/pkg/utils"
2018-08-23 14:22:03 +02:00
"github.com/sirupsen/logrus"
)
// context:
// we want to only show 'safe' branches (ones that haven't e.g. been deleted)
// which `git branch -a` gives us, but we also want the recency data that
// git reflog gives us.
// So we get the HEAD, then append get the reflog branches that intersect with
// our safe branches, then add the remaining safe branches, ensuring uniqueness
// along the way
// 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
2018-08-13 20:26:02 +10:00
// BranchListBuilder returns a list of Branch objects for the current repo
2018-08-12 21:04:47 +10:00
type BranchListBuilder struct {
Log *logrus.Entry
GitCommand *GitCommand
2020-09-29 18:36:54 +10:00
ReflogCommits []*models.Commit
2018-08-12 21:04:47 +10:00
}
2018-08-13 20:26:02 +10:00
// NewBranchListBuilder builds a new branch list builder
2020-09-29 18:36:54 +10:00
func NewBranchListBuilder(log *logrus.Entry, gitCommand *GitCommand, reflogCommits []*models.Commit) (*BranchListBuilder, error) {
2018-08-13 20:26:02 +10:00
return &BranchListBuilder{
Log: log,
GitCommand: gitCommand,
ReflogCommits: reflogCommits,
2018-08-13 20:26:02 +10:00
}, nil
}
func (b *BranchListBuilder) obtainBranches() []*models.Branch {
2020-03-23 21:26:24 +11:00
cmdStr := `git for-each-ref --sort=-committerdate --format="%(HEAD)|%(refname:short)|%(upstream:short)|%(upstream:track)" refs/heads`
output, err := b.GitCommand.OSCommand.RunCommandWithOutput(cmdStr)
if err != nil {
panic(err)
}
trimmedOutput := strings.TrimSpace(output)
outputLines := strings.Split(trimmedOutput, "\n")
branches := make([]*models.Branch, 0, len(outputLines))
2020-03-23 21:26:24 +11:00
for _, line := range outputLines {
if line == "" {
continue
}
split := strings.Split(line, SEPARATION_CHAR)
2021-07-26 11:07:42 +02:00
if len(split) != 4 {
// Ignore line if it isn't separated into 4 parts
// This is probably a warning message, for more info see:
// https://github.com/jesseduffield/lazygit/issues/1385#issuecomment-885580439
continue
}
name := strings.TrimPrefix(split[1], "heads/")
branch := &models.Branch{
Name: name,
Pullables: "?",
Pushables: "?",
2020-03-19 12:04:17 +11:00
Head: split[0] == "*",
}
2020-03-23 21:26:24 +11:00
2020-03-19 12:04:17 +11:00
upstreamName := split[2]
if upstreamName == "" {
2020-03-23 21:26:24 +11:00
branches = append(branches, branch)
continue
}
2020-03-23 21:26:24 +11:00
branch.UpstreamName = upstreamName
2020-03-19 12:04:17 +11:00
track := split[3]
re := regexp.MustCompile(`ahead (\d+)`)
match := re.FindStringSubmatch(track)
if len(match) > 1 {
2020-03-23 21:26:24 +11:00
branch.Pushables = match[1]
} else {
2020-03-23 21:26:24 +11:00
branch.Pushables = "0"
}
re = regexp.MustCompile(`behind (\d+)`)
match = re.FindStringSubmatch(track)
if len(match) > 1 {
2020-03-23 21:26:24 +11:00
branch.Pullables = match[1]
} else {
2020-03-23 21:26:24 +11:00
branch.Pullables = "0"
2018-08-11 16:11:17 +10:00
}
2020-03-23 21:26:24 +11:00
branches = append(branches, branch)
2018-08-11 16:11:17 +10:00
}
return branches
2018-08-11 16:11:17 +10:00
}
2018-08-13 20:26:02 +10:00
// Build the list of branches for the current repo
func (b *BranchListBuilder) Build() []*models.Branch {
branches := b.obtainBranches()
2020-03-19 12:04:17 +11:00
reflogBranches := b.obtainReflogBranches()
// loop through reflog branches. If there is a match, merge them, then remove it from the branches and keep it in the reflog branches
branchesWithRecency := make([]*models.Branch, 0)
outer:
for _, reflogBranch := range reflogBranches {
for j, branch := range branches {
2020-03-19 12:04:17 +11:00
if branch.Head {
continue
}
if strings.EqualFold(reflogBranch.Name, branch.Name) {
branch.Recency = reflogBranch.Recency
branchesWithRecency = append(branchesWithRecency, branch)
branches = append(branches[0:j], branches[j+1:]...)
continue outer
}
}
}
branches = append(branchesWithRecency, branches...)
2020-03-23 21:26:24 +11:00
foundHead := false
for i, branch := range branches {
2020-03-19 12:04:17 +11:00
if branch.Head {
2020-03-23 21:26:24 +11:00
foundHead = true
2020-03-19 12:04:17 +11:00
branch.Recency = " *"
branches = append(branches[0:i], branches[i+1:]...)
branches = append([]*models.Branch{branch}, branches...)
break
2018-08-13 20:26:02 +10:00
}
}
2020-03-23 21:26:24 +11:00
if !foundHead {
2020-03-26 20:29:35 +11:00
currentBranchName, currentBranchDisplayName, err := b.GitCommand.CurrentBranchName()
if err != nil {
panic(err)
}
branches = append([]*models.Branch{{Name: currentBranchName, DisplayName: currentBranchDisplayName, Head: true, Recency: " *"}}, branches...)
2020-03-23 21:26:24 +11:00
}
return branches
}
// TODO: only look at the new reflog commits, and otherwise store the recencies in
// int form against the branch to recalculate the time ago
func (b *BranchListBuilder) obtainReflogBranches() []*models.Branch {
foundBranchesMap := map[string]bool{}
re := regexp.MustCompile(`checkout: moving from ([\S]+) to ([\S]+)`)
reflogBranches := make([]*models.Branch, 0, len(b.ReflogCommits))
for _, commit := range b.ReflogCommits {
if match := re.FindStringSubmatch(commit.Name); len(match) == 3 {
recency := utils.UnixToTimeAgo(commit.UnixTimestamp)
for _, branchName := range match[1:] {
if !foundBranchesMap[branchName] {
foundBranchesMap[branchName] = true
reflogBranches = append(reflogBranches, &models.Branch{
Recency: recency,
Name: branchName,
})
}
}
}
}
return reflogBranches
}