mirror of
https://github.com/jesseduffield/lazygit.git
synced 2025-01-06 03:53:59 +02:00
264 lines
7.3 KiB
Go
264 lines
7.3 KiB
Go
package commands
|
|
|
|
import (
|
|
"io/ioutil"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/go-errors/errors"
|
|
|
|
gogit "github.com/jesseduffield/go-git/v5"
|
|
"github.com/jesseduffield/lazygit/pkg/commands/git_config"
|
|
"github.com/jesseduffield/lazygit/pkg/commands/loaders"
|
|
"github.com/jesseduffield/lazygit/pkg/commands/oscommands"
|
|
"github.com/jesseduffield/lazygit/pkg/commands/patch"
|
|
"github.com/jesseduffield/lazygit/pkg/common"
|
|
"github.com/jesseduffield/lazygit/pkg/env"
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
|
)
|
|
|
|
// GitCommand is our main git interface
|
|
type GitCommand struct {
|
|
Loaders Loaders
|
|
|
|
Cmd oscommands.ICmdObjBuilder
|
|
|
|
Submodule *SubmoduleCommands
|
|
Tag *TagCommands
|
|
WorkingTree *WorkingTreeCommands
|
|
File *FileCommands
|
|
Branch *BranchCommands
|
|
Commit *CommitCommands
|
|
Rebase *RebaseCommands
|
|
Stash *StashCommands
|
|
Status *StatusCommands
|
|
Config *ConfigCommands
|
|
Patch *PatchCommands
|
|
Remote *RemoteCommands
|
|
Sync *SyncCommands
|
|
Flow *FlowCommands
|
|
}
|
|
|
|
type Loaders struct {
|
|
Commits *loaders.CommitLoader
|
|
Branches *loaders.BranchLoader
|
|
Files *loaders.FileLoader
|
|
CommitFiles *loaders.CommitFileLoader
|
|
Remotes *loaders.RemoteLoader
|
|
ReflogCommits *loaders.ReflogCommitLoader
|
|
Stash *loaders.StashLoader
|
|
Tags *loaders.TagLoader
|
|
}
|
|
|
|
func NewGitCommand(
|
|
cmn *common.Common,
|
|
osCommand *oscommands.OSCommand,
|
|
gitConfig git_config.IGitConfig,
|
|
) (*GitCommand, error) {
|
|
if err := navigateToRepoRootDirectory(os.Stat, os.Chdir); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
repo, err := setupRepository(gogit.PlainOpen, cmn.Tr.GitconfigParseErr)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
dotGitDir, err := findDotGitDir(os.Stat, ioutil.ReadFile)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return NewGitCommandAux(
|
|
cmn,
|
|
osCommand,
|
|
gitConfig,
|
|
dotGitDir,
|
|
repo,
|
|
), nil
|
|
}
|
|
|
|
func NewGitCommandAux(
|
|
cmn *common.Common,
|
|
osCommand *oscommands.OSCommand,
|
|
gitConfig git_config.IGitConfig,
|
|
dotGitDir string,
|
|
repo *gogit.Repository,
|
|
) *GitCommand {
|
|
cmd := NewGitCmdObjBuilder(cmn.Log, osCommand.Cmd)
|
|
|
|
// here we're doing a bunch of dependency injection for each of our commands structs.
|
|
// This is admittedly messy, but allows us to test each command struct in isolation,
|
|
// and allows for better namespacing when compared to having every method living
|
|
// on the one struct.
|
|
configCommands := NewConfigCommands(cmn, gitConfig)
|
|
statusCommands := NewStatusCommands(cmn, osCommand, repo, dotGitDir)
|
|
fileLoader := loaders.NewFileLoader(cmn, cmd, configCommands)
|
|
flowCommands := NewFlowCommands(cmn, cmd, configCommands)
|
|
remoteCommands := NewRemoteCommands(cmn, cmd)
|
|
branchCommands := NewBranchCommands(cmn, cmd)
|
|
syncCommands := NewSyncCommands(cmn, cmd)
|
|
tagCommands := NewTagCommands(cmn, cmd)
|
|
commitCommands := NewCommitCommands(cmn, cmd)
|
|
fileCommands := NewFileCommands(cmn, cmd, configCommands, osCommand)
|
|
submoduleCommands := NewSubmoduleCommands(cmn, cmd, dotGitDir)
|
|
workingTreeCommands := NewWorkingTreeCommands(cmn, cmd, submoduleCommands, osCommand, fileLoader)
|
|
rebaseCommands := NewRebaseCommands(
|
|
cmn,
|
|
cmd,
|
|
osCommand,
|
|
commitCommands,
|
|
workingTreeCommands,
|
|
configCommands,
|
|
dotGitDir,
|
|
)
|
|
stashCommands := NewStashCommands(cmn, cmd, osCommand, fileLoader, workingTreeCommands)
|
|
// TODO: have patch manager take workingTreeCommands in its entirety
|
|
patchManager := patch.NewPatchManager(cmn.Log, workingTreeCommands.ApplyPatch, workingTreeCommands.ShowFileDiff)
|
|
patchCommands := NewPatchCommands(cmn, cmd, rebaseCommands, commitCommands, configCommands, statusCommands, patchManager)
|
|
|
|
return &GitCommand{
|
|
Cmd: cmd,
|
|
|
|
Submodule: submoduleCommands,
|
|
Tag: tagCommands,
|
|
WorkingTree: workingTreeCommands,
|
|
File: fileCommands,
|
|
Branch: branchCommands,
|
|
Commit: commitCommands,
|
|
Rebase: rebaseCommands,
|
|
Config: configCommands,
|
|
Stash: stashCommands,
|
|
Status: statusCommands,
|
|
Patch: patchCommands,
|
|
Remote: remoteCommands,
|
|
Sync: syncCommands,
|
|
Flow: flowCommands,
|
|
Loaders: Loaders{
|
|
Commits: loaders.NewCommitLoader(cmn, cmd, dotGitDir, branchCommands.CurrentBranchName, statusCommands.RebaseMode),
|
|
Branches: loaders.NewBranchLoader(cmn, branchCommands.GetRawBranches, branchCommands.CurrentBranchName),
|
|
Files: fileLoader,
|
|
CommitFiles: loaders.NewCommitFileLoader(cmn, cmd),
|
|
Remotes: loaders.NewRemoteLoader(cmn, cmd, repo.Remotes),
|
|
ReflogCommits: loaders.NewReflogCommitLoader(cmn, cmd),
|
|
Stash: loaders.NewStashLoader(cmn, cmd),
|
|
Tags: loaders.NewTagLoader(cmn, cmd),
|
|
},
|
|
}
|
|
}
|
|
|
|
func navigateToRepoRootDirectory(stat func(string) (os.FileInfo, error), chdir func(string) error) error {
|
|
gitDir := env.GetGitDirEnv()
|
|
if gitDir != "" {
|
|
// we've been given the git directory explicitly so no need to navigate to it
|
|
_, err := stat(gitDir)
|
|
if err != nil {
|
|
return utils.WrapError(err)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// we haven't been given the git dir explicitly so we assume it's in the current working directory as `.git/` (or an ancestor directory)
|
|
|
|
for {
|
|
_, err := stat(".git")
|
|
|
|
if err == nil {
|
|
return nil
|
|
}
|
|
|
|
if !os.IsNotExist(err) {
|
|
return utils.WrapError(err)
|
|
}
|
|
|
|
if err = chdir(".."); err != nil {
|
|
return utils.WrapError(err)
|
|
}
|
|
|
|
currentPath, err := os.Getwd()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
atRoot := currentPath == filepath.Dir(currentPath)
|
|
if atRoot {
|
|
// we should never really land here: the code that creates GitCommand should
|
|
// verify we're in a git directory
|
|
return errors.New("Must open lazygit in a git repository")
|
|
}
|
|
}
|
|
}
|
|
|
|
// resolvePath takes a path containing a symlink and returns the true path
|
|
func resolvePath(path string) (string, error) {
|
|
l, err := os.Lstat(path)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if l.Mode()&os.ModeSymlink == 0 {
|
|
return path, nil
|
|
}
|
|
|
|
return filepath.EvalSymlinks(path)
|
|
}
|
|
|
|
func setupRepository(openGitRepository func(string) (*gogit.Repository, error), gitConfigParseErrorStr string) (*gogit.Repository, error) {
|
|
unresolvedPath := env.GetGitDirEnv()
|
|
if unresolvedPath == "" {
|
|
var err error
|
|
unresolvedPath, err = os.Getwd()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
path, err := resolvePath(unresolvedPath)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
repository, err := openGitRepository(path)
|
|
|
|
if err != nil {
|
|
if strings.Contains(err.Error(), `unquoted '\' must be followed by new line`) {
|
|
return nil, errors.New(gitConfigParseErrorStr)
|
|
}
|
|
|
|
return nil, err
|
|
}
|
|
|
|
return repository, err
|
|
}
|
|
|
|
func findDotGitDir(stat func(string) (os.FileInfo, error), readFile func(filename string) ([]byte, error)) (string, error) {
|
|
if env.GetGitDirEnv() != "" {
|
|
return env.GetGitDirEnv(), nil
|
|
}
|
|
|
|
f, err := stat(".git")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
if f.IsDir() {
|
|
return ".git", nil
|
|
}
|
|
|
|
fileBytes, err := readFile(".git")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
fileContent := string(fileBytes)
|
|
if !strings.HasPrefix(fileContent, "gitdir: ") {
|
|
return "", errors.New(".git is a file which suggests we are in a submodule but the file's contents do not contain a gitdir pointing to the actual .git directory")
|
|
}
|
|
return strings.TrimSpace(strings.TrimPrefix(fileContent, "gitdir: ")), nil
|
|
}
|
|
|
|
func VerifyInGitRepo(osCommand *oscommands.OSCommand) error {
|
|
return osCommand.Cmd.New("git rev-parse --git-dir").DontLog().Run()
|
|
}
|