1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2024-12-14 11:23:09 +02:00
lazygit/pkg/commands/git_test.go

2143 lines
52 KiB
Go
Raw Normal View History

package commands
import (
2018-08-29 22:55:57 +02:00
"fmt"
"io/ioutil"
"os"
2018-08-27 22:41:46 +02:00
"os/exec"
"regexp"
"runtime"
"testing"
"time"
2019-05-12 09:04:32 +02:00
"github.com/go-errors/errors"
2020-10-06 11:50:54 +02:00
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/config"
2018-08-29 22:55:57 +02:00
"github.com/jesseduffield/lazygit/pkg/i18n"
"github.com/jesseduffield/lazygit/pkg/secureexec"
"github.com/jesseduffield/lazygit/pkg/test"
"github.com/jesseduffield/lazygit/pkg/utils"
"github.com/stretchr/testify/assert"
)
type fileInfoMock struct {
name string
size int64
fileMode os.FileMode
fileModTime time.Time
isDir bool
sys interface{}
}
2018-11-30 02:47:14 +02:00
// Name is a function.
func (f fileInfoMock) Name() string {
return f.name
}
2018-11-30 02:47:14 +02:00
// Size is a function.
func (f fileInfoMock) Size() int64 {
return f.size
}
2018-11-30 02:47:14 +02:00
// Mode is a function.
func (f fileInfoMock) Mode() os.FileMode {
return f.fileMode
}
2018-11-30 02:47:14 +02:00
// ModTime is a function.
func (f fileInfoMock) ModTime() time.Time {
return f.fileModTime
}
2018-11-30 02:47:14 +02:00
// IsDir is a function.
func (f fileInfoMock) IsDir() bool {
return f.isDir
}
2018-11-30 02:47:14 +02:00
// Sys is a function.
func (f fileInfoMock) Sys() interface{} {
return f.sys
}
2018-11-30 02:47:14 +02:00
// TestNavigateToRepoRootDirectory is a function.
func TestNavigateToRepoRootDirectory(t *testing.T) {
type scenario struct {
2018-09-05 22:16:26 +02:00
testName string
stat func(string) (os.FileInfo, error)
chdir func(string) error
test func(error)
}
scenarios := []scenario{
2018-08-29 22:55:57 +02:00
{
2018-09-05 22:16:26 +02:00
"Navigate to git repository",
func(string) (os.FileInfo, error) {
return fileInfoMock{isDir: true}, nil
},
func(string) error {
return nil
},
func(err error) {
assert.NoError(t, err)
},
},
{
2018-09-05 22:16:26 +02:00
"An error occurred when getting path informations",
func(string) (os.FileInfo, error) {
return nil, fmt.Errorf("An error occurred")
2018-08-29 22:55:57 +02:00
},
func(string) error {
return nil
2018-08-29 22:55:57 +02:00
},
func(err error) {
assert.Error(t, err)
assert.EqualError(t, err, "An error occurred")
2018-08-29 22:55:57 +02:00
},
},
{
2018-09-05 22:16:26 +02:00
"An error occurred when trying to move one path backward",
func(string) (os.FileInfo, error) {
return nil, os.ErrNotExist
2018-08-29 22:55:57 +02:00
},
func(string) error {
return fmt.Errorf("An error occurred")
2018-08-29 22:55:57 +02:00
},
func(err error) {
assert.Error(t, err)
assert.EqualError(t, err, "An error occurred")
2018-08-29 22:55:57 +02:00
},
},
}
for _, s := range scenarios {
2018-09-05 22:16:26 +02:00
t.Run(s.testName, func(t *testing.T) {
s.test(navigateToRepoRootDirectory(s.stat, s.chdir))
})
}
}
2020-09-27 07:36:04 +02:00
// TestSetupRepository is a function.
func TestSetupRepository(t *testing.T) {
type scenario struct {
2018-09-05 22:16:26 +02:00
testName string
openGitRepository func(string) (*gogit.Repository, error)
2020-10-04 02:00:48 +02:00
errorStr string
2020-09-27 07:36:04 +02:00
test func(*gogit.Repository, error)
}
2018-08-29 22:55:57 +02:00
scenarios := []scenario{
{
2018-09-05 22:16:26 +02:00
"A gitconfig parsing error occurred",
func(string) (*gogit.Repository, error) {
return nil, fmt.Errorf(`unquoted '\' must be followed by new line`)
},
2020-10-04 02:00:48 +02:00
"error translated",
2020-09-27 07:36:04 +02:00
func(r *gogit.Repository, err error) {
assert.Error(t, err)
assert.EqualError(t, err, "error translated")
},
},
{
2018-09-05 22:16:26 +02:00
"A gogit error occurred",
func(string) (*gogit.Repository, error) {
return nil, fmt.Errorf("Error from inside gogit")
},
2020-10-04 02:00:48 +02:00
"",
2020-09-27 07:36:04 +02:00
func(r *gogit.Repository, err error) {
assert.Error(t, err)
assert.EqualError(t, err, "Error from inside gogit")
},
},
{
2018-09-05 22:16:26 +02:00
"Setup done properly",
func(string) (*gogit.Repository, error) {
assert.NoError(t, os.RemoveAll("/tmp/lazygit-test"))
r, err := gogit.PlainInit("/tmp/lazygit-test", false)
assert.NoError(t, err)
return r, nil
},
2020-10-04 02:00:48 +02:00
"",
2020-09-27 07:36:04 +02:00
func(r *gogit.Repository, err error) {
assert.NoError(t, err)
2018-09-05 22:16:26 +02:00
assert.NotNil(t, r)
},
},
}
for _, s := range scenarios {
2018-09-05 22:16:26 +02:00
t.Run(s.testName, func(t *testing.T) {
2020-10-04 02:00:48 +02:00
s.test(setupRepository(s.openGitRepository, s.errorStr))
2018-09-05 22:16:26 +02:00
})
}
}
2018-11-30 02:47:14 +02:00
// TestNewGitCommand is a function.
2018-09-02 17:18:33 +02:00
func TestNewGitCommand(t *testing.T) {
actual, err := os.Getwd()
assert.NoError(t, err)
defer func() {
assert.NoError(t, os.Chdir(actual))
}()
type scenario struct {
2018-09-05 22:16:26 +02:00
testName string
setup func()
test func(*GitCommand, error)
2018-09-02 17:18:33 +02:00
}
scenarios := []scenario{
{
2018-09-05 22:16:26 +02:00
"An error occurred, folder doesn't contains a git repository",
2018-09-02 17:18:33 +02:00
func() {
assert.NoError(t, os.Chdir("/tmp"))
},
func(gitCmd *GitCommand, err error) {
assert.Error(t, err)
2021-03-30 13:17:42 +02:00
assert.Regexp(t, `Must open lazygit in a git repository`, err.Error())
2018-09-02 17:18:33 +02:00
},
},
{
2018-09-05 22:16:26 +02:00
"New GitCommand object created",
2018-09-02 17:18:33 +02:00
func() {
assert.NoError(t, os.RemoveAll("/tmp/lazygit-test"))
_, err := gogit.PlainInit("/tmp/lazygit-test", false)
assert.NoError(t, err)
assert.NoError(t, os.Chdir("/tmp/lazygit-test"))
},
func(gitCmd *GitCommand, err error) {
assert.NoError(t, err)
},
},
}
for _, s := range scenarios {
2018-09-05 22:16:26 +02:00
t.Run(s.testName, func(t *testing.T) {
s.setup()
2020-10-04 02:00:48 +02:00
s.test(NewGitCommand(utils.NewDummyLog(), oscommands.NewDummyOSCommand(), i18n.NewTranslationSet(utils.NewDummyLog()), config.NewDummyAppConfig()))
2018-09-05 22:16:26 +02:00
})
2018-09-02 17:18:33 +02:00
}
}
2018-11-30 02:47:14 +02:00
// TestGitCommandGetStashEntries is a function.
2018-08-27 22:41:46 +02:00
func TestGitCommandGetStashEntries(t *testing.T) {
type scenario struct {
2018-09-05 22:16:26 +02:00
testName string
command func(string, ...string) *exec.Cmd
2020-09-29 10:46:45 +02:00
test func([]*models.StashEntry)
}
2018-08-27 22:41:46 +02:00
scenarios := []scenario{
{
2018-09-05 22:16:26 +02:00
"No stash entries found",
2018-08-27 22:41:46 +02:00
func(string, ...string) *exec.Cmd {
return secureexec.Command("echo")
2018-08-27 22:41:46 +02:00
},
2020-09-29 10:46:45 +02:00
func(entries []*models.StashEntry) {
2018-08-27 22:41:46 +02:00
assert.Len(t, entries, 0)
},
},
{
2018-09-05 22:16:26 +02:00
"Several stash entries found",
2018-08-27 22:41:46 +02:00
func(string, ...string) *exec.Cmd {
return secureexec.Command("echo", "WIP on add-pkg-commands-test: 55c6af2 increase parallel build\nWIP on master: bb86a3f update github template")
2018-08-27 22:41:46 +02:00
},
2020-09-29 10:46:45 +02:00
func(entries []*models.StashEntry) {
expected := []*models.StashEntry{
2018-08-27 22:41:46 +02:00
{
2020-10-03 06:54:55 +02:00
Index: 0,
Name: "WIP on add-pkg-commands-test: 55c6af2 increase parallel build",
2018-08-27 22:41:46 +02:00
},
{
2020-10-03 06:54:55 +02:00
Index: 1,
Name: "WIP on master: bb86a3f update github template",
2018-08-27 22:41:46 +02:00
},
}
assert.Len(t, entries, 2)
assert.EqualValues(t, expected, entries)
},
},
}
for _, s := range scenarios {
2018-09-05 22:16:26 +02:00
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2018-08-27 22:41:46 +02:00
s.test(gitCmd.GetStashEntries(""))
2018-09-05 22:16:26 +02:00
})
2018-08-27 22:41:46 +02:00
}
}
2018-11-30 02:47:14 +02:00
// TestGitCommandGetStatusFiles is a function.
2018-09-06 23:05:35 +02:00
func TestGitCommandGetStatusFiles(t *testing.T) {
2018-08-27 23:20:44 +02:00
type scenario struct {
2018-09-05 22:16:26 +02:00
testName string
command func(string, ...string) *exec.Cmd
2020-09-29 10:45:00 +02:00
test func([]*models.File)
2018-08-27 23:20:44 +02:00
}
scenarios := []scenario{
{
2018-09-05 22:16:26 +02:00
"No files found",
2018-08-27 23:20:44 +02:00
func(cmd string, args ...string) *exec.Cmd {
return secureexec.Command("echo")
2018-08-27 23:20:44 +02:00
},
2020-09-29 10:45:00 +02:00
func(files []*models.File) {
2018-08-27 23:20:44 +02:00
assert.Len(t, files, 0)
},
},
{
2018-09-05 22:16:26 +02:00
"Several files found",
2018-08-27 23:20:44 +02:00
func(cmd string, args ...string) *exec.Cmd {
return secureexec.Command(
2018-08-27 23:20:44 +02:00
"echo",
"MM file1.txt\nA file3.txt\nAM file2.txt\n?? file4.txt\nUU file5.txt",
2018-08-27 23:20:44 +02:00
)
},
2020-09-29 10:45:00 +02:00
func(files []*models.File) {
assert.Len(t, files, 5)
2018-08-27 23:20:44 +02:00
2020-09-29 10:45:00 +02:00
expected := []*models.File{
2018-08-27 23:20:44 +02:00
{
Name: "file1.txt",
HasStagedChanges: true,
HasUnstagedChanges: true,
Tracked: true,
2021-03-20 03:07:11 +02:00
Added: false,
Deleted: false,
HasMergeConflicts: false,
HasInlineMergeConflicts: false,
DisplayString: "MM file1.txt",
Type: "other",
ShortStatus: "MM",
2018-08-27 23:20:44 +02:00
},
{
Name: "file3.txt",
HasStagedChanges: true,
HasUnstagedChanges: false,
Tracked: false,
2021-03-20 03:07:11 +02:00
Added: true,
Deleted: false,
HasMergeConflicts: false,
HasInlineMergeConflicts: false,
DisplayString: "A file3.txt",
Type: "other",
ShortStatus: "A ",
2018-08-27 23:20:44 +02:00
},
{
Name: "file2.txt",
HasStagedChanges: true,
HasUnstagedChanges: true,
Tracked: false,
2021-03-20 03:07:11 +02:00
Added: true,
Deleted: false,
HasMergeConflicts: false,
HasInlineMergeConflicts: false,
DisplayString: "AM file2.txt",
Type: "other",
ShortStatus: "AM",
2018-08-27 23:20:44 +02:00
},
{
Name: "file4.txt",
HasStagedChanges: false,
HasUnstagedChanges: true,
Tracked: false,
2021-03-20 03:07:11 +02:00
Added: true,
Deleted: false,
HasMergeConflicts: false,
HasInlineMergeConflicts: false,
DisplayString: "?? file4.txt",
Type: "other",
ShortStatus: "??",
},
{
Name: "file5.txt",
HasStagedChanges: false,
HasUnstagedChanges: true,
Tracked: true,
2021-03-20 03:07:11 +02:00
Added: false,
Deleted: false,
HasMergeConflicts: true,
HasInlineMergeConflicts: true,
DisplayString: "UU file5.txt",
Type: "other",
ShortStatus: "UU",
2018-08-27 23:20:44 +02:00
},
}
assert.EqualValues(t, expected, files)
},
},
}
for _, s := range scenarios {
2018-09-05 22:16:26 +02:00
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2018-08-27 23:20:44 +02:00
2020-08-07 09:52:17 +02:00
s.test(gitCmd.GetStatusFiles(GetStatusFileOptions{}))
2018-09-05 22:16:26 +02:00
})
2018-08-27 23:20:44 +02:00
}
}
2018-11-30 02:47:14 +02:00
// TestGitCommandStashDo is a function.
func TestGitCommandStashDo(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"stash", "drop", "stash@{1}"}, args)
return secureexec.Command("echo")
}
assert.NoError(t, gitCmd.StashDo(1, "drop"))
}
2018-11-30 02:47:14 +02:00
// TestGitCommandStashSave is a function.
func TestGitCommandStashSave(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"stash", "save", "A stash message"}, args)
return secureexec.Command("echo")
}
assert.NoError(t, gitCmd.StashSave("A stash message"))
}
2018-11-30 02:47:14 +02:00
// TestGitCommandCommitAmend is a function.
func TestGitCommandCommitAmend(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"commit", "--amend", "--allow-empty"}, args)
return secureexec.Command("echo")
}
_, err := gitCmd.PrepareCommitAmendSubProcess().CombinedOutput()
assert.NoError(t, err)
}
2018-12-07 09:52:31 +02:00
// TestGitCommandGetCommitDifferences is a function.
func TestGitCommandGetCommitDifferences(t *testing.T) {
type scenario struct {
testName string
command func(string, ...string) *exec.Cmd
test func(string, string)
}
scenarios := []scenario{
{
"Can't retrieve pushable count",
func(string, ...string) *exec.Cmd {
return secureexec.Command("test")
},
func(pushableCount string, pullableCount string) {
assert.EqualValues(t, "?", pushableCount)
assert.EqualValues(t, "?", pullableCount)
},
},
{
"Can't retrieve pullable count",
func(cmd string, args ...string) *exec.Cmd {
if args[1] == "HEAD..@{u}" {
return secureexec.Command("test")
}
return secureexec.Command("echo")
},
func(pushableCount string, pullableCount string) {
assert.EqualValues(t, "?", pushableCount)
assert.EqualValues(t, "?", pullableCount)
},
},
{
"Retrieve pullable and pushable count",
func(cmd string, args ...string) *exec.Cmd {
if args[1] == "HEAD..@{u}" {
return secureexec.Command("echo", "10")
}
return secureexec.Command("echo", "11")
},
func(pushableCount string, pullableCount string) {
assert.EqualValues(t, "11", pushableCount)
assert.EqualValues(t, "10", pullableCount)
},
},
}
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2018-12-07 09:52:31 +02:00
s.test(gitCmd.GetCommitDifferences("HEAD", "@{u}"))
})
}
}
2018-11-30 02:47:14 +02:00
// TestGitCommandRenameCommit is a function.
func TestGitCommandRenameCommit(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2021-04-06 10:00:47 +02:00
assert.EqualValues(t, []string{"commit", "--allow-empty", "--amend", "--only", "-m", "test"}, args)
return secureexec.Command("echo")
}
assert.NoError(t, gitCmd.RenameCommit("test"))
}
2018-11-30 02:47:14 +02:00
// TestGitCommandResetToCommit is a function.
func TestGitCommandResetToCommit(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"reset", "--hard", "78976bc"}, args)
return secureexec.Command("echo")
}
2020-09-29 12:23:58 +02:00
assert.NoError(t, gitCmd.ResetToCommit("78976bc", "hard", oscommands.RunCommandOptions{}))
}
2018-11-30 02:47:14 +02:00
// TestGitCommandNewBranch is a function.
func TestGitCommandNewBranch(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"checkout", "-b", "test", "master"}, args)
return secureexec.Command("echo")
}
assert.NoError(t, gitCmd.NewBranch("test", "master"))
}
2018-11-30 02:47:14 +02:00
// TestGitCommandDeleteBranch is a function.
func TestGitCommandDeleteBranch(t *testing.T) {
type scenario struct {
testName string
branch string
force bool
command func(string, ...string) *exec.Cmd
test func(error)
}
scenarios := []scenario{
{
"Delete a branch",
"test",
false,
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"branch", "-d", "test"}, args)
return secureexec.Command("echo")
},
func(err error) {
assert.NoError(t, err)
},
},
{
"Force delete a branch",
"test",
true,
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"branch", "-D", "test"}, args)
return secureexec.Command("echo")
},
func(err error) {
assert.NoError(t, err)
},
},
}
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
s.test(gitCmd.DeleteBranch(s.branch, s.force))
})
}
}
2018-11-30 02:47:14 +02:00
// TestGitCommandMerge is a function.
func TestGitCommandMerge(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"merge", "--no-edit", "test"}, args)
return secureexec.Command("echo")
}
2020-08-11 13:18:38 +02:00
assert.NoError(t, gitCmd.Merge("test", MergeOpts{}))
}
2018-11-30 02:47:14 +02:00
// TestGitCommandUsingGpg is a function.
func TestGitCommandUsingGpg(t *testing.T) {
type scenario struct {
2020-12-21 00:38:36 +02:00
testName string
getGitConfigValue func(string) (string, error)
test func(bool)
}
scenarios := []scenario{
{
"Option global and local config commit.gpgsign is not set",
2020-12-21 00:38:36 +02:00
func(string) (string, error) { return "", nil },
func(gpgEnabled bool) {
assert.False(t, gpgEnabled)
},
},
{
"Option commit.gpgsign is true",
func(string) (string, error) {
return "True", nil
},
func(gpgEnabled bool) {
assert.True(t, gpgEnabled)
},
},
{
"Option commit.gpgsign is on",
func(string) (string, error) {
return "ON", nil
},
func(gpgEnabled bool) {
assert.True(t, gpgEnabled)
},
},
{
"Option commit.gpgsign is yes",
func(string) (string, error) {
return "YeS", nil
},
func(gpgEnabled bool) {
assert.True(t, gpgEnabled)
},
},
{
"Option commit.gpgsign is 1",
func(string) (string, error) {
return "1", nil
},
func(gpgEnabled bool) {
assert.True(t, gpgEnabled)
},
},
}
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-12-21 00:38:36 +02:00
gitCmd.getGitConfigValue = s.getGitConfigValue
s.test(gitCmd.usingGpg())
})
}
}
2018-11-30 02:47:14 +02:00
// TestGitCommandCommit is a function.
func TestGitCommandCommit(t *testing.T) {
type scenario struct {
2020-12-21 00:38:36 +02:00
testName string
command func(string, ...string) *exec.Cmd
getGitConfigValue func(string) (string, error)
test func(*exec.Cmd, error)
flags string
}
scenarios := []scenario{
{
"Commit using gpg",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "bash", cmd)
assert.EqualValues(t, []string{"-c", "git commit -m \"test\""}, args)
return secureexec.Command("echo")
},
func(string) (string, error) {
return "true", nil
},
func(cmd *exec.Cmd, err error) {
assert.NotNil(t, cmd)
assert.Nil(t, err)
},
"",
},
{
"Commit without using gpg",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"commit", "-m", "test"}, args)
return secureexec.Command("echo")
},
func(string) (string, error) {
return "false", nil
},
func(cmd *exec.Cmd, err error) {
assert.Nil(t, cmd)
assert.Nil(t, err)
},
"",
},
{
"Commit with --no-verify flag",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"commit", "--no-verify", "-m", "test"}, args)
return secureexec.Command("echo")
},
func(string) (string, error) {
return "false", nil
},
func(cmd *exec.Cmd, err error) {
assert.Nil(t, cmd)
assert.Nil(t, err)
},
"--no-verify",
},
{
"Commit without using gpg with an error",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"commit", "-m", "test"}, args)
return secureexec.Command("test")
},
func(string) (string, error) {
return "false", nil
},
func(cmd *exec.Cmd, err error) {
assert.Nil(t, cmd)
assert.Error(t, err)
},
"",
},
}
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-12-21 00:38:36 +02:00
gitCmd.getGitConfigValue = s.getGitConfigValue
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
s.test(gitCmd.Commit("test", s.flags))
})
}
}
// TestGitCommandAmendHead is a function.
func TestGitCommandAmendHead(t *testing.T) {
type scenario struct {
2020-12-21 00:38:36 +02:00
testName string
command func(string, ...string) *exec.Cmd
getGitConfigValue func(string) (string, error)
test func(*exec.Cmd, error)
}
scenarios := []scenario{
{
"Amend commit using gpg",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "bash", cmd)
assert.EqualValues(t, []string{"-c", "git commit --amend --no-edit --allow-empty"}, args)
return secureexec.Command("echo")
},
func(string) (string, error) {
return "true", nil
},
func(cmd *exec.Cmd, err error) {
assert.NotNil(t, cmd)
assert.Nil(t, err)
},
},
{
"Amend commit without using gpg",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2019-11-05 10:03:36 +02:00
assert.EqualValues(t, []string{"commit", "--amend", "--no-edit", "--allow-empty"}, args)
return secureexec.Command("echo")
},
func(string) (string, error) {
return "false", nil
},
func(cmd *exec.Cmd, err error) {
assert.Nil(t, cmd)
assert.Nil(t, err)
},
},
{
"Amend commit without using gpg with an error",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2019-11-05 10:03:36 +02:00
assert.EqualValues(t, []string{"commit", "--amend", "--no-edit", "--allow-empty"}, args)
return secureexec.Command("test")
},
func(string) (string, error) {
return "false", nil
},
func(cmd *exec.Cmd, err error) {
assert.Nil(t, cmd)
assert.Error(t, err)
},
},
}
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-12-21 00:38:36 +02:00
gitCmd.getGitConfigValue = s.getGitConfigValue
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
s.test(gitCmd.AmendHead())
})
}
}
2018-11-30 02:47:14 +02:00
// TestGitCommandPush is a function.
func TestGitCommandPush(t *testing.T) {
type scenario struct {
2020-12-21 00:38:36 +02:00
testName string
getGitConfigValue func(string) (string, error)
command func(string, ...string) *exec.Cmd
forcePush bool
test func(error)
}
scenarios := []scenario{
{
"Push with force disabled, follow-tags on",
func(string) (string, error) {
return "", nil
},
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2019-11-21 12:45:18 +02:00
assert.EqualValues(t, []string{"push", "--follow-tags"}, args)
return secureexec.Command("echo")
},
false,
func(err error) {
2019-02-16 01:18:38 +02:00
assert.NoError(t, err)
},
},
{
"Push with force enabled, follow-tags on",
func(string) (string, error) {
return "", nil
},
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2019-11-21 12:45:18 +02:00
assert.EqualValues(t, []string{"push", "--follow-tags", "--force-with-lease"}, args)
return secureexec.Command("echo")
},
true,
func(err error) {
2019-02-16 01:18:38 +02:00
assert.NoError(t, err)
},
},
{
2020-12-21 00:38:36 +02:00
"Push with force disabled, follow-tags off",
func(string) (string, error) {
return "false", nil
},
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"push"}, args)
return secureexec.Command("echo")
},
false,
func(err error) {
assert.NoError(t, err)
},
},
{
"Push with an error occurring, follow-tags on",
func(string) (string, error) {
return "", nil
},
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2019-11-21 12:45:18 +02:00
assert.EqualValues(t, []string{"push", "--follow-tags"}, args)
return secureexec.Command("test")
},
false,
func(err error) {
2019-02-16 01:18:38 +02:00
assert.Error(t, err)
},
},
}
2018-10-29 09:23:56 +02:00
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2020-12-21 00:38:36 +02:00
gitCmd.getGitConfigValue = s.getGitConfigValue
err := gitCmd.Push("test", s.forcePush, "", "", func(passOrUname string) string {
return "\n"
})
2018-10-29 09:23:56 +02:00
s.test(err)
})
}
}
// TestGitCommandCatFile tests emitting a file using commands, where commands vary by OS.
2018-09-13 21:44:26 +02:00
func TestGitCommandCatFile(t *testing.T) {
var osCmd string
switch os := runtime.GOOS; os {
case "windows":
osCmd = "type"
default:
osCmd = "cat"
}
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, osCmd, cmd)
2018-09-13 21:44:26 +02:00
assert.EqualValues(t, []string{"test.txt"}, args)
return secureexec.Command("echo", "-n", "test")
2018-09-13 21:44:26 +02:00
}
o, err := gitCmd.CatFile("test.txt")
assert.NoError(t, err)
assert.Equal(t, "test", o)
}
2018-11-30 02:47:14 +02:00
// TestGitCommandStageFile is a function.
2018-09-13 21:44:26 +02:00
func TestGitCommandStageFile(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = func(cmd string, args ...string) *exec.Cmd {
2018-09-13 21:44:26 +02:00
assert.EqualValues(t, "git", cmd)
2021-03-02 06:36:04 +02:00
assert.EqualValues(t, []string{"add", "--", "test.txt"}, args)
2018-09-13 21:44:26 +02:00
return secureexec.Command("echo")
2018-09-13 21:44:26 +02:00
}
assert.NoError(t, gitCmd.StageFile("test.txt"))
}
2018-11-30 02:47:14 +02:00
// TestGitCommandUnstageFile is a function.
2018-09-13 21:44:26 +02:00
func TestGitCommandUnstageFile(t *testing.T) {
type scenario struct {
testName string
command func(string, ...string) *exec.Cmd
test func(error)
2021-03-20 23:41:06 +02:00
reset bool
2018-09-13 21:44:26 +02:00
}
scenarios := []scenario{
{
"Remove an untracked file from staging",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2021-03-02 06:36:04 +02:00
assert.EqualValues(t, []string{"rm", "--cached", "--force", "--", "test.txt"}, args)
2018-09-13 21:44:26 +02:00
return secureexec.Command("echo")
2018-09-13 21:44:26 +02:00
},
func(err error) {
assert.NoError(t, err)
},
false,
},
{
"Remove a tracked file from staging",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2021-03-02 06:36:04 +02:00
assert.EqualValues(t, []string{"reset", "HEAD", "--", "test.txt"}, args)
2018-09-13 21:44:26 +02:00
return secureexec.Command("echo")
2018-09-13 21:44:26 +02:00
},
func(err error) {
assert.NoError(t, err)
},
true,
},
}
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2021-03-20 23:41:06 +02:00
s.test(gitCmd.UnStageFile([]string{"test.txt"}, s.reset))
2018-09-13 21:44:26 +02:00
})
}
}
2019-03-18 11:44:33 +02:00
// TestGitCommandDiscardAllFileChanges is a function.
2021-03-20 03:07:11 +02:00
// these tests don't cover everything, in part because we already have an integration
// test which does cover everything. I don't want to unnecessarily assert on the 'how'
// when the 'what' is what matters
2019-03-18 11:44:33 +02:00
func TestGitCommandDiscardAllFileChanges(t *testing.T) {
type scenario struct {
testName string
command func() (func(string, ...string) *exec.Cmd, *[][]string)
test func(*[][]string, error)
2020-09-29 10:45:00 +02:00
file *models.File
removeFile func(string) error
}
scenarios := []scenario{
{
"An error occurred when resetting",
func() (func(string, ...string) *exec.Cmd, *[][]string) {
cmdsCalled := [][]string{}
return func(cmd string, args ...string) *exec.Cmd {
cmdsCalled = append(cmdsCalled, args)
return secureexec.Command("test")
}, &cmdsCalled
},
func(cmdsCalled *[][]string, err error) {
assert.Error(t, err)
assert.Len(t, *cmdsCalled, 1)
assert.EqualValues(t, *cmdsCalled, [][]string{
{"reset", "--", "test"},
})
},
2020-09-29 10:45:00 +02:00
&models.File{
Name: "test",
HasStagedChanges: true,
},
func(string) error {
return nil
},
},
{
"An error occurred when removing file",
func() (func(string, ...string) *exec.Cmd, *[][]string) {
cmdsCalled := [][]string{}
return func(cmd string, args ...string) *exec.Cmd {
cmdsCalled = append(cmdsCalled, args)
return secureexec.Command("test")
}, &cmdsCalled
},
func(cmdsCalled *[][]string, err error) {
assert.Error(t, err)
assert.EqualError(t, err, "an error occurred when removing file")
assert.Len(t, *cmdsCalled, 0)
},
2020-09-29 10:45:00 +02:00
&models.File{
Name: "test",
Tracked: false,
2021-03-20 03:07:11 +02:00
Added: true,
},
func(string) error {
return fmt.Errorf("an error occurred when removing file")
},
},
{
"An error occurred with checkout",
func() (func(string, ...string) *exec.Cmd, *[][]string) {
cmdsCalled := [][]string{}
return func(cmd string, args ...string) *exec.Cmd {
cmdsCalled = append(cmdsCalled, args)
return secureexec.Command("test")
}, &cmdsCalled
},
func(cmdsCalled *[][]string, err error) {
assert.Error(t, err)
assert.Len(t, *cmdsCalled, 1)
assert.EqualValues(t, *cmdsCalled, [][]string{
{"checkout", "--", "test"},
})
},
2020-09-29 10:45:00 +02:00
&models.File{
Name: "test",
Tracked: true,
HasStagedChanges: false,
},
func(string) error {
return nil
},
},
{
"Checkout only",
func() (func(string, ...string) *exec.Cmd, *[][]string) {
cmdsCalled := [][]string{}
return func(cmd string, args ...string) *exec.Cmd {
cmdsCalled = append(cmdsCalled, args)
return secureexec.Command("echo")
}, &cmdsCalled
},
func(cmdsCalled *[][]string, err error) {
assert.NoError(t, err)
assert.Len(t, *cmdsCalled, 1)
assert.EqualValues(t, *cmdsCalled, [][]string{
{"checkout", "--", "test"},
})
},
2020-09-29 10:45:00 +02:00
&models.File{
Name: "test",
Tracked: true,
HasStagedChanges: false,
},
func(string) error {
return nil
},
},
{
"Reset and checkout staged changes",
func() (func(string, ...string) *exec.Cmd, *[][]string) {
cmdsCalled := [][]string{}
return func(cmd string, args ...string) *exec.Cmd {
cmdsCalled = append(cmdsCalled, args)
return secureexec.Command("echo")
}, &cmdsCalled
},
func(cmdsCalled *[][]string, err error) {
assert.NoError(t, err)
assert.Len(t, *cmdsCalled, 2)
assert.EqualValues(t, *cmdsCalled, [][]string{
{"reset", "--", "test"},
{"checkout", "--", "test"},
})
},
2020-09-29 10:45:00 +02:00
&models.File{
Name: "test",
Tracked: true,
HasStagedChanges: true,
},
func(string) error {
return nil
},
},
{
"Reset and checkout merge conflicts",
func() (func(string, ...string) *exec.Cmd, *[][]string) {
cmdsCalled := [][]string{}
return func(cmd string, args ...string) *exec.Cmd {
cmdsCalled = append(cmdsCalled, args)
return secureexec.Command("echo")
}, &cmdsCalled
},
func(cmdsCalled *[][]string, err error) {
assert.NoError(t, err)
assert.Len(t, *cmdsCalled, 2)
assert.EqualValues(t, *cmdsCalled, [][]string{
{"reset", "--", "test"},
{"checkout", "--", "test"},
})
},
2020-09-29 10:45:00 +02:00
&models.File{
Name: "test",
Tracked: true,
HasMergeConflicts: true,
},
func(string) error {
return nil
},
},
{
"Reset and remove",
func() (func(string, ...string) *exec.Cmd, *[][]string) {
cmdsCalled := [][]string{}
return func(cmd string, args ...string) *exec.Cmd {
cmdsCalled = append(cmdsCalled, args)
return secureexec.Command("echo")
}, &cmdsCalled
},
func(cmdsCalled *[][]string, err error) {
assert.NoError(t, err)
assert.Len(t, *cmdsCalled, 1)
assert.EqualValues(t, *cmdsCalled, [][]string{
{"reset", "--", "test"},
})
},
2020-09-29 10:45:00 +02:00
&models.File{
Name: "test",
Tracked: false,
2021-03-20 03:07:11 +02:00
Added: true,
HasStagedChanges: true,
},
func(filename string) error {
assert.Equal(t, "test", filename)
return nil
},
},
{
"Remove only",
func() (func(string, ...string) *exec.Cmd, *[][]string) {
cmdsCalled := [][]string{}
return func(cmd string, args ...string) *exec.Cmd {
cmdsCalled = append(cmdsCalled, args)
return secureexec.Command("echo")
}, &cmdsCalled
},
func(cmdsCalled *[][]string, err error) {
assert.NoError(t, err)
assert.Len(t, *cmdsCalled, 0)
},
2020-09-29 10:45:00 +02:00
&models.File{
Name: "test",
Tracked: false,
2021-03-20 03:07:11 +02:00
Added: true,
HasStagedChanges: false,
},
func(filename string) error {
assert.Equal(t, "test", filename)
return nil
},
},
}
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
var cmdsCalled *[][]string
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command, cmdsCalled = s.command()
gitCmd.removeFile = s.removeFile
2019-03-18 11:44:33 +02:00
s.test(cmdsCalled, gitCmd.DiscardAllFileChanges(s.file))
})
}
}
2018-11-30 02:47:14 +02:00
// TestGitCommandCheckout is a function.
func TestGitCommandCheckout(t *testing.T) {
type scenario struct {
testName string
command func(string, ...string) *exec.Cmd
test func(error)
force bool
}
scenarios := []scenario{
{
"Checkout",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"checkout", "test"}, args)
return secureexec.Command("echo")
},
func(err error) {
assert.NoError(t, err)
},
false,
},
{
"Checkout forced",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"checkout", "--force", "test"}, args)
return secureexec.Command("echo")
},
func(err error) {
assert.NoError(t, err)
},
true,
},
}
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2020-03-21 11:05:02 +02:00
s.test(gitCmd.Checkout("test", CheckoutOptions{Force: s.force}))
})
}
}
2018-11-30 02:47:14 +02:00
// TestGitCommandGetBranchGraph is a function.
func TestGitCommandGetBranchGraph(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2020-03-05 23:21:51 +02:00
assert.EqualValues(t, []string{"log", "--graph", "--color=always", "--abbrev-commit", "--decorate", "--date=relative", "--pretty=medium", "test", "--"}, args)
return secureexec.Command("echo")
}
_, err := gitCmd.GetBranchGraph("test")
assert.NoError(t, err)
}
2020-11-27 09:07:14 +02:00
func TestGitCommandGetAllBranchGraph(t *testing.T) {
gitCmd := NewDummyGitCommand()
gitCmd.OSCommand.Command = func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
assert.EqualValues(t, []string{"log", "--graph", "--all", "--color=always", "--abbrev-commit", "--decorate", "--date=relative", "--pretty=medium"}, args)
return secureexec.Command("echo")
2020-11-27 09:07:14 +02:00
}
cmdStr := gitCmd.Config.GetUserConfig().Git.AllBranchesLogCmd
_, err := gitCmd.OSCommand.RunCommandWithOutput(cmdStr)
assert.NoError(t, err)
}
2018-11-30 02:47:14 +02:00
// TestGitCommandDiff is a function.
func TestGitCommandDiff(t *testing.T) {
type scenario struct {
testName string
command func(string, ...string) *exec.Cmd
2020-09-29 10:45:00 +02:00
file *models.File
2018-12-05 10:33:46 +02:00
plain bool
cached bool
}
scenarios := []scenario{
{
"Default case",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2020-09-29 10:19:04 +02:00
assert.EqualValues(t, []string{"diff", "--submodule", "--no-ext-diff", "--color=always", "--", "test.txt"}, args)
return secureexec.Command("echo")
},
2020-09-29 10:45:00 +02:00
&models.File{
Name: "test.txt",
HasStagedChanges: false,
Tracked: true,
},
2018-12-05 10:33:46 +02:00
false,
false,
2018-12-05 10:33:46 +02:00
},
{
"cached",
2018-12-05 10:33:46 +02:00
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2020-09-29 10:19:04 +02:00
assert.EqualValues(t, []string{"diff", "--submodule", "--no-ext-diff", "--color=always", "--cached", "--", "test.txt"}, args)
2018-12-05 10:33:46 +02:00
return secureexec.Command("echo")
2018-12-05 10:33:46 +02:00
},
2020-09-29 10:45:00 +02:00
&models.File{
2018-12-05 10:33:46 +02:00
Name: "test.txt",
HasStagedChanges: false,
Tracked: true,
},
false,
2018-12-05 10:33:46 +02:00
true,
},
{
"plain",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2020-09-29 10:19:04 +02:00
assert.EqualValues(t, []string{"diff", "--submodule", "--no-ext-diff", "--color=never", "--", "test.txt"}, args)
return secureexec.Command("echo")
},
2020-09-29 10:45:00 +02:00
&models.File{
Name: "test.txt",
HasStagedChanges: false,
Tracked: true,
},
true,
2018-12-05 10:33:46 +02:00
false,
},
{
"File not tracked and file has no staged changes",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
2021-03-02 06:36:04 +02:00
assert.EqualValues(t, []string{"diff", "--submodule", "--no-ext-diff", "--color=always", "--no-index", "--", "/dev/null", "test.txt"}, args)
return secureexec.Command("echo")
},
2020-09-29 10:45:00 +02:00
&models.File{
Name: "test.txt",
HasStagedChanges: false,
Tracked: false,
},
2018-12-05 10:33:46 +02:00
false,
false,
},
}
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2020-08-22 09:26:23 +02:00
gitCmd.WorktreeFileDiff(s.file, s.plain, s.cached)
2018-09-05 22:16:26 +02:00
})
}
}
2018-09-25 12:25:04 +02:00
2018-11-30 02:47:14 +02:00
// TestGitCommandCurrentBranchName is a function.
2018-09-25 12:31:19 +02:00
func TestGitCommandCurrentBranchName(t *testing.T) {
2018-10-05 01:11:19 +02:00
type scenario struct {
testName string
command func(string, ...string) *exec.Cmd
2020-03-26 11:29:35 +02:00
test func(string, string, error)
2018-10-05 01:11:19 +02:00
}
scenarios := []scenario{
{
"says we are on the master branch if we are",
func(cmd string, args ...string) *exec.Cmd {
assert.Equal(t, "git", cmd)
return secureexec.Command("echo", "master")
2018-10-05 01:11:19 +02:00
},
2020-03-26 11:29:35 +02:00
func(name string, displayname string, err error) {
2018-10-05 01:11:19 +02:00
assert.NoError(t, err)
2020-03-26 11:29:35 +02:00
assert.EqualValues(t, "master", name)
assert.EqualValues(t, "master", displayname)
2018-10-05 01:11:19 +02:00
},
},
{
2019-11-21 12:45:18 +02:00
"falls back to git `git branch --contains` if symbolic-ref fails",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
switch args[0] {
case "symbolic-ref":
assert.EqualValues(t, []string{"symbolic-ref", "--short", "HEAD"}, args)
return secureexec.Command("test")
2019-11-21 12:45:18 +02:00
case "branch":
assert.EqualValues(t, []string{"branch", "--contains"}, args)
return secureexec.Command("echo", "* master")
}
return nil
},
2020-03-26 11:29:35 +02:00
func(name string, displayname string, err error) {
assert.NoError(t, err)
2020-03-26 11:29:35 +02:00
assert.EqualValues(t, "master", name)
assert.EqualValues(t, "master", displayname)
},
},
{
"handles a detached head",
func(cmd string, args ...string) *exec.Cmd {
assert.EqualValues(t, "git", cmd)
switch args[0] {
case "symbolic-ref":
assert.EqualValues(t, []string{"symbolic-ref", "--short", "HEAD"}, args)
return secureexec.Command("test")
2020-03-26 11:29:35 +02:00
case "branch":
assert.EqualValues(t, []string{"branch", "--contains"}, args)
return secureexec.Command("echo", "* (HEAD detached at 123abcd)")
2020-03-26 11:29:35 +02:00
}
return nil
},
func(name string, displayname string, err error) {
assert.NoError(t, err)
assert.EqualValues(t, "123abcd", name)
assert.EqualValues(t, "(HEAD detached at 123abcd)", displayname)
},
},
2018-10-05 01:11:19 +02:00
{
"bubbles up error if there is one",
func(cmd string, args ...string) *exec.Cmd {
assert.Equal(t, "git", cmd)
return secureexec.Command("test")
2018-10-05 01:11:19 +02:00
},
2020-03-26 11:29:35 +02:00
func(name string, displayname string, err error) {
2018-10-05 01:11:19 +02:00
assert.Error(t, err)
2020-03-26 11:29:35 +02:00
assert.EqualValues(t, "", name)
assert.EqualValues(t, "", displayname)
2018-10-05 01:11:19 +02:00
},
},
}
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2018-10-05 01:11:19 +02:00
s.test(gitCmd.CurrentBranchName())
})
2018-09-25 12:31:19 +02:00
}
}
2018-12-05 10:33:46 +02:00
func TestGitCommandApplyPatch(t *testing.T) {
type scenario struct {
testName string
command func(string, ...string) *exec.Cmd
test func(error)
2018-12-05 10:33:46 +02:00
}
scenarios := []scenario{
{
"valid case",
func(cmd string, args ...string) *exec.Cmd {
assert.Equal(t, "git", cmd)
assert.EqualValues(t, []string{"apply", "--cached"}, args[0:2])
filename := args[2]
content, err := ioutil.ReadFile(filename)
assert.NoError(t, err)
assert.Equal(t, "test", string(content))
return secureexec.Command("echo", "done")
2018-12-05 10:33:46 +02:00
},
func(err error) {
2018-12-05 10:33:46 +02:00
assert.NoError(t, err)
},
},
{
"command returns error",
func(cmd string, args ...string) *exec.Cmd {
assert.Equal(t, "git", cmd)
assert.EqualValues(t, []string{"apply", "--cached"}, args[0:2])
filename := args[2]
// TODO: Ideally we want to mock out OSCommand here so that we're not
// double handling testing it's CreateTempFile functionality,
// but it is going to take a bit of work to make a proper mock for it
// so I'm leaving it for another PR
content, err := ioutil.ReadFile(filename)
assert.NoError(t, err)
assert.Equal(t, "test", string(content))
return secureexec.Command("test")
2018-12-05 10:33:46 +02:00
},
func(err error) {
2018-12-05 10:33:46 +02:00
assert.Error(t, err)
},
},
}
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
gitCmd := NewDummyGitCommand()
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
s.test(gitCmd.ApplyPatch("test", "cached"))
2018-12-05 10:33:46 +02:00
})
}
}
// TestGitCommandRebaseBranch is a function.
func TestGitCommandRebaseBranch(t *testing.T) {
type scenario struct {
testName string
arg string
command func(string, ...string) *exec.Cmd
test func(error)
}
scenarios := []scenario{
{
"successful rebase",
"master",
test.CreateMockCommand(t, []*test.CommandSwapper{
{
Expect: "git rebase --interactive --autostash --keep-empty master",
Replace: "echo",
},
}),
func(err error) {
assert.NoError(t, err)
},
},
{
"unsuccessful rebase",
"master",
test.CreateMockCommand(t, []*test.CommandSwapper{
{
Expect: "git rebase --interactive --autostash --keep-empty master",
Replace: "test",
},
}),
func(err error) {
assert.Error(t, err)
},
},
}
gitCmd := NewDummyGitCommand()
for _, s := range scenarios {
2019-03-12 10:56:04 +02:00
t.Run(s.testName, func(t *testing.T) {
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2019-03-12 10:56:04 +02:00
s.test(gitCmd.RebaseBranch(s.arg))
})
}
}
// TestGitCommandCheckoutFile is a function.
func TestGitCommandCheckoutFile(t *testing.T) {
type scenario struct {
testName string
commitSha string
fileName string
command func(string, ...string) *exec.Cmd
test func(error)
}
scenarios := []scenario{
{
"typical case",
"11af912",
"test999.txt",
test.CreateMockCommand(t, []*test.CommandSwapper{
{
Expect: "git checkout 11af912 test999.txt",
Replace: "echo",
},
}),
func(err error) {
assert.NoError(t, err)
},
},
{
"returns error if there is one",
"11af912",
"test999.txt",
test.CreateMockCommand(t, []*test.CommandSwapper{
{
Expect: "git checkout 11af912 test999.txt",
Replace: "test",
},
}),
func(err error) {
assert.Error(t, err)
},
},
}
gitCmd := NewDummyGitCommand()
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2019-03-12 10:56:04 +02:00
s.test(gitCmd.CheckoutFile(s.commitSha, s.fileName))
})
}
}
// TestGitCommandDiscardOldFileChanges is a function.
func TestGitCommandDiscardOldFileChanges(t *testing.T) {
type scenario struct {
testName string
2020-12-21 00:38:36 +02:00
getGitConfigValue func(string) (string, error)
2020-09-29 10:36:54 +02:00
commits []*models.Commit
2019-03-12 10:56:04 +02:00
commitIndex int
fileName string
command func(string, ...string) *exec.Cmd
test func(error)
}
scenarios := []scenario{
{
"returns error when index outside of range of commits",
func(string) (string, error) {
return "", nil
},
2020-09-29 10:36:54 +02:00
[]*models.Commit{},
2019-03-12 10:56:04 +02:00
0,
"test999.txt",
nil,
func(err error) {
assert.Error(t, err)
},
},
{
"returns error when using gpg",
func(string) (string, error) {
return "true", nil
},
2020-09-29 10:36:54 +02:00
[]*models.Commit{{Name: "commit", Sha: "123456"}},
2019-03-12 10:56:04 +02:00
0,
"test999.txt",
nil,
func(err error) {
assert.Error(t, err)
},
},
{
"checks out file if it already existed",
func(string) (string, error) {
return "", nil
},
2020-09-29 10:36:54 +02:00
[]*models.Commit{
2019-03-12 10:57:59 +02:00
{Name: "commit", Sha: "123456"},
{Name: "commit2", Sha: "abcdef"},
2019-03-12 10:56:04 +02:00
},
0,
"test999.txt",
test.CreateMockCommand(t, []*test.CommandSwapper{
{
Expect: "git rebase --interactive --autostash --keep-empty abcdef",
2019-03-12 10:56:04 +02:00
Replace: "echo",
},
{
Expect: "git cat-file -e HEAD^:test999.txt",
Replace: "echo",
},
{
Expect: "git checkout HEAD^ test999.txt",
Replace: "echo",
},
{
2019-11-05 10:03:36 +02:00
Expect: "git commit --amend --no-edit --allow-empty",
2019-03-12 10:56:04 +02:00
Replace: "echo",
},
{
Expect: "git rebase --continue",
Replace: "echo",
},
}),
func(err error) {
assert.NoError(t, err)
},
},
// test for when the file was created within the commit requires a refactor to support proper mocks
// currently we'd need to mock out the os.Remove function and that's gonna introduce tech debt
}
gitCmd := NewDummyGitCommand()
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2020-12-21 00:38:36 +02:00
gitCmd.getGitConfigValue = s.getGitConfigValue
2019-03-12 10:56:04 +02:00
s.test(gitCmd.DiscardOldFileChanges(s.commits, s.commitIndex, s.fileName))
})
}
}
2019-03-16 01:15:46 +02:00
// TestGitCommandDiscardUnstagedFileChanges is a function.
func TestGitCommandDiscardUnstagedFileChanges(t *testing.T) {
2019-03-18 11:44:33 +02:00
type scenario struct {
testName string
2020-09-29 10:45:00 +02:00
file *models.File
2019-03-18 11:44:33 +02:00
command func(string, ...string) *exec.Cmd
test func(error)
}
scenarios := []scenario{
{
"valid case",
2020-09-29 10:45:00 +02:00
&models.File{Name: "test.txt"},
2019-03-18 11:44:33 +02:00
test.CreateMockCommand(t, []*test.CommandSwapper{
{
Expect: `git checkout -- "test.txt"`,
Replace: "echo",
},
}),
func(err error) {
assert.NoError(t, err)
},
},
}
gitCmd := NewDummyGitCommand()
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2019-03-18 11:44:33 +02:00
s.test(gitCmd.DiscardUnstagedFileChanges(s.file))
})
}
}
// TestGitCommandDiscardAnyUnstagedFileChanges is a function.
func TestGitCommandDiscardAnyUnstagedFileChanges(t *testing.T) {
type scenario struct {
testName string
command func(string, ...string) *exec.Cmd
test func(error)
}
scenarios := []scenario{
{
"valid case",
test.CreateMockCommand(t, []*test.CommandSwapper{
{
Expect: `git checkout -- .`,
Replace: "echo",
},
}),
func(err error) {
assert.NoError(t, err)
},
},
}
gitCmd := NewDummyGitCommand()
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
s.test(gitCmd.DiscardAnyUnstagedFileChanges())
})
}
}
// TestGitCommandRemoveUntrackedFiles is a function.
func TestGitCommandRemoveUntrackedFiles(t *testing.T) {
type scenario struct {
testName string
command func(string, ...string) *exec.Cmd
test func(error)
}
scenarios := []scenario{
{
"valid case",
test.CreateMockCommand(t, []*test.CommandSwapper{
{
Expect: `git clean -fd`,
Replace: "echo",
},
}),
func(err error) {
assert.NoError(t, err)
},
},
}
gitCmd := NewDummyGitCommand()
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
s.test(gitCmd.RemoveUntrackedFiles())
})
}
}
// TestGitCommandResetHard is a function.
func TestGitCommandResetHard(t *testing.T) {
type scenario struct {
testName string
ref string
command func(string, ...string) *exec.Cmd
test func(error)
}
scenarios := []scenario{
{
"valid case",
"HEAD",
test.CreateMockCommand(t, []*test.CommandSwapper{
{
Expect: `git reset --hard HEAD`,
Replace: "echo",
},
}),
func(err error) {
assert.NoError(t, err)
},
},
}
gitCmd := NewDummyGitCommand()
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
s.test(gitCmd.ResetHard(s.ref))
})
}
}
2019-04-07 03:35:34 +02:00
// TestGitCommandCreateFixupCommit is a function.
func TestGitCommandCreateFixupCommit(t *testing.T) {
type scenario struct {
testName string
sha string
command func(string, ...string) *exec.Cmd
test func(error)
}
scenarios := []scenario{
{
"valid case",
"12345",
test.CreateMockCommand(t, []*test.CommandSwapper{
{
Expect: `git commit --fixup=12345`,
Replace: "echo",
},
}),
func(err error) {
assert.NoError(t, err)
},
},
}
gitCmd := NewDummyGitCommand()
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
2020-09-29 12:23:58 +02:00
gitCmd.OSCommand.Command = s.command
2019-04-07 03:35:34 +02:00
s.test(gitCmd.CreateFixupCommit(s.sha))
})
}
}
2019-05-12 09:04:32 +02:00
// TestGitCommandSkipEditorCommand confirms that SkipEditorCommand injects
// environment variables that suppress an interactive editor
func TestGitCommandSkipEditorCommand(t *testing.T) {
cmd := NewDummyGitCommand()
cmd.OSCommand.SetBeforeExecuteCmd(func(cmd *exec.Cmd) {
test.AssertContainsMatch(
t,
cmd.Env,
regexp.MustCompile("^VISUAL="),
"expected VISUAL to be set for a non-interactive external command",
)
test.AssertContainsMatch(
t,
cmd.Env,
regexp.MustCompile("^EDITOR="),
"expected EDITOR to be set for a non-interactive external command",
)
2020-03-27 00:16:41 +02:00
test.AssertContainsMatch(
t,
cmd.Env,
regexp.MustCompile("^GIT_EDITOR="),
"expected GIT_EDITOR to be set for a non-interactive external command",
)
test.AssertContainsMatch(
t,
cmd.Env,
regexp.MustCompile("^LAZYGIT_CLIENT_COMMAND=EXIT_IMMEDIATELY$"),
"expected LAZYGIT_CLIENT_COMMAND to be set for a non-interactive external command",
)
})
2020-09-29 12:03:39 +02:00
_ = cmd.runSkipEditorCommand("true")
}
2019-05-12 09:04:32 +02:00
func TestFindDotGitDir(t *testing.T) {
type scenario struct {
testName string
stat func(string) (os.FileInfo, error)
readFile func(filename string) ([]byte, error)
test func(string, error)
}
scenarios := []scenario{
{
".git is a directory",
func(dotGit string) (os.FileInfo, error) {
assert.Equal(t, ".git", dotGit)
return os.Stat("testdata/a_dir")
},
func(dotGit string) ([]byte, error) {
assert.Fail(t, "readFile should not be called if .git is a directory")
return nil, nil
},
func(gitDir string, err error) {
assert.NoError(t, err)
assert.Equal(t, ".git", gitDir)
},
},
{
".git is a file",
func(dotGit string) (os.FileInfo, error) {
assert.Equal(t, ".git", dotGit)
return os.Stat("testdata/a_file")
},
func(dotGit string) ([]byte, error) {
assert.Equal(t, ".git", dotGit)
return []byte("gitdir: blah\n"), nil
},
func(gitDir string, err error) {
assert.NoError(t, err)
assert.Equal(t, "blah", gitDir)
},
},
{
"os.Stat returns an error",
func(dotGit string) (os.FileInfo, error) {
assert.Equal(t, ".git", dotGit)
return nil, errors.New("error")
},
func(dotGit string) ([]byte, error) {
assert.Fail(t, "readFile should not be called os.Stat returns an error")
return nil, nil
},
func(gitDir string, err error) {
assert.Error(t, err)
},
},
{
"readFile returns an error",
func(dotGit string) (os.FileInfo, error) {
assert.Equal(t, ".git", dotGit)
return os.Stat("testdata/a_file")
},
func(dotGit string) ([]byte, error) {
return nil, errors.New("error")
},
func(gitDir string, err error) {
assert.Error(t, err)
},
},
}
for _, s := range scenarios {
t.Run(s.testName, func(t *testing.T) {
s.test(findDotGitDir(s.stat, s.readFile))
})
}
}
// TestEditFile is a function.
func TestEditFile(t *testing.T) {
type scenario struct {
2020-12-21 00:38:36 +02:00
filename string
command func(string, ...string) *exec.Cmd
getenv func(string) string
getGitConfigValue func(string) (string, error)
test func(*exec.Cmd, error)
}
scenarios := []scenario{
{
"test",
func(name string, arg ...string) *exec.Cmd {
return secureexec.Command("exit", "1")
},
func(env string) string {
return ""
},
func(cf string) (string, error) {
return "", nil
},
func(cmd *exec.Cmd, err error) {
2021-04-01 11:27:06 +02:00
assert.EqualError(t, err, "No editor defined in $GIT_EDITOR, $VISUAL, $EDITOR, or git config")
},
},
{
"test",
func(name string, arg ...string) *exec.Cmd {
if name == "which" {
return secureexec.Command("exit", "1")
}
assert.EqualValues(t, "nano", name)
return nil
},
func(env string) string {
return ""
},
func(cf string) (string, error) {
return "nano", nil
},
func(cmd *exec.Cmd, err error) {
assert.NoError(t, err)
},
},
{
"test",
func(name string, arg ...string) *exec.Cmd {
if name == "which" {
return secureexec.Command("exit", "1")
}
assert.EqualValues(t, "nano", name)
return nil
},
func(env string) string {
if env == "VISUAL" {
return "nano"
}
return ""
},
func(cf string) (string, error) {
return "", nil
},
func(cmd *exec.Cmd, err error) {
assert.NoError(t, err)
},
},
{
"test",
func(name string, arg ...string) *exec.Cmd {
if name == "which" {
return secureexec.Command("exit", "1")
}
assert.EqualValues(t, "emacs", name)
return nil
},
func(env string) string {
if env == "EDITOR" {
return "emacs"
}
return ""
},
func(cf string) (string, error) {
return "", nil
},
func(cmd *exec.Cmd, err error) {
assert.NoError(t, err)
},
},
{
"test",
func(name string, arg ...string) *exec.Cmd {
if name == "which" {
return secureexec.Command("echo")
}
assert.EqualValues(t, "vi", name)
return nil
},
func(env string) string {
return ""
},
func(cf string) (string, error) {
return "", nil
},
func(cmd *exec.Cmd, err error) {
assert.NoError(t, err)
},
},
{
"file/with space",
func(name string, args ...string) *exec.Cmd {
if name == "which" {
return secureexec.Command("echo")
}
assert.EqualValues(t, "vi", name)
assert.EqualValues(t, "file/with space", args[0])
return nil
},
func(env string) string {
return ""
},
func(cf string) (string, error) {
return "", nil
},
func(cmd *exec.Cmd, err error) {
assert.NoError(t, err)
},
},
}
for _, s := range scenarios {
gitCmd := NewDummyGitCommand()
gitCmd.OSCommand.Command = s.command
gitCmd.OSCommand.Getenv = s.getenv
2020-12-21 00:38:36 +02:00
gitCmd.getGitConfigValue = s.getGitConfigValue
s.test(gitCmd.EditFile(s.filename))
}
}