1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2024-12-12 11:15:00 +02:00
lazygit/pkg/gui/gui_test.go
2020-10-10 00:23:01 +11:00

260 lines
5.8 KiB
Go

package gui
import (
"fmt"
"io"
"io/ioutil"
"os"
"os/exec"
"path/filepath"
"testing"
"github.com/creack/pty"
"github.com/jesseduffield/lazygit/pkg/commands/oscommands"
"github.com/stretchr/testify/assert"
)
// To run an integration test, e.g. for test 'commit', go:
// go test pkg/gui/gui_test.go -run /commit
//
// To record keypresses for an integration test, pass RECORD_EVENTS=true like so:
// RECORD_EVENTS=true go test pkg/gui/gui_test.go -run /commit
//
// To update a snapshot for an integration test, pass UPDATE_SNAPSHOT=true
// UPDATE_SNAPSHOT=true go test pkg/gui/gui_test.go -run /commit
//
// When RECORD_EVENTS is true, updates will be updated automatically
//
// integration tests are run in test/integration_test and the final test does
// not clean up that directory so you can cd into it to see for yourself what
// happened when a test failed.
//
// TODO: support passing an env var for playback speed, given it's currently pretty fast
type integrationTest struct {
name string
fixture string
}
func tests() []integrationTest {
return []integrationTest{
{
name: "commit",
fixture: "newFile",
},
{
name: "squash",
fixture: "manyCommits",
},
{
name: "patchBuilding",
fixture: "updatedFile",
},
{
name: "patchBuilding2",
fixture: "updatedFile",
},
{
name: "mergeConflicts",
fixture: "mergeConflicts",
},
{
name: "searching",
fixture: "newFile",
},
{
name: "searchingInStagingPanel",
fixture: "newFile2",
},
}
}
func generateSnapshot(t *testing.T) string {
osCommand := oscommands.NewDummyOSCommand()
cmd := `sh -c "git status; cat ./*; git log --pretty=%B -p"`
snapshot, err := osCommand.RunCommandWithOutput(cmd)
assert.NoError(t, err)
return snapshot
}
func findOrCreateDir(path string) {
_, err := os.Stat(path)
if err != nil {
if os.IsNotExist(err) {
err = os.MkdirAll(path, 0777)
if err != nil {
panic(err)
}
} else {
panic(err)
}
}
}
func Test(t *testing.T) {
tests := tests()
gotoRootDirectory()
rootDir, err := os.Getwd()
if err != nil {
panic(err)
}
for _, test := range tests {
test := test
t.Run(test.name, func(t *testing.T) {
speeds := []int{10, 5, 1}
for i, speed := range speeds {
fmt.Printf("%s: trying again at speed %d\n", test.name, speed)
testPath := filepath.Join(rootDir, "test", "integration", test.name)
findOrCreateDir(testPath)
snapshotPath := filepath.Join(testPath, "snapshot.txt")
err := os.Chdir(rootDir)
assert.NoError(t, err)
prepareIntegrationTestDir()
err = createFixture(rootDir, test.fixture)
assert.NoError(t, err)
record := os.Getenv("RECORD_EVENTS") != ""
runLazygit(t, testPath, rootDir, record, speed)
updateSnapshot := record || os.Getenv("UPDATE_SNAPSHOT") != ""
actual := generateSnapshot(t)
if updateSnapshot {
err := ioutil.WriteFile(snapshotPath, []byte(actual), 0600)
assert.NoError(t, err)
}
expectedBytes, err := ioutil.ReadFile(snapshotPath)
assert.NoError(t, err)
expected := string(expectedBytes)
if expected == actual {
break
}
// if the snapshots and we haven't tried all playback speeds different we'll retry at a slower speed
if i == len(speeds)-1 {
assert.Equal(t, expected, actual, fmt.Sprintf("expected:\n%s\nactual:\n%s\n", expected, actual))
}
}
})
}
}
func createFixture(rootDir string, name string) error {
osCommand := oscommands.NewDummyOSCommand()
cmd := exec.Command("sh", filepath.Join(rootDir, "test", "fixtures", fmt.Sprintf("%s.sh", name)))
if err := osCommand.RunExecutable(cmd); err != nil {
return err
}
return nil
}
func gotoRootDirectory() {
for {
_, err := os.Stat(".git")
if err == nil {
return
}
if !os.IsNotExist(err) {
panic(err)
}
if err = os.Chdir(".."); err != nil {
panic(err)
}
}
}
func runLazygit(t *testing.T, testPath string, rootDir string, record bool, speed int) {
osCommand := oscommands.NewDummyOSCommand()
replayPath := filepath.Join(testPath, "recording.json")
cmdStr := fmt.Sprintf("go run %s", filepath.Join(rootDir, "main.go"))
templateConfigDir := filepath.Join(rootDir, "test", "default_test_config")
exists, err := osCommand.FileExists(filepath.Join(testPath, "config"))
assert.NoError(t, err)
if exists {
templateConfigDir = filepath.Join(testPath, "config")
}
configDir := filepath.Join(rootDir, "test", "integration_test_config")
err = os.RemoveAll(configDir)
assert.NoError(t, err)
err = oscommands.CopyDir(templateConfigDir, configDir)
assert.NoError(t, err)
cmdStr = fmt.Sprintf("%s --use-config-dir=%s", cmdStr, configDir)
cmd := osCommand.ExecutableFromString(cmdStr)
cmd.Env = append(cmd.Env, fmt.Sprintf("REPLAY_SPEED=%d", speed))
if record {
cmd.Env = append(
cmd.Env,
fmt.Sprintf("RECORD_EVENTS_TO=%s", replayPath),
)
} else {
cmd.Env = append(
cmd.Env,
fmt.Sprintf("REPLAY_EVENTS_FROM=%s", replayPath),
)
}
// if we're on CI we'll need to use a PTY. We can work that out by seeing if the 'TERM' env is defined.
if os.Getenv("TERM") == "" {
cmd.Env = append(cmd.Env, "TERM=xterm")
f, err := pty.StartWithSize(cmd, &pty.Winsize{Rows: 100, Cols: 100})
assert.NoError(t, err)
_, err = io.Copy(os.Stdout, f)
assert.NoError(t, err)
} else {
err := osCommand.RunExecutable(cmd)
assert.NoError(t, err)
}
}
func prepareIntegrationTestDir() {
path := filepath.Join("test", "integration_test")
// remove contents of integration test directory
dir, err := ioutil.ReadDir(path)
if err != nil {
if os.IsNotExist(err) {
err = os.Mkdir(path, 0777)
if err != nil {
panic(err)
}
} else {
panic(err)
}
}
for _, d := range dir {
os.RemoveAll(filepath.Join(path, d.Name()))
}
if err := os.Chdir(path); err != nil {
panic(err)
}
}