2018-08-12 19:31:27 +10:00
package commands
import (
"os"
2019-11-05 12:42:07 +11:00
"path/filepath"
2018-08-12 19:50:55 +10:00
"strings"
2019-02-18 21:29:43 +11:00
2019-02-11 21:30:27 +11:00
"github.com/go-errors/errors"
2022-08-07 09:44:50 +10:00
"github.com/sasha-s/go-deadlock"
2019-02-11 21:30:27 +11:00
2020-10-06 20:50:54 +11:00
gogit "github.com/jesseduffield/go-git/v5"
2022-01-08 14:00:36 +11:00
"github.com/jesseduffield/lazygit/pkg/commands/git_commands"
2021-10-23 09:52:19 +11:00
"github.com/jesseduffield/lazygit/pkg/commands/git_config"
2020-09-29 19:10:57 +10:00
"github.com/jesseduffield/lazygit/pkg/commands/oscommands"
2020-08-15 11:18:40 +10:00
"github.com/jesseduffield/lazygit/pkg/commands/patch"
2021-12-29 11:37:15 +11:00
"github.com/jesseduffield/lazygit/pkg/common"
2020-09-27 16:17:26 +10:00
"github.com/jesseduffield/lazygit/pkg/env"
2018-08-12 21:04:47 +10:00
"github.com/jesseduffield/lazygit/pkg/utils"
2018-08-12 19:31:27 +10:00
)
2022-01-02 10:34:33 +11:00
// GitCommand is our main git interface
type GitCommand struct {
2022-01-08 14:00:36 +11:00
Branch * git_commands . BranchCommands
Commit * git_commands . CommitCommands
Config * git_commands . ConfigCommands
Custom * git_commands . CustomCommands
File * git_commands . FileCommands
Flow * git_commands . FlowCommands
Patch * git_commands . PatchCommands
Rebase * git_commands . RebaseCommands
Remote * git_commands . RemoteCommands
Stash * git_commands . StashCommands
Status * git_commands . StatusCommands
Submodule * git_commands . SubmoduleCommands
Sync * git_commands . SyncCommands
Tag * git_commands . TagCommands
WorkingTree * git_commands . WorkingTreeCommands
2022-01-19 18:32:27 +11:00
Bisect * git_commands . BisectCommands
2022-01-07 20:36:11 +11:00
Loaders Loaders
2022-01-02 10:34:33 +11:00
}
2019-11-17 12:07:36 +11:00
2022-01-02 10:21:32 +11:00
type Loaders struct {
2022-11-10 21:19:29 -05:00
BranchLoader * git_commands . BranchLoader
CommitFileLoader * git_commands . CommitFileLoader
CommitLoader * git_commands . CommitLoader
FileLoader * git_commands . FileLoader
ReflogCommitLoader * git_commands . ReflogCommitLoader
RemoteLoader * git_commands . RemoteLoader
StashLoader * git_commands . StashLoader
TagLoader * git_commands . TagLoader
2022-01-02 10:21:32 +11:00
}
2021-10-23 09:52:19 +11:00
func NewGitCommand (
2021-12-29 11:37:15 +11:00
cmn * common . Common ,
2022-12-26 23:43:08 +09:00
version * git_commands . GitVersion ,
2021-10-23 09:52:19 +11:00
osCommand * oscommands . OSCommand ,
gitConfig git_config . IGitConfig ,
2022-08-07 09:44:50 +10:00
syncMutex * deadlock . Mutex ,
2021-10-23 09:52:19 +11:00
) ( * GitCommand , error ) {
2020-09-27 15:36:04 +10:00
if err := navigateToRepoRootDirectory ( os . Stat , os . Chdir ) ; err != nil {
return nil , err
}
2022-04-07 18:45:08 +02:00
repo , err := setupRepository ( gogit . PlainOpenWithOptions , gogit . PlainOpenOptions { DetectDotGit : false , EnableDotGitCommonDir : true } , cmn . Tr . GitconfigParseErr )
2022-01-02 10:34:33 +11:00
if err != nil {
2020-09-27 15:36:04 +10:00
return nil , err
2018-09-02 17:15:27 +02:00
}
2022-09-13 18:11:03 +08:00
dotGitDir , err := findDotGitDir ( os . Stat , os . ReadFile )
2019-05-12 17:04:32 +10:00
if err != nil {
return nil , err
}
2022-01-02 10:34:33 +11:00
return NewGitCommandAux (
cmn ,
2022-12-26 23:43:08 +09:00
version ,
2022-01-02 10:34:33 +11:00
osCommand ,
gitConfig ,
dotGitDir ,
repo ,
2022-01-16 14:46:53 +11:00
syncMutex ,
2022-01-02 10:34:33 +11:00
) , nil
}
2021-12-29 14:33:38 +11:00
2022-01-02 10:34:33 +11:00
func NewGitCommandAux (
cmn * common . Common ,
2022-12-26 23:43:08 +09:00
version * git_commands . GitVersion ,
2022-01-02 10:34:33 +11:00
osCommand * oscommands . OSCommand ,
gitConfig git_config . IGitConfig ,
dotGitDir string ,
repo * gogit . Repository ,
2022-08-07 09:44:50 +10:00
syncMutex * deadlock . Mutex ,
2022-01-02 10:34:33 +11:00
) * GitCommand {
cmd := NewGitCmdObjBuilder ( cmn . Log , osCommand . Cmd )
2019-11-05 18:10:47 +11:00
2022-01-07 14:45:18 +11:00
// 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.
2022-01-18 21:26:21 +11:00
// common ones are: cmn, osCommand, dotGitDir, configCommands
2022-01-08 14:44:07 +11:00
configCommands := git_commands . NewConfigCommands ( cmn , gitConfig , repo )
2022-01-18 21:26:21 +11:00
2022-11-10 21:19:29 -05:00
fileLoader := git_commands . NewFileLoader ( cmn , cmd , configCommands )
2022-12-26 23:43:08 +09:00
gitCommon := git_commands . NewGitCommon ( cmn , version , cmd , osCommand , dotGitDir , repo , configCommands , syncMutex )
2022-01-18 21:26:21 +11:00
statusCommands := git_commands . NewStatusCommands ( gitCommon )
flowCommands := git_commands . NewFlowCommands ( gitCommon )
remoteCommands := git_commands . NewRemoteCommands ( gitCommon )
branchCommands := git_commands . NewBranchCommands ( gitCommon )
syncCommands := git_commands . NewSyncCommands ( gitCommon )
tagCommands := git_commands . NewTagCommands ( gitCommon )
commitCommands := git_commands . NewCommitCommands ( gitCommon )
customCommands := git_commands . NewCustomCommands ( gitCommon )
fileCommands := git_commands . NewFileCommands ( gitCommon )
submoduleCommands := git_commands . NewSubmoduleCommands ( gitCommon )
workingTreeCommands := git_commands . NewWorkingTreeCommands ( gitCommon , submoduleCommands , fileLoader )
rebaseCommands := git_commands . NewRebaseCommands ( gitCommon , commitCommands , workingTreeCommands )
stashCommands := git_commands . NewStashCommands ( gitCommon , fileLoader , workingTreeCommands )
2023-03-19 16:09:03 +11:00
// TODO: have patch builder take workingTreeCommands in its entirety
patchBuilder := patch . NewPatchBuilder ( cmn . Log , workingTreeCommands . ApplyPatch ,
2023-02-03 20:20:20 +01:00
func ( from string , to string , reverse bool , filename string , plain bool ) ( string , error ) {
2023-03-19 16:09:03 +11:00
// TODO: make patch builder take Gui.IgnoreWhitespaceInDiffView into
2023-02-03 20:20:20 +01:00
// account. For now we just pass false.
return workingTreeCommands . ShowFileDiff ( from , to , reverse , filename , plain , false )
} )
2023-03-19 16:09:03 +11:00
patchCommands := git_commands . NewPatchCommands ( gitCommon , rebaseCommands , commitCommands , statusCommands , stashCommands , patchBuilder )
2022-01-19 18:32:27 +11:00
bisectCommands := git_commands . NewBisectCommands ( gitCommon )
2022-01-02 10:34:33 +11:00
2022-10-16 21:31:42 +09:00
branchLoader := git_commands . NewBranchLoader ( cmn , branchCommands . GetRawBranches , branchCommands . CurrentBranchInfo , configCommands )
2022-11-10 21:19:29 -05:00
commitFileLoader := git_commands . NewCommitFileLoader ( cmn , cmd )
2022-10-16 21:31:42 +09:00
commitLoader := git_commands . NewCommitLoader ( cmn , cmd , dotGitDir , branchCommands . CurrentBranchInfo , statusCommands . RebaseMode )
2022-11-10 21:19:29 -05:00
reflogCommitLoader := git_commands . NewReflogCommitLoader ( cmn , cmd )
remoteLoader := git_commands . NewRemoteLoader ( cmn , cmd , repo . Remotes )
stashLoader := git_commands . NewStashLoader ( cmn , cmd )
2023-02-19 11:36:06 +01:00
tagLoader := git_commands . NewTagLoader ( cmn , cmd )
2022-11-10 21:19:29 -05:00
2022-01-02 10:34:33 +11:00
return & GitCommand {
Branch : branchCommands ,
Commit : commitCommands ,
Config : configCommands ,
2022-01-07 20:36:11 +11:00
Custom : customCommands ,
File : fileCommands ,
Flow : flowCommands ,
2022-01-02 10:34:33 +11:00
Patch : patchCommands ,
2022-01-07 20:36:11 +11:00
Rebase : rebaseCommands ,
2022-01-02 10:34:33 +11:00
Remote : remoteCommands ,
2022-01-07 20:36:11 +11:00
Stash : stashCommands ,
Status : statusCommands ,
Submodule : submoduleCommands ,
2022-01-02 10:34:33 +11:00
Sync : syncCommands ,
2022-01-07 20:36:11 +11:00
Tag : tagCommands ,
2022-01-19 18:32:27 +11:00
Bisect : bisectCommands ,
2022-01-07 20:36:11 +11:00
WorkingTree : workingTreeCommands ,
2022-01-02 10:34:33 +11:00
Loaders : Loaders {
2022-11-10 21:19:29 -05:00
BranchLoader : branchLoader ,
CommitFileLoader : commitFileLoader ,
CommitLoader : commitLoader ,
FileLoader : fileLoader ,
ReflogCommitLoader : reflogCommitLoader ,
RemoteLoader : remoteLoader ,
StashLoader : stashLoader ,
TagLoader : tagLoader ,
2022-01-02 10:34:33 +11:00
} ,
2022-01-02 10:21:32 +11:00
}
2018-09-02 17:15:27 +02:00
}
2020-09-29 20:03:39 +10:00
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 )
2020-09-29 08:47:14 +10:00
}
2018-08-12 19:50:55 +10:00
2020-09-29 20:03:39 +10:00
return nil
2018-11-29 17:57:28 +01:00
}
2020-09-29 20:03:39 +10:00
// 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)
2020-08-11 21:18:38 +10:00
2020-09-29 20:03:39 +10:00
for {
_ , err := stat ( ".git" )
2020-08-11 21:18:38 +10:00
2020-09-29 20:03:39 +10:00
if err == nil {
return nil
2018-11-25 13:15:36 +01:00
}
2018-08-12 19:50:55 +10:00
2020-09-29 20:03:39 +10:00
if ! os . IsNotExist ( err ) {
return utils . WrapError ( err )
2020-03-26 20:29:35 +11:00
}
2018-09-10 21:57:08 +02:00
2020-09-29 20:03:39 +10:00
if err = chdir ( ".." ) ; err != nil {
return utils . WrapError ( err )
2019-02-20 19:47:01 +11:00
}
2021-03-30 22:17:42 +11:00
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" )
}
2019-02-20 19:47:01 +11:00
}
2018-08-12 20:22:20 +10:00
}
2020-09-29 20:03:39 +10:00
// resolvePath takes a path containing a symlink and returns the true path
func resolvePath ( path string ) ( string , error ) {
l , err := os . Lstat ( path )
2019-03-02 20:00:26 +11:00
if err != nil {
return "" , err
}
2020-08-07 17:52:17 +10:00
2020-09-29 20:03:39 +10:00
if l . Mode ( ) & os . ModeSymlink == 0 {
return path , nil
2020-08-07 17:52:17 +10:00
}
2020-09-29 20:03:39 +10:00
return filepath . EvalSymlinks ( path )
2020-08-07 17:52:17 +10:00
}
2022-04-07 18:45:08 +02:00
func setupRepository ( openGitRepository func ( string , * gogit . PlainOpenOptions ) ( * gogit . Repository , error ) , options gogit . PlainOpenOptions , gitConfigParseErrorStr string ) ( * gogit . Repository , error ) {
2020-09-29 20:03:39 +10:00
unresolvedPath := env . GetGitDirEnv ( )
if unresolvedPath == "" {
var err error
unresolvedPath , err = os . Getwd ( )
2020-08-07 17:52:17 +10:00
if err != nil {
2020-09-29 20:03:39 +10:00
return nil , err
2020-03-28 12:52:45 +11:00
}
2019-11-04 19:47:25 +11:00
}
2020-09-29 20:03:39 +10:00
path , err := resolvePath ( unresolvedPath )
2019-02-18 21:29:43 +11:00
if err != nil {
2019-02-18 23:27:54 +11:00
return nil , err
2019-02-18 21:29:43 +11:00
}
2022-04-07 18:45:08 +02:00
repository , err := openGitRepository ( path , & options )
2019-02-24 13:51:52 +11:00
if err != nil {
2020-09-29 20:03:39 +10:00
if strings . Contains ( err . Error ( ) , ` unquoted '\' must be followed by new line ` ) {
2020-10-04 11:00:48 +11:00
return nil , errors . New ( gitConfigParseErrorStr )
2019-03-11 13:04:08 +11:00
}
2020-09-29 20:03:39 +10:00
return nil , err
2019-05-30 22:45:56 +10:00
}
2020-09-29 20:03:39 +10:00
return repository , err
2019-05-30 22:45:56 +10:00
}
2019-11-04 19:47:25 +11:00
2020-09-29 20:03:39 +10:00
func findDotGitDir ( stat func ( string ) ( os . FileInfo , error ) , readFile func ( filename string ) ( [ ] byte , error ) ) ( string , error ) {
if env . GetGitDirEnv ( ) != "" {
return env . GetGitDirEnv ( ) , nil
2019-11-04 19:47:25 +11:00
}
2020-09-29 20:03:39 +10:00
f , err := stat ( ".git" )
2019-11-04 19:47:25 +11:00
if err != nil {
2020-09-29 20:03:39 +10:00
return "" , err
2019-11-04 19:47:25 +11:00
}
2020-09-29 20:03:39 +10:00
if f . IsDir ( ) {
return ".git" , nil
2019-11-04 19:47:25 +11:00
}
2020-09-29 20:03:39 +10:00
fileBytes , err := readFile ( ".git" )
2020-03-01 21:00:44 +11:00
if err != nil {
2020-09-29 20:03:39 +10:00
return "" , err
2020-03-28 12:08:13 +11:00
}
2020-09-29 20:03:39 +10:00
fileContent := string ( fileBytes )
if ! strings . HasPrefix ( fileContent , "gitdir: " ) {
2022-04-07 18:45:08 +02:00
return "" , errors . New ( ".git is a file which suggests we are in a submodule or a worktree but the file's contents do not contain a gitdir pointing to the actual .git directory" )
2020-03-28 12:08:13 +11:00
}
2020-09-29 20:03:39 +10:00
return strings . TrimSpace ( strings . TrimPrefix ( fileContent , "gitdir: " ) ) , nil
2020-09-27 16:02:20 +10:00
}
2021-03-30 22:17:42 +11:00
func VerifyInGitRepo ( osCommand * oscommands . OSCommand ) error {
2022-01-05 11:57:32 +11:00
return osCommand . Cmd . New ( "git rev-parse --git-dir" ) . DontLog ( ) . Run ( )
2021-10-20 22:21:16 +11:00
}