2021-11-07 13:25:06 +11:00
|
|
|
package tasks
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"io"
|
|
|
|
"os/exec"
|
2023-03-10 13:35:19 +01:00
|
|
|
"reflect"
|
|
|
|
"strings"
|
2021-11-07 13:25:06 +11:00
|
|
|
"sync"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/secureexec"
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
|
|
|
)
|
|
|
|
|
|
|
|
func getCounter() (func(), func() int) {
|
|
|
|
counter := 0
|
|
|
|
return func() { counter++ }, func() int { return counter }
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestNewCmdTaskInstantStop(t *testing.T) {
|
|
|
|
writer := bytes.NewBuffer(nil)
|
|
|
|
beforeStart, getBeforeStartCallCount := getCounter()
|
|
|
|
refreshView, getRefreshViewCallCount := getCounter()
|
|
|
|
onEndOfInput, getOnEndOfInputCallCount := getCounter()
|
|
|
|
onNewKey, getOnNewKeyCallCount := getCounter()
|
|
|
|
onDone, getOnDoneCallCount := getCounter()
|
|
|
|
|
|
|
|
manager := NewViewBufferManager(
|
|
|
|
utils.NewDummyLog(),
|
|
|
|
writer,
|
|
|
|
beforeStart,
|
|
|
|
refreshView,
|
|
|
|
onEndOfInput,
|
|
|
|
onNewKey,
|
|
|
|
)
|
|
|
|
|
|
|
|
stop := make(chan struct{})
|
|
|
|
reader := bytes.NewBufferString("test")
|
|
|
|
start := func() (*exec.Cmd, io.Reader) {
|
|
|
|
// not actually starting this because it's not necessary
|
|
|
|
cmd := secureexec.Command("blah blah")
|
|
|
|
|
|
|
|
close(stop)
|
|
|
|
|
|
|
|
return cmd, reader
|
|
|
|
}
|
|
|
|
|
2023-03-10 13:35:19 +01:00
|
|
|
fn := manager.NewCmdTask(start, "prefix\n", LinesToRead{20, -1}, onDone)
|
2021-11-07 13:25:06 +11:00
|
|
|
|
|
|
|
_ = fn(stop)
|
|
|
|
|
|
|
|
callCountExpectations := []struct {
|
|
|
|
expected int
|
|
|
|
actual int
|
|
|
|
name string
|
|
|
|
}{
|
|
|
|
{0, getBeforeStartCallCount(), "beforeStart"},
|
|
|
|
{1, getRefreshViewCallCount(), "refreshView"},
|
|
|
|
{0, getOnEndOfInputCallCount(), "onEndOfInput"},
|
|
|
|
{0, getOnNewKeyCallCount(), "onNewKey"},
|
|
|
|
{1, getOnDoneCallCount(), "onDone"},
|
|
|
|
}
|
|
|
|
for _, expectation := range callCountExpectations {
|
|
|
|
if expectation.actual != expectation.expected {
|
|
|
|
t.Errorf("expected %s to be called %d times, got %d", expectation.name, expectation.expected, expectation.actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
expectedContent := ""
|
|
|
|
actualContent := writer.String()
|
|
|
|
if actualContent != expectedContent {
|
2022-04-03 15:19:15 -05:00
|
|
|
t.Errorf("expected writer to receive the following content: \n%s\n. But instead it received: %s", expectedContent, actualContent)
|
2021-11-07 13:25:06 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestNewCmdTask(t *testing.T) {
|
|
|
|
writer := bytes.NewBuffer(nil)
|
|
|
|
beforeStart, getBeforeStartCallCount := getCounter()
|
|
|
|
refreshView, getRefreshViewCallCount := getCounter()
|
|
|
|
onEndOfInput, getOnEndOfInputCallCount := getCounter()
|
|
|
|
onNewKey, getOnNewKeyCallCount := getCounter()
|
|
|
|
onDone, getOnDoneCallCount := getCounter()
|
|
|
|
|
|
|
|
manager := NewViewBufferManager(
|
|
|
|
utils.NewDummyLog(),
|
|
|
|
writer,
|
|
|
|
beforeStart,
|
|
|
|
refreshView,
|
|
|
|
onEndOfInput,
|
|
|
|
onNewKey,
|
|
|
|
)
|
|
|
|
|
|
|
|
stop := make(chan struct{})
|
|
|
|
reader := bytes.NewBufferString("test")
|
|
|
|
start := func() (*exec.Cmd, io.Reader) {
|
|
|
|
// not actually starting this because it's not necessary
|
|
|
|
cmd := secureexec.Command("blah blah")
|
|
|
|
|
|
|
|
return cmd, reader
|
|
|
|
}
|
|
|
|
|
2023-03-10 13:35:19 +01:00
|
|
|
fn := manager.NewCmdTask(start, "prefix\n", LinesToRead{20, -1}, onDone)
|
2021-11-07 13:25:06 +11:00
|
|
|
wg := sync.WaitGroup{}
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
close(stop)
|
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
_ = fn(stop)
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
callCountExpectations := []struct {
|
|
|
|
expected int
|
|
|
|
actual int
|
|
|
|
name string
|
|
|
|
}{
|
|
|
|
{1, getBeforeStartCallCount(), "beforeStart"},
|
|
|
|
{1, getRefreshViewCallCount(), "refreshView"},
|
|
|
|
{1, getOnEndOfInputCallCount(), "onEndOfInput"},
|
|
|
|
{0, getOnNewKeyCallCount(), "onNewKey"},
|
|
|
|
{1, getOnDoneCallCount(), "onDone"},
|
|
|
|
}
|
|
|
|
for _, expectation := range callCountExpectations {
|
|
|
|
if expectation.actual != expectation.expected {
|
|
|
|
t.Errorf("expected %s to be called %d times, got %d", expectation.name, expectation.expected, expectation.actual)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
expectedContent := "prefix\ntest\n"
|
|
|
|
actualContent := writer.String()
|
|
|
|
if actualContent != expectedContent {
|
2022-04-03 15:19:15 -05:00
|
|
|
t.Errorf("expected writer to receive the following content: \n%s\n. But instead it received: %s", expectedContent, actualContent)
|
2021-11-07 13:25:06 +11:00
|
|
|
}
|
|
|
|
}
|
2023-03-10 13:35:19 +01:00
|
|
|
|
|
|
|
// A dummy reader that simply yields as many blank lines as requested. The only
|
|
|
|
// thing we want to do with the output is count the number of lines.
|
|
|
|
type BlankLineReader struct {
|
|
|
|
totalLinesToYield int
|
|
|
|
linesYielded int
|
|
|
|
}
|
|
|
|
|
|
|
|
func (d *BlankLineReader) Read(p []byte) (n int, err error) {
|
|
|
|
if d.totalLinesToYield == d.linesYielded {
|
|
|
|
return 0, io.EOF
|
|
|
|
}
|
|
|
|
|
|
|
|
d.linesYielded += 1
|
|
|
|
p[0] = '\n'
|
|
|
|
return 1, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestNewCmdTaskRefresh(t *testing.T) {
|
|
|
|
type scenario struct {
|
|
|
|
name string
|
|
|
|
totalTaskLines int
|
|
|
|
linesToRead LinesToRead
|
|
|
|
expectedLineCountsOnRefresh []int
|
|
|
|
}
|
|
|
|
|
|
|
|
scenarios := []scenario{
|
|
|
|
{
|
|
|
|
"total < initialRefreshAfter",
|
|
|
|
150,
|
|
|
|
LinesToRead{100, 120},
|
2023-04-02 15:44:05 +10:00
|
|
|
[]int{100},
|
2023-03-10 13:35:19 +01:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"total == initialRefreshAfter",
|
|
|
|
150,
|
|
|
|
LinesToRead{100, 100},
|
2023-04-02 15:44:05 +10:00
|
|
|
[]int{100},
|
2023-03-10 13:35:19 +01:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"total > initialRefreshAfter",
|
|
|
|
150,
|
|
|
|
LinesToRead{100, 50},
|
2023-04-02 15:44:05 +10:00
|
|
|
[]int{50, 100},
|
2023-03-10 13:35:19 +01:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"initialRefreshAfter == -1",
|
|
|
|
150,
|
|
|
|
LinesToRead{100, -1},
|
2023-04-02 15:44:05 +10:00
|
|
|
[]int{100},
|
2023-03-10 13:35:19 +01:00
|
|
|
},
|
|
|
|
{
|
|
|
|
"totalTaskLines < initialRefreshAfter",
|
|
|
|
25,
|
|
|
|
LinesToRead{100, 50},
|
|
|
|
[]int{25},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"totalTaskLines between total and initialRefreshAfter",
|
|
|
|
75,
|
|
|
|
LinesToRead{100, 50},
|
|
|
|
[]int{50, 75},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, s := range scenarios {
|
|
|
|
writer := bytes.NewBuffer(nil)
|
|
|
|
lineCountsOnRefresh := []int{}
|
|
|
|
refreshView := func() {
|
|
|
|
lineCountsOnRefresh = append(lineCountsOnRefresh, strings.Count(writer.String(), "\n"))
|
|
|
|
}
|
|
|
|
|
|
|
|
manager := NewViewBufferManager(
|
|
|
|
utils.NewDummyLog(),
|
|
|
|
writer,
|
|
|
|
func() {},
|
|
|
|
refreshView,
|
|
|
|
func() {},
|
|
|
|
func() {},
|
|
|
|
)
|
|
|
|
|
|
|
|
stop := make(chan struct{})
|
|
|
|
reader := BlankLineReader{totalLinesToYield: s.totalTaskLines}
|
|
|
|
start := func() (*exec.Cmd, io.Reader) {
|
|
|
|
// not actually starting this because it's not necessary
|
|
|
|
cmd := secureexec.Command("blah blah")
|
|
|
|
|
|
|
|
return cmd, &reader
|
|
|
|
}
|
|
|
|
|
|
|
|
fn := manager.NewCmdTask(start, "", s.linesToRead, func() {})
|
|
|
|
wg := sync.WaitGroup{}
|
|
|
|
wg.Add(1)
|
|
|
|
go func() {
|
|
|
|
time.Sleep(100 * time.Millisecond)
|
|
|
|
close(stop)
|
|
|
|
wg.Done()
|
|
|
|
}()
|
|
|
|
_ = fn(stop)
|
|
|
|
|
|
|
|
wg.Wait()
|
|
|
|
|
|
|
|
if !reflect.DeepEqual(lineCountsOnRefresh, s.expectedLineCountsOnRefresh) {
|
|
|
|
t.Errorf("%s: expected line counts on refresh: %v, got %v",
|
|
|
|
s.name, s.expectedLineCountsOnRefresh, lineCountsOnRefresh)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|