2020-03-27 06:54:57 +00:00
|
|
|
package initialsync
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"testing"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestStateMachine_Stringify(t *testing.T) {
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
epochs []*epochState
|
|
|
|
want string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
"empty epoch state list",
|
|
|
|
make([]*epochState, 0, lookaheadEpochs),
|
|
|
|
"[]",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"newly created state machine",
|
|
|
|
[]*epochState{
|
2020-04-03 05:09:15 +00:00
|
|
|
{epoch: 8, state: stateNew},
|
|
|
|
{epoch: 9, state: stateScheduled},
|
|
|
|
{epoch: 10, state: stateDataParsed},
|
|
|
|
{epoch: 11, state: stateSkipped},
|
|
|
|
{epoch: 12, state: stateSkippedExt},
|
|
|
|
{epoch: 13, state: stateComplete},
|
|
|
|
{epoch: 14, state: stateSent},
|
2020-03-27 06:54:57 +00:00
|
|
|
},
|
|
|
|
"[8:new 9:scheduled 10:dataParsed 11:skipped 12:skippedExt 13:complete 14:sent]",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
sm := &stateMachine{
|
|
|
|
epochs: tt.epochs,
|
|
|
|
}
|
|
|
|
if got := sm.String(); got != tt.want {
|
|
|
|
t.Errorf("unexpected output, got: %v, want: %v", got, tt.want)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStateMachine_addHandler(t *testing.T) {
|
|
|
|
sm := newStateMachine()
|
|
|
|
|
|
|
|
sm.addHandler(stateNew, eventSchedule, func(state *epochState, i interface{}) (id stateID, err error) {
|
|
|
|
return stateScheduled, nil
|
|
|
|
})
|
|
|
|
if len(sm.events[eventSchedule].actions) != 1 {
|
|
|
|
t.Errorf("unexpected size, got: %v, want: %v", len(sm.events[eventSchedule].actions), 1)
|
|
|
|
}
|
|
|
|
state, err := sm.events[eventSchedule].actions[0].handlerFn(nil, nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
if state != stateScheduled {
|
|
|
|
t.Errorf("unexpected state, got: %v, want: %v", state, stateScheduled)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add second handler to the same event
|
|
|
|
sm.addHandler(stateSent, eventSchedule, func(state *epochState, i interface{}) (id stateID, err error) {
|
|
|
|
return stateDataParsed, nil
|
|
|
|
})
|
|
|
|
if len(sm.events[eventSchedule].actions) != 2 {
|
|
|
|
t.Errorf("unexpected size, got: %v, want: %v", len(sm.events[eventSchedule].actions), 2)
|
|
|
|
}
|
|
|
|
state, err = sm.events[eventSchedule].actions[1].handlerFn(nil, nil)
|
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
if state != stateDataParsed {
|
|
|
|
t.Errorf("unexpected state, got: %v, want: %v", state, stateScheduled)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStateMachine_trigger(t *testing.T) {
|
|
|
|
type event struct {
|
|
|
|
state stateID
|
|
|
|
event eventID
|
|
|
|
returnState stateID
|
|
|
|
err bool
|
|
|
|
}
|
|
|
|
type args struct {
|
|
|
|
name eventID
|
2020-04-03 05:09:15 +00:00
|
|
|
returnState stateID
|
2020-03-27 06:54:57 +00:00
|
|
|
epoch uint64
|
|
|
|
data interface{}
|
|
|
|
}
|
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
events []event
|
|
|
|
epochs []uint64
|
|
|
|
args args
|
|
|
|
err error
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
name: "event not found",
|
|
|
|
events: []event{},
|
|
|
|
epochs: []uint64{},
|
2020-04-03 05:09:15 +00:00
|
|
|
args: args{name: eventSchedule, epoch: 12, data: nil, returnState: stateNew},
|
2020-03-27 06:54:57 +00:00
|
|
|
err: fmt.Errorf("event not found: %v", eventSchedule),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "epoch not found",
|
|
|
|
events: []event{
|
|
|
|
{stateNew, eventSchedule, stateScheduled, false},
|
|
|
|
},
|
|
|
|
epochs: []uint64{},
|
2020-04-03 05:09:15 +00:00
|
|
|
args: args{name: eventSchedule, epoch: 12, data: nil, returnState: stateScheduled},
|
2020-03-27 06:54:57 +00:00
|
|
|
err: fmt.Errorf("state for %v epoch not found", 12),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "single action",
|
|
|
|
events: []event{
|
|
|
|
{stateNew, eventSchedule, stateScheduled, false},
|
|
|
|
},
|
|
|
|
epochs: []uint64{12, 13},
|
2020-04-03 05:09:15 +00:00
|
|
|
args: args{name: eventSchedule, epoch: 12, data: nil, returnState: stateScheduled},
|
2020-03-27 06:54:57 +00:00
|
|
|
err: nil,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "multiple actions, has error",
|
|
|
|
events: []event{
|
|
|
|
{stateNew, eventSchedule, stateScheduled, false},
|
|
|
|
{stateScheduled, eventSchedule, stateSent, true},
|
|
|
|
{stateSent, eventSchedule, stateComplete, false},
|
|
|
|
},
|
|
|
|
epochs: []uint64{12, 13},
|
2020-04-03 05:09:15 +00:00
|
|
|
args: args{name: eventSchedule, epoch: 12, data: nil, returnState: stateScheduled},
|
2020-03-27 06:54:57 +00:00
|
|
|
err: nil,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "multiple actions, no error, cascade",
|
|
|
|
events: []event{
|
|
|
|
{stateNew, eventSchedule, stateScheduled, false},
|
|
|
|
{stateScheduled, eventSchedule, stateSent, false},
|
|
|
|
{stateSent, eventSchedule, stateComplete, false},
|
|
|
|
},
|
|
|
|
epochs: []uint64{12, 13},
|
2020-04-03 05:09:15 +00:00
|
|
|
args: args{name: eventSchedule, epoch: 12, data: nil, returnState: stateComplete},
|
2020-03-27 06:54:57 +00:00
|
|
|
err: nil,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "multiple actions, no error, no cascade",
|
|
|
|
events: []event{
|
|
|
|
{stateNew, eventSchedule, stateScheduled, false},
|
|
|
|
{stateScheduled, eventSchedule, stateSent, false},
|
|
|
|
{stateNew, eventSchedule, stateComplete, false},
|
|
|
|
},
|
|
|
|
epochs: []uint64{12, 13},
|
2020-04-03 05:09:15 +00:00
|
|
|
args: args{name: eventSchedule, epoch: 12, data: nil, returnState: stateSent},
|
2020-03-27 06:54:57 +00:00
|
|
|
err: nil,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
fn := func(e event) eventHandlerFn {
|
|
|
|
return func(es *epochState, in interface{}) (stateID, error) {
|
|
|
|
if e.err {
|
|
|
|
return es.state, errors.New("invalid")
|
|
|
|
}
|
|
|
|
return e.returnState, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
|
|
sm := newStateMachine()
|
|
|
|
expectHandlerError := false
|
|
|
|
for _, event := range tt.events {
|
|
|
|
sm.addHandler(event.state, event.event, fn(event))
|
|
|
|
if event.err {
|
|
|
|
expectHandlerError = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for _, epoch := range tt.epochs {
|
|
|
|
sm.addEpochState(epoch)
|
|
|
|
}
|
|
|
|
err := sm.trigger(tt.args.name, tt.args.epoch, tt.args.data)
|
|
|
|
if tt.err != nil && (err == nil || tt.err.Error() != err.Error()) {
|
|
|
|
t.Errorf("unexpected error = '%v', want '%v'", err, tt.err)
|
|
|
|
}
|
|
|
|
if tt.err == nil {
|
|
|
|
if err != nil && !expectHandlerError {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
ind, ok := sm.findEpochState(tt.args.epoch)
|
|
|
|
if !ok {
|
|
|
|
t.Errorf("expected epoch not found: %v", tt.args.epoch)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if sm.epochs[ind].state != tt.args.returnState {
|
|
|
|
t.Errorf("unexpected final state: %v, want: %v (%v)", sm.epochs[ind].state, tt.args.returnState, sm.epochs)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStateMachine_findEpochState(t *testing.T) {
|
|
|
|
sm := newStateMachine()
|
|
|
|
if ind, ok := sm.findEpochState(12); ok || ind != 0 {
|
|
|
|
t.Errorf("unexpected index: %v, want: %v", ind, 0)
|
|
|
|
}
|
|
|
|
sm.addEpochState(12)
|
|
|
|
if ind, ok := sm.findEpochState(12); !ok || ind != 0 {
|
|
|
|
t.Errorf("unexpected index: %v, want: %v", ind, 0)
|
|
|
|
}
|
|
|
|
sm.addEpochState(13)
|
|
|
|
sm.addEpochState(14)
|
|
|
|
sm.addEpochState(15)
|
|
|
|
if ind, ok := sm.findEpochState(14); !ok || ind != 2 {
|
|
|
|
t.Errorf("unexpected index: %v, want: %v", ind, 2)
|
|
|
|
}
|
|
|
|
if ind, ok := sm.findEpochState(16); ok || ind != len(sm.epochs) {
|
|
|
|
t.Errorf("unexpected index: %v, want: %v", ind, len(sm.epochs))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStateMachine_isLowestEpochState(t *testing.T) {
|
|
|
|
sm := newStateMachine()
|
|
|
|
sm.addEpochState(12)
|
|
|
|
sm.addEpochState(13)
|
|
|
|
sm.addEpochState(14)
|
|
|
|
if res := sm.isLowestEpochState(15); res {
|
|
|
|
t.Errorf("unexpected lowest state: %v", 15)
|
|
|
|
}
|
|
|
|
if res := sm.isLowestEpochState(13); res {
|
|
|
|
t.Errorf("unexpected lowest state: %v", 15)
|
|
|
|
}
|
|
|
|
if res := sm.isLowestEpochState(12); !res {
|
|
|
|
t.Errorf("expected lowest state not found: %v", 12)
|
|
|
|
}
|
|
|
|
if err := sm.removeEpochState(12); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
if res := sm.isLowestEpochState(12); res {
|
|
|
|
t.Errorf("unexpected lowest state: %v", 12)
|
|
|
|
}
|
|
|
|
if res := sm.isLowestEpochState(13); !res {
|
|
|
|
t.Errorf("expected lowest state not found: %v", 13)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-04-02 03:54:05 +00:00
|
|
|
func TestStateMachine_highestEpoch(t *testing.T) {
|
2020-03-27 06:54:57 +00:00
|
|
|
sm := newStateMachine()
|
2020-04-02 03:54:05 +00:00
|
|
|
if _, err := sm.highestEpoch(); err == nil {
|
2020-03-27 06:54:57 +00:00
|
|
|
t.Error("expected error")
|
|
|
|
}
|
|
|
|
sm.addEpochState(12)
|
|
|
|
sm.addEpochState(13)
|
|
|
|
sm.addEpochState(14)
|
2020-04-02 03:54:05 +00:00
|
|
|
epoch, err := sm.highestEpoch()
|
2020-03-27 06:54:57 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
2020-04-02 03:54:05 +00:00
|
|
|
if epoch != 14 {
|
|
|
|
t.Errorf("incorrect highest epoch: %v, want: %v", epoch, 14)
|
2020-03-27 06:54:57 +00:00
|
|
|
}
|
|
|
|
if err := sm.removeEpochState(14); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
2020-04-02 03:54:05 +00:00
|
|
|
epoch, err = sm.highestEpoch()
|
2020-03-27 06:54:57 +00:00
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
2020-04-02 03:54:05 +00:00
|
|
|
if epoch != 13 {
|
|
|
|
t.Errorf("incorrect highest epoch: %v, want: %v", epoch, 13)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestStateMachine_lowestEpoch(t *testing.T) {
|
|
|
|
sm := newStateMachine()
|
|
|
|
if _, err := sm.highestEpoch(); err == nil {
|
|
|
|
t.Error("expected error")
|
|
|
|
}
|
|
|
|
sm.addEpochState(12)
|
|
|
|
sm.addEpochState(13)
|
|
|
|
sm.addEpochState(14)
|
|
|
|
epoch, err := sm.lowestEpoch()
|
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
if epoch != 12 {
|
|
|
|
t.Errorf("incorrect highest epoch: %v, want: %v", epoch, 12)
|
|
|
|
}
|
|
|
|
if err := sm.removeEpochState(12); err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
epoch, err = sm.lowestEpoch()
|
|
|
|
if err != nil {
|
|
|
|
t.Error(err)
|
|
|
|
}
|
|
|
|
if epoch != 13 {
|
|
|
|
t.Errorf("incorrect highest epoch: %v, want: %v", epoch, 13)
|
2020-03-27 06:54:57 +00:00
|
|
|
}
|
|
|
|
}
|