mirror of
https://github.com/jesseduffield/lazygit.git
synced 2025-01-22 05:29:44 +02:00
862ebd25cb
We're: * using concurrency with wait groups * avoiding regex * processing lines of input as they come rather than storing everything in one string * avoiding an inner loop by creating a mapping of remote names to branches
120 lines
2.9 KiB
Go
120 lines
2.9 KiB
Go
package git_commands
|
|
|
|
import (
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/jesseduffield/generics/slices"
|
|
gogit "github.com/jesseduffield/go-git/v5"
|
|
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
|
"github.com/jesseduffield/lazygit/pkg/commands/oscommands"
|
|
"github.com/jesseduffield/lazygit/pkg/common"
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
|
)
|
|
|
|
type RemoteLoader struct {
|
|
*common.Common
|
|
cmd oscommands.ICmdObjBuilder
|
|
getGoGitRemotes func() ([]*gogit.Remote, error)
|
|
}
|
|
|
|
func NewRemoteLoader(
|
|
common *common.Common,
|
|
cmd oscommands.ICmdObjBuilder,
|
|
getGoGitRemotes func() ([]*gogit.Remote, error),
|
|
) *RemoteLoader {
|
|
return &RemoteLoader{
|
|
Common: common,
|
|
cmd: cmd,
|
|
getGoGitRemotes: getGoGitRemotes,
|
|
}
|
|
}
|
|
|
|
func (self *RemoteLoader) GetRemotes() ([]*models.Remote, error) {
|
|
wg := sync.WaitGroup{}
|
|
wg.Add(1)
|
|
|
|
var remoteBranchesByRemoteName map[string][]*models.RemoteBranch
|
|
var remoteBranchesErr error
|
|
go utils.Safe(func() {
|
|
defer wg.Done()
|
|
|
|
remoteBranchesByRemoteName, remoteBranchesErr = self.getRemoteBranchesByRemoteName()
|
|
})
|
|
|
|
goGitRemotes, err := self.getGoGitRemotes()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
wg.Wait()
|
|
|
|
if remoteBranchesErr != nil {
|
|
return nil, remoteBranchesErr
|
|
}
|
|
|
|
remotes := slices.Map(goGitRemotes, func(goGitRemote *gogit.Remote) *models.Remote {
|
|
remoteName := goGitRemote.Config().Name
|
|
branches := remoteBranchesByRemoteName[remoteName]
|
|
|
|
return &models.Remote{
|
|
Name: goGitRemote.Config().Name,
|
|
Urls: goGitRemote.Config().URLs,
|
|
Branches: branches,
|
|
}
|
|
})
|
|
|
|
// now lets sort our remotes by name alphabetically
|
|
slices.SortFunc(remotes, func(a, b *models.Remote) bool {
|
|
// we want origin at the top because we'll be most likely to want it
|
|
if a.Name == "origin" {
|
|
return true
|
|
}
|
|
if b.Name == "origin" {
|
|
return false
|
|
}
|
|
return strings.ToLower(a.Name) < strings.ToLower(b.Name)
|
|
})
|
|
|
|
return remotes, nil
|
|
}
|
|
|
|
func (self *RemoteLoader) getRemoteBranchesByRemoteName() (map[string][]*models.RemoteBranch, error) {
|
|
remoteBranchesByRemoteName := make(map[string][]*models.RemoteBranch)
|
|
|
|
cmdArgs := NewGitCmd("branch").Arg("-r").ToArgv()
|
|
err := self.cmd.New(cmdArgs).DontLog().RunAndProcessLines(func(line string) (bool, error) {
|
|
// excluding lines like 'origin/HEAD -> origin/master' (there will be a separate
|
|
// line for 'origin/master')
|
|
if strings.Contains(line, "->") {
|
|
return false, nil
|
|
}
|
|
|
|
line = strings.TrimSpace(line)
|
|
|
|
split := strings.SplitN(line, "/", 2)
|
|
if len(split) != 2 {
|
|
return false, nil
|
|
}
|
|
remoteName := split[0]
|
|
name := split[1]
|
|
|
|
_, ok := remoteBranchesByRemoteName[remoteName]
|
|
if !ok {
|
|
remoteBranchesByRemoteName[remoteName] = []*models.RemoteBranch{}
|
|
}
|
|
|
|
remoteBranchesByRemoteName[remoteName] = append(remoteBranchesByRemoteName[remoteName],
|
|
&models.RemoteBranch{
|
|
Name: name,
|
|
RemoteName: remoteName,
|
|
})
|
|
return false, nil
|
|
})
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return remoteBranchesByRemoteName, nil
|
|
}
|