2020-10-04 17:41:21 +11:00
|
|
|
package gui
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"io/ioutil"
|
|
|
|
"log"
|
|
|
|
"os"
|
2020-10-05 20:55:15 +11:00
|
|
|
"strconv"
|
2020-10-04 17:41:21 +11:00
|
|
|
"time"
|
2020-10-05 19:55:39 +11:00
|
|
|
|
|
|
|
"github.com/jesseduffield/gocui"
|
2020-10-07 21:19:38 +11:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
2020-10-04 17:41:21 +11:00
|
|
|
)
|
|
|
|
|
|
|
|
func recordingEvents() bool {
|
2020-10-04 18:41:33 +11:00
|
|
|
return recordEventsTo() != ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func recordEventsTo() string {
|
|
|
|
return os.Getenv("RECORD_EVENTS_TO")
|
2020-10-04 17:41:21 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
func (gui *Gui) timeSinceStart() int64 {
|
2020-10-14 12:40:52 +02:00
|
|
|
return time.Since(gui.StartTime).Nanoseconds() / 1e6
|
2020-10-04 17:41:21 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
func (gui *Gui) replayRecordedEvents() {
|
|
|
|
if os.Getenv("REPLAY_EVENTS_FROM") == "" {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2020-10-07 21:19:38 +11:00
|
|
|
go utils.Safe(func() {
|
2020-10-06 08:01:25 +11:00
|
|
|
time.Sleep(time.Second * 20)
|
|
|
|
log.Fatal("20 seconds is up, lazygit recording took too long to complete")
|
2020-10-07 21:19:38 +11:00
|
|
|
})
|
2020-10-06 08:01:25 +11:00
|
|
|
|
2020-10-04 17:41:21 +11:00
|
|
|
events, err := gui.loadRecordedEvents()
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
ticker := time.NewTicker(time.Millisecond)
|
|
|
|
defer ticker.Stop()
|
|
|
|
|
2020-10-04 18:41:33 +11:00
|
|
|
// might need to add leeway if this ends up flakey
|
|
|
|
var leeway int64 = 0
|
|
|
|
// humans are slow so this speeds things up.
|
2020-10-05 20:55:15 +11:00
|
|
|
speed := 1
|
|
|
|
envReplaySpeed := os.Getenv("REPLAY_SPEED")
|
|
|
|
if envReplaySpeed != "" {
|
|
|
|
var err error
|
|
|
|
speed, err = strconv.Atoi(envReplaySpeed)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal(err)
|
|
|
|
}
|
|
|
|
}
|
2020-10-04 17:41:21 +11:00
|
|
|
|
2020-10-06 08:01:25 +11:00
|
|
|
// The playback could be paused at any time because integration tests run concurrently.
|
|
|
|
// Therefore we can't just check for a given event whether we've passed its timestamp,
|
|
|
|
// or else we'll have an explosion of keypresses after the test is resumed.
|
|
|
|
// We need to check if we've waited long enough since the last event was replayed.
|
|
|
|
for i, event := range events {
|
|
|
|
var prevEventTimestamp int64 = 0
|
|
|
|
if i > 0 {
|
|
|
|
prevEventTimestamp = events[i-1].Timestamp
|
|
|
|
}
|
|
|
|
timeToWait := (event.Timestamp - prevEventTimestamp) / int64(speed)
|
|
|
|
if i == 0 {
|
|
|
|
timeToWait += leeway
|
|
|
|
}
|
|
|
|
var timeWaited int64 = 0
|
2020-10-05 20:43:07 +11:00
|
|
|
middle:
|
2020-10-05 20:05:09 +11:00
|
|
|
for {
|
|
|
|
select {
|
|
|
|
case <-ticker.C:
|
2020-10-06 08:01:25 +11:00
|
|
|
timeWaited += 1
|
|
|
|
if gui.g != nil && timeWaited >= timeToWait {
|
2020-10-05 20:05:09 +11:00
|
|
|
gui.g.ReplayedEvents <- *event.Event
|
2020-10-05 20:43:07 +11:00
|
|
|
break middle
|
2020-10-05 20:05:09 +11:00
|
|
|
}
|
|
|
|
case <-gui.stopChan:
|
|
|
|
return
|
2020-10-04 17:41:21 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-10-05 19:55:39 +11:00
|
|
|
|
|
|
|
time.Sleep(time.Second * 1)
|
|
|
|
|
|
|
|
gui.g.Update(func(*gocui.Gui) error {
|
|
|
|
return gocui.ErrQuit
|
|
|
|
})
|
|
|
|
|
|
|
|
time.Sleep(time.Second * 1)
|
|
|
|
|
|
|
|
log.Fatal("lazygit should have already exited")
|
2020-10-04 17:41:21 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
func (gui *Gui) loadRecordedEvents() ([]RecordedEvent, error) {
|
|
|
|
path := os.Getenv("REPLAY_EVENTS_FROM")
|
|
|
|
|
|
|
|
data, err := ioutil.ReadFile(path)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
events := []RecordedEvent{}
|
|
|
|
|
|
|
|
err = json.Unmarshal(data, &events)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return events, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (gui *Gui) saveRecordedEvents() error {
|
|
|
|
if !recordingEvents() {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
jsonEvents, err := json.Marshal(gui.RecordedEvents)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-10-04 18:41:33 +11:00
|
|
|
path := recordEventsTo()
|
|
|
|
|
|
|
|
return ioutil.WriteFile(path, jsonEvents, 0600)
|
2020-10-04 17:41:21 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
func (gui *Gui) recordEvents() {
|
|
|
|
for event := range gui.g.RecordedEvents {
|
|
|
|
recordedEvent := RecordedEvent{
|
|
|
|
Timestamp: gui.timeSinceStart(),
|
|
|
|
Event: event,
|
|
|
|
}
|
|
|
|
|
|
|
|
gui.RecordedEvents = append(gui.RecordedEvents, recordedEvent)
|
|
|
|
}
|
|
|
|
}
|