diff --git a/logging/fields/fields.go b/logging/fields/fields.go index 61d53747b3..e0893c365f 100644 --- a/logging/fields/fields.go +++ b/logging/fields/fields.go @@ -82,9 +82,9 @@ const ( FieldRole = "role" FieldRound = "round" FieldSlot = "slot" + FieldSlotTickerID = "slot_ticker_id" FieldStartTimeUnixMilli = "start_time_unix_milli" FieldSubmissionTime = "submission_time" - FieldTotalConsensusTime = "total_consensus_time" FieldSubnets = "subnets" FieldSyncOffset = "sync_offset" FieldSyncResults = "sync_results" @@ -92,6 +92,7 @@ const ( FieldToBlock = "to_block" FieldTook = "took" FieldTopic = "topic" + FieldTotalConsensusTime = "total_consensus_time" FieldTxHash = "tx_hash" FieldType = "type" FieldUpdatedENRLocalNode = "updated_enr" @@ -413,3 +414,15 @@ func Type(v any) zapcore.Field { func FormatDuration(val time.Duration) string { return strconv.FormatFloat(val.Seconds(), 'f', 5, 64) } + +func FormatSlotTickerID(epoch phase0.Epoch, slot phase0.Slot) string { + return fmt.Sprintf("e%v-s%v-#%v", epoch, slot, slot%32+1) +} + +func FormatSlotTickerCommitteeID(period uint64, epoch phase0.Epoch, slot phase0.Slot) string { + return fmt.Sprintf("p%v-%s", period, FormatSlotTickerID(epoch, slot)) +} + +func SlotTickerID(val string) zap.Field { + return zap.String(FieldSlotTickerID, val) +} diff --git a/operator/duties/attester.go b/operator/duties/attester.go index 396ba70cd2..852a33b7da 100644 --- a/operator/duties/attester.go +++ b/operator/duties/attester.go @@ -21,13 +21,15 @@ type AttesterHandler struct { duties *dutystore.Duties[eth2apiv1.AttesterDuty] fetchCurrentEpoch bool fetchNextEpoch bool + + firstRun bool } func NewAttesterHandler(duties *dutystore.Duties[eth2apiv1.AttesterDuty]) *AttesterHandler { h := &AttesterHandler{ - duties: duties, + duties: duties, + firstRun: true, } - h.fetchCurrentEpoch = true return h } @@ -35,37 +37,11 @@ func (h *AttesterHandler) Name() string { return spectypes.BNRoleAttester.String() } -// HandleDuties manages the duty lifecycle, handling different cases: -// -// On First Run: -// 1. Fetch duties for the current epoch. -// 2. If necessary, fetch duties for the next epoch. -// 3. Execute duties. -// -// On Re-org: -// -// If the previous dependent root changed: -// 1. Fetch duties for the current epoch. -// 2. Execute duties. -// If the current dependent root changed: -// 1. Execute duties. -// 2. If necessary, fetch duties for the next epoch. -// -// On Indices Change: -// 1. Execute duties. -// 2. ResetEpoch duties for the current epoch. -// 3. Fetch duties for the current epoch. -// 4. If necessary, fetch duties for the next epoch. -// -// On Ticker event: -// 1. Execute duties. -// 2. If necessary, fetch duties for the next epoch. +// HandleDuties manages the duty lifecycle func (h *AttesterHandler) HandleDuties(ctx context.Context) { h.logger.Info("starting duty handler") defer h.logger.Info("duty handler exited") - h.fetchNextEpoch = true - next := h.ticker.Next() for { select { @@ -75,74 +51,52 @@ func (h *AttesterHandler) HandleDuties(ctx context.Context) { case <-next: slot := h.ticker.Slot() next = h.ticker.Next() - currentEpoch := h.network.Beacon.EstimatedEpochAtSlot(slot) - buildStr := fmt.Sprintf("e%v-s%v-#%v", currentEpoch, slot, slot%32+1) - h.logger.Debug("🛠 ticker event", zap.String("epoch_slot_pos", buildStr)) - - h.processExecution(currentEpoch, slot) - h.processFetching(ctx, currentEpoch, slot) - - slotsPerEpoch := h.network.Beacon.SlotsPerEpoch() + epoch := h.network.Beacon.EstimatedEpochAtSlot(slot) + tickerID := fields.FormatSlotTickerID(epoch, slot) + h.logger.Debug("🛠 ticker event", fields.SlotTickerID(tickerID)) - // If we have reached the mid-point of the epoch, fetch the duties for the next epoch in the next slot. - // This allows us to set them up at a time when the beacon node should be less busy. - if uint64(slot)%slotsPerEpoch == slotsPerEpoch/2-1 { - h.fetchNextEpoch = true - } - - // last slot of epoch - if uint64(slot)%slotsPerEpoch == slotsPerEpoch-1 { - h.duties.ResetEpoch(currentEpoch - 1) + if !h.network.PastAlanForkAtEpoch(epoch) { + if h.firstRun { + h.processFirstRun(ctx, epoch, slot) + } + h.processExecution(epoch, slot) + if h.indicesChanged { + h.processIndicesChange(epoch, slot) + } + h.processFetching(ctx, epoch, slot) + h.processSlotTransition(epoch, slot) } case reorgEvent := <-h.reorg: - currentEpoch := h.network.Beacon.EstimatedEpochAtSlot(reorgEvent.Slot) - buildStr := fmt.Sprintf("e%v-s%v-#%v", currentEpoch, reorgEvent.Slot, reorgEvent.Slot%32+1) - h.logger.Info("🔀 reorg event received", zap.String("epoch_slot_pos", buildStr), zap.Any("event", reorgEvent)) - - // reset current epoch duties - if reorgEvent.Previous { - h.duties.ResetEpoch(currentEpoch) - h.fetchCurrentEpoch = true - if h.shouldFetchNexEpoch(reorgEvent.Slot) { - h.duties.ResetEpoch(currentEpoch + 1) - h.fetchNextEpoch = true - } + epoch := h.network.Beacon.EstimatedEpochAtSlot(reorgEvent.Slot) + tickerID := fields.FormatSlotTickerID(epoch, reorgEvent.Slot) + h.logger.Info("🔀 reorg event received", fields.SlotTickerID(tickerID), zap.Any("event", reorgEvent)) - h.processFetching(ctx, currentEpoch, reorgEvent.Slot) - } else if reorgEvent.Current { - // reset & re-fetch next epoch duties if in appropriate slot range, - // otherwise they will be fetched by the appropriate slot tick. - if h.shouldFetchNexEpoch(reorgEvent.Slot) { - h.duties.ResetEpoch(currentEpoch + 1) - h.fetchNextEpoch = true - } + if !h.network.PastAlanForkAtEpoch(epoch) { + h.processReorg(ctx, epoch, reorgEvent) } case <-h.indicesChange: slot := h.network.Beacon.EstimatedCurrentSlot() - currentEpoch := h.network.Beacon.EstimatedEpochAtSlot(slot) - buildStr := fmt.Sprintf("e%v-s%v-#%v", currentEpoch, slot, slot%32+1) - h.logger.Info("🔁 indices change received", zap.String("epoch_slot_pos", buildStr)) - - h.fetchCurrentEpoch = true + epoch := h.network.Beacon.EstimatedEpochAtSlot(slot) + tickerID := fields.FormatSlotTickerID(epoch, slot) + h.logger.Info("🔁 indices change received", fields.SlotTickerID(tickerID)) - // reset next epoch duties if in appropriate slot range - if h.shouldFetchNexEpoch(slot) { - h.duties.ResetEpoch(currentEpoch + 1) - h.fetchNextEpoch = true + if !h.network.PastAlanForkAtEpoch(epoch) { + h.indicesChanged = true } } } } -func (h *AttesterHandler) HandleInitialDuties(ctx context.Context) { - ctx, cancel := context.WithTimeout(ctx, h.network.Beacon.SlotDurationSec()/2) - defer cancel() - - slot := h.network.Beacon.EstimatedCurrentSlot() - epoch := h.network.Beacon.EstimatedEpochAtSlot(slot) +func (h *AttesterHandler) processFirstRun(ctx context.Context, epoch phase0.Epoch, slot phase0.Slot) { + h.fetchCurrentEpoch = true h.processFetching(ctx, epoch, slot) + + if uint64(slot)%h.network.Beacon.SlotsPerEpoch() > h.network.Beacon.SlotsPerEpoch()/2-1 { + h.fetchNextEpoch = true + } + h.firstRun = false } func (h *AttesterHandler) processFetching(ctx context.Context, epoch phase0.Epoch, slot phase0.Slot) { @@ -157,7 +111,7 @@ func (h *AttesterHandler) processFetching(ctx context.Context, epoch phase0.Epoc h.fetchCurrentEpoch = false } - if h.fetchNextEpoch && h.shouldFetchNexEpoch(slot) { + if h.fetchNextEpoch { if err := h.fetchAndProcessDuties(ctx, epoch+1); err != nil { h.logger.Error("failed to fetch duties for next epoch", zap.Error(err)) return @@ -172,27 +126,63 @@ func (h *AttesterHandler) processExecution(epoch phase0.Epoch, slot phase0.Slot) return } - if !h.network.PastAlanForkAtEpoch(h.network.Beacon.EstimatedEpochAtSlot(slot)) { - toExecute := make([]*genesisspectypes.Duty, 0, len(duties)*2) - for _, d := range duties { - if h.shouldExecute(d) { - toExecute = append(toExecute, h.toGenesisSpecDuty(d, genesisspectypes.BNRoleAttester)) - toExecute = append(toExecute, h.toGenesisSpecDuty(d, genesisspectypes.BNRoleAggregator)) - } + toExecute := make([]*genesisspectypes.Duty, 0, len(duties)*2) + for _, d := range duties { + if h.shouldExecute(d) { + toExecute = append(toExecute, h.toGenesisSpecDuty(d, genesisspectypes.BNRoleAttester)) + toExecute = append(toExecute, h.toGenesisSpecDuty(d, genesisspectypes.BNRoleAggregator)) } + } - h.dutiesExecutor.ExecuteGenesisDuties(h.logger, toExecute) - return + h.dutiesExecutor.ExecuteGenesisDuties(h.logger, toExecute) +} + +func (h *AttesterHandler) processIndicesChange(epoch phase0.Epoch, slot phase0.Slot) { + h.fetchCurrentEpoch = true + + // reset next epoch duties if in appropriate slot range + if h.shouldFetchNexEpoch(slot) { + h.duties.Reset(epoch + 1) + h.fetchNextEpoch = true } - toExecute := make([]*spectypes.ValidatorDuty, 0, len(duties)) - for _, d := range duties { - if h.shouldExecute(d) { - toExecute = append(toExecute, h.toSpecDuty(d, spectypes.BNRoleAggregator)) + h.indicesChanged = false +} + +func (h *AttesterHandler) processReorg(ctx context.Context, epoch phase0.Epoch, reorgEvent ReorgEvent) { + // reset current epoch duties + if reorgEvent.Previous { + h.duties.Reset(epoch) + h.fetchCurrentEpoch = true + if h.shouldFetchNexEpoch(reorgEvent.Slot) { + h.duties.Reset(epoch + 1) + h.fetchNextEpoch = true + } + + h.processFetching(ctx, epoch, reorgEvent.Slot) + } else if reorgEvent.Current { + // reset & re-fetch next epoch duties if in appropriate slot range, + // otherwise they will be fetched by the appropriate slot tick. + if h.shouldFetchNexEpoch(reorgEvent.Slot) { + h.duties.Reset(epoch + 1) + h.fetchNextEpoch = true } } +} - h.dutiesExecutor.ExecuteDuties(h.logger, toExecute) +func (h *AttesterHandler) processSlotTransition(epoch phase0.Epoch, slot phase0.Slot) { + slotsPerEpoch := h.network.Beacon.SlotsPerEpoch() + + // If we have reached the mid-point of the epoch, fetch the duties for the next epoch in the next slot. + // This allows us to set them up at a time when the beacon node should be less busy. + if uint64(slot)%slotsPerEpoch == slotsPerEpoch/2-1 { + h.fetchNextEpoch = true + } + + // last slot of epoch + if uint64(slot)%slotsPerEpoch == slotsPerEpoch-1 { + h.duties.Reset(epoch - 1) + } } func (h *AttesterHandler) fetchAndProcessDuties(ctx context.Context, epoch phase0.Epoch) error { @@ -311,5 +301,5 @@ func toBeaconCommitteeSubscription(duty *eth2apiv1.AttesterDuty, role spectypes. } func (h *AttesterHandler) shouldFetchNexEpoch(slot phase0.Slot) bool { - return uint64(slot)%h.network.Beacon.SlotsPerEpoch() > h.network.Beacon.SlotsPerEpoch()/2-2 + return uint64(slot)%h.network.Beacon.SlotsPerEpoch() >= h.network.Beacon.SlotsPerEpoch()/2-1 } diff --git a/operator/duties/attester_genesis_test.go b/operator/duties/attester_genesis_test.go index e43e0ca544..662c7c5553 100644 --- a/operator/duties/attester_genesis_test.go +++ b/operator/duties/attester_genesis_test.go @@ -19,19 +19,16 @@ import ( "github.com/ssvlabs/ssv/protocol/v2/types" ) -func setupAttesterGenesisDutiesMock( +func setupDutiesMockAttesterGenesis( s *Scheduler, dutiesMap *hashmap.Map[phase0.Epoch, []*eth2apiv1.AttesterDuty], - waitForDuties *SafeValue[bool], ) (chan struct{}, chan []*genesisspectypes.Duty) { fetchDutiesCall := make(chan struct{}) executeDutiesCall := make(chan []*genesisspectypes.Duty) s.beaconNode.(*MockBeaconNode).EXPECT().AttesterDuties(gomock.Any(), gomock.Any(), gomock.Any()).DoAndReturn( func(ctx context.Context, epoch phase0.Epoch, indices []phase0.ValidatorIndex) ([]*eth2apiv1.AttesterDuty, error) { - if waitForDuties.Get() { - fetchDutiesCall <- struct{}{} - } + fetchDutiesCall <- struct{}{} duties, _ := dutiesMap.Get(epoch) return duties, nil }).AnyTimes() @@ -64,7 +61,7 @@ func setupAttesterGenesisDutiesMock( return fetchDutiesCall, executeDutiesCall } -func expectedExecutedGenesisAttesterDuties(handler *AttesterHandler, duties []*eth2apiv1.AttesterDuty) []*genesisspectypes.Duty { +func expectedExecutedDutiesAttesterGenesis(handler *AttesterHandler, duties []*eth2apiv1.AttesterDuty) []*genesisspectypes.Duty { expectedDuties := make([]*genesisspectypes.Duty, 0) for _, d := range duties { expectedDuties = append(expectedDuties, handler.toGenesisSpecDuty(d, genesisspectypes.BNRoleAttester)) @@ -75,11 +72,10 @@ func expectedExecutedGenesisAttesterDuties(handler *AttesterHandler, duties []*e func TestScheduler_Attester_Genesis_Same_Slot(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + forkEpoch = goclient.FarFutureEpoch ) dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.AttesterDuty{ { @@ -91,16 +87,17 @@ func TestScheduler_Attester_Genesis_Same_Slot(t *testing.T) { currentSlot.Set(phase0.Slot(1)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedGenesisAttesterDuties(handler, duties) + expected := expectedExecutedDutiesAttesterGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) startTime := time.Now() ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) require.Less(t, scheduler.network.Beacon.SlotDurationSec()/3, time.Since(startTime)) @@ -111,11 +108,10 @@ func TestScheduler_Attester_Genesis_Same_Slot(t *testing.T) { func TestScheduler_Attester_Genesis_Diff_Slots(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + forkEpoch = goclient.FarFutureEpoch ) dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.AttesterDuty{ { @@ -127,11 +123,11 @@ func TestScheduler_Attester_Genesis_Diff_Slots(t *testing.T) { currentSlot.Set(phase0.Slot(0)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() ticker.Send(currentSlot.Get()) - waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) currentSlot.Set(phase0.Slot(1)) ticker.Send(currentSlot.Get()) @@ -139,11 +135,11 @@ func TestScheduler_Attester_Genesis_Diff_Slots(t *testing.T) { currentSlot.Set(phase0.Slot(2)) duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedGenesisAttesterDuties(handler, duties) + expected := expectedExecutedDutiesAttesterGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -152,19 +148,18 @@ func TestScheduler_Attester_Genesis_Diff_Slots(t *testing.T) { func TestScheduler_Attester_Genesis_Indices_Changed(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + forkEpoch = goclient.FarFutureEpoch ) currentSlot.Set(phase0.Slot(0)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() // STEP 1: wait for no action to be taken - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger a change in active indices @@ -191,20 +186,19 @@ func TestScheduler_Attester_Genesis_Indices_Changed(t *testing.T) { // STEP 3: wait for attester duties to be fetched again currentSlot.Set(phase0.Slot(1)) - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // no execution should happen in slot 1 waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 4: wait for attester duties to be executed currentSlot.Set(phase0.Slot(2)) duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedGenesisAttesterDuties(handler, []*eth2apiv1.AttesterDuty{duties[2]}) + expected := expectedExecutedDutiesAttesterGenesis(handler, []*eth2apiv1.AttesterDuty{duties[2]}) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -213,24 +207,23 @@ func TestScheduler_Attester_Genesis_Indices_Changed(t *testing.T) { func TestScheduler_Attester_Genesis_Multiple_Indices_Changed_Same_Slot(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + forkEpoch = goclient.FarFutureEpoch ) currentSlot.Set(phase0.Slot(0)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() // STEP 1: wait for no action to be taken - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: wait for no action to be taken currentSlot.Set(phase0.Slot(1)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 3: trigger a change in active indices @@ -255,27 +248,26 @@ func TestScheduler_Attester_Genesis_Multiple_Indices_Changed_Same_Slot(t *testin // STEP 5: wait for attester duties to be fetched currentSlot.Set(phase0.Slot(2)) - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 6: wait for attester duties to be executed currentSlot.Set(phase0.Slot(3)) duties, _ = dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedGenesisAttesterDuties(handler, []*eth2apiv1.AttesterDuty{duties[0]}) + expected := expectedExecutedDutiesAttesterGenesis(handler, []*eth2apiv1.AttesterDuty{duties[0]}) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // STEP 7: wait for attester duties to be executed currentSlot.Set(phase0.Slot(4)) duties, _ = dutiesMap.Get(phase0.Epoch(0)) - expected = expectedExecutedGenesisAttesterDuties(handler, []*eth2apiv1.AttesterDuty{duties[1]}) + expected = expectedExecutedDutiesAttesterGenesis(handler, []*eth2apiv1.AttesterDuty{duties[1]}) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -285,15 +277,14 @@ func TestScheduler_Attester_Genesis_Multiple_Indices_Changed_Same_Slot(t *testin // reorg previous dependent root changed func TestScheduler_Attester_Genesis_Reorg_Previous_Epoch_Transition(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + forkEpoch = goclient.FarFutureEpoch ) currentSlot.Set(phase0.Slot(63)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ @@ -305,9 +296,8 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Epoch_Transition(t *testing.T }) // STEP 1: wait for attester duties to be fetched for next epoch - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger head event e := ð2apiv1.Event{ @@ -322,7 +312,7 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Epoch_Transition(t *testing.T // STEP 3: Ticker with no action currentSlot.Set(phase0.Slot(64)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 4: trigger reorg on epoch transition @@ -340,26 +330,26 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Epoch_Transition(t *testing.T }, }) scheduler.HandleHeadEvent(logger)(e) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 5: wait for attester duties to be fetched again for the current epoch currentSlot.Set(phase0.Slot(65)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 6: The first assigned duty should not be executed currentSlot.Set(phase0.Slot(66)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 7: The second assigned duty should be executed currentSlot.Set(phase0.Slot(67)) duties, _ := dutiesMap.Get(phase0.Epoch(2)) - expected := expectedExecutedGenesisAttesterDuties(handler, duties) + expected := expectedExecutedDutiesAttesterGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -369,15 +359,14 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Epoch_Transition(t *testing.T // reorg previous dependent root changed and the indices changed as well func TestScheduler_Attester_Genesis_Reorg_Previous_Epoch_Transition_Indices_Changed(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + forkEpoch = goclient.FarFutureEpoch ) currentSlot.Set(phase0.Slot(63)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ @@ -389,9 +378,8 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Epoch_Transition_Indices_Chan }) // STEP 1: wait for attester duties to be fetched for next epoch - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger head event @@ -407,7 +395,7 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Epoch_Transition_Indices_Chan // STEP 3: Ticker with no action currentSlot.Set(phase0.Slot(64)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 4: trigger reorg on epoch transition @@ -425,7 +413,7 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Epoch_Transition_Indices_Chan }, }) scheduler.HandleHeadEvent(logger)(e) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 5: trigger indices change scheduler.indicesChg <- struct{}{} @@ -439,22 +427,22 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Epoch_Transition_Indices_Chan // STEP 6: wait for attester duties to be fetched again for the current epoch currentSlot.Set(phase0.Slot(65)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 7: The first assigned duty should not be executed currentSlot.Set(phase0.Slot(66)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 8: The second assigned duty should be executed currentSlot.Set(phase0.Slot(67)) duties, _ = dutiesMap.Get(phase0.Epoch(2)) - expected := expectedExecutedGenesisAttesterDuties(handler, duties) + expected := expectedExecutedDutiesAttesterGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -464,11 +452,10 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Epoch_Transition_Indices_Chan // reorg previous dependent root changed func TestScheduler_Attester_Genesis_Reorg_Previous(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + forkEpoch = goclient.FarFutureEpoch ) dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ { @@ -480,12 +467,12 @@ func TestScheduler_Attester_Genesis_Reorg_Previous(t *testing.T) { currentSlot.Set(phase0.Slot(32)) // STEP 1: wait for attester duties to be fetched (handle initial duties) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() - mockTicker.Send(currentSlot.Get()) - waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger head event e := ð2apiv1.Event{ @@ -498,9 +485,8 @@ func TestScheduler_Attester_Genesis_Reorg_Previous(t *testing.T) { waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 3: Ticker with no action - waitForDuties.Set(true) currentSlot.Set(phase0.Slot(33)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 4: trigger reorg @@ -518,26 +504,26 @@ func TestScheduler_Attester_Genesis_Reorg_Previous(t *testing.T) { }, }) scheduler.HandleHeadEvent(logger)(e) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 5: wait for no action to be taken currentSlot.Set(phase0.Slot(34)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 6: The first assigned duty should not be executed currentSlot.Set(phase0.Slot(35)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 7: The second assigned duty should be executed currentSlot.Set(phase0.Slot(36)) duties, _ := dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedGenesisAttesterDuties(handler, duties) + expected := expectedExecutedDutiesAttesterGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -547,11 +533,10 @@ func TestScheduler_Attester_Genesis_Reorg_Previous(t *testing.T) { // reorg previous dependent root changed and the indices changed the same slot func TestScheduler_Attester_Genesis_Reorg_Previous_Indices_Change_Same_Slot(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + forkEpoch = goclient.FarFutureEpoch ) dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ { @@ -563,12 +548,12 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Indices_Change_Same_Slot(t *t currentSlot.Set(phase0.Slot(32)) // STEP 1: wait for attester duties to be fetched (handle initial duties) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() - mockTicker.Send(currentSlot.Get()) - waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger head event e := ð2apiv1.Event{ @@ -582,8 +567,7 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Indices_Change_Same_Slot(t *t // STEP 3: Ticker with no action currentSlot.Set(phase0.Slot(33)) - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 4: trigger reorg @@ -601,7 +585,7 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Indices_Change_Same_Slot(t *t }, }) scheduler.HandleHeadEvent(logger)(e) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 5: trigger indices change scheduler.indicesChg <- struct{}{} @@ -615,22 +599,22 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Indices_Change_Same_Slot(t *t // STEP 6: wait for attester duties to be fetched again for the current epoch currentSlot.Set(phase0.Slot(34)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 7: The first assigned duty should not be executed currentSlot.Set(phase0.Slot(35)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 8: The second and new from indices change assigned duties should be executed currentSlot.Set(phase0.Slot(36)) duties, _ = dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedGenesisAttesterDuties(handler, duties) + expected := expectedExecutedDutiesAttesterGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -640,15 +624,14 @@ func TestScheduler_Attester_Genesis_Reorg_Previous_Indices_Change_Same_Slot(t *t // reorg current dependent root changed func TestScheduler_Attester_Genesis_Reorg_Current(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + forkEpoch = goclient.FarFutureEpoch ) currentSlot.Set(phase0.Slot(48)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ @@ -660,9 +643,8 @@ func TestScheduler_Attester_Genesis_Reorg_Current(t *testing.T) { }) // STEP 1: wait for attester duties to be fetched for next epoch - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger head event e := ð2apiv1.Event{ @@ -676,7 +658,7 @@ func TestScheduler_Attester_Genesis_Reorg_Current(t *testing.T) { // STEP 3: Ticker with no action currentSlot.Set(phase0.Slot(49)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 4: trigger reorg @@ -698,30 +680,30 @@ func TestScheduler_Attester_Genesis_Reorg_Current(t *testing.T) { // STEP 5: wait for attester duties to be fetched again for the current epoch currentSlot.Set(phase0.Slot(50)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 6: skip to the next epoch currentSlot.Set(phase0.Slot(51)) for slot := currentSlot.Get(); slot < 64; slot++ { - mockTicker.Send(slot) + ticker.Send(slot) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) currentSlot.Set(slot + 1) } // STEP 7: The first assigned duty should not be executed // slot = 64 - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 8: The second assigned duty should be executed currentSlot.Set(phase0.Slot(65)) duties, _ := dutiesMap.Get(phase0.Epoch(2)) - expected := expectedExecutedGenesisAttesterDuties(handler, duties) + expected := expectedExecutedDutiesAttesterGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -731,15 +713,15 @@ func TestScheduler_Attester_Genesis_Reorg_Current(t *testing.T) { // reorg current dependent root changed including indices change in the same slot func TestScheduler_Attester_Genesis_Reorg_Current_Indices_Changed(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + + forkEpoch = goclient.FarFutureEpoch ) currentSlot.Set(phase0.Slot(48)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ @@ -751,9 +733,8 @@ func TestScheduler_Attester_Genesis_Reorg_Current_Indices_Changed(t *testing.T) }) // STEP 1: wait for attester duties to be fetched for next epoch - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger head event e := ð2apiv1.Event{ @@ -767,7 +748,7 @@ func TestScheduler_Attester_Genesis_Reorg_Current_Indices_Changed(t *testing.T) // STEP 3: Ticker with no action currentSlot.Set(phase0.Slot(49)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 4: trigger reorg @@ -799,30 +780,30 @@ func TestScheduler_Attester_Genesis_Reorg_Current_Indices_Changed(t *testing.T) // STEP 6: wait for attester duties to be fetched again for the next epoch due to indices change currentSlot.Set(phase0.Slot(50)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 7: skip to the next epoch currentSlot.Set(phase0.Slot(51)) for slot := currentSlot.Get(); slot < 64; slot++ { - mockTicker.Send(slot) + ticker.Send(slot) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) currentSlot.Set(slot + 1) } // STEP 8: The first assigned duty should not be executed // slot = 64 - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 9: The second assigned duty should be executed currentSlot.Set(phase0.Slot(65)) duties, _ = dutiesMap.Get(phase0.Epoch(2)) - expected := expectedExecutedGenesisAttesterDuties(handler, duties) + expected := expectedExecutedDutiesAttesterGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -831,11 +812,10 @@ func TestScheduler_Attester_Genesis_Reorg_Current_Indices_Changed(t *testing.T) func TestScheduler_Attester_Genesis_Early_Block(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + forkEpoch = goclient.FarFutureEpoch ) dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.AttesterDuty{ { @@ -847,24 +827,24 @@ func TestScheduler_Attester_Genesis_Early_Block(t *testing.T) { currentSlot.Set(phase0.Slot(0)) // STEP 1: wait for attester duties to be fetched (handle initial duties) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() - mockTicker.Send(currentSlot.Get()) - waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: wait for no action to be taken currentSlot.Set(phase0.Slot(1)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 3: wait for attester duties to be executed faster than 1/3 of the slot duration startTime := time.Now() currentSlot.Set(phase0.Slot(2)) - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedGenesisAttesterDuties(handler, duties) + expected := expectedExecutedDutiesAttesterGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) // STEP 4: trigger head event (block arrival) @@ -874,7 +854,7 @@ func TestScheduler_Attester_Genesis_Early_Block(t *testing.T) { }, } scheduler.HandleHeadEvent(logger)(e) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) require.Less(t, time.Since(startTime), scheduler.network.Beacon.SlotDurationSec()/3) // Stop scheduler & wait for graceful exit. @@ -884,15 +864,15 @@ func TestScheduler_Attester_Genesis_Early_Block(t *testing.T) { func TestScheduler_Attester_Genesis_Start_In_The_End_Of_The_Epoch(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + + forkEpoch = goclient.FarFutureEpoch ) currentSlot.Set(phase0.Slot(31)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ @@ -904,18 +884,17 @@ func TestScheduler_Attester_Genesis_Start_In_The_End_Of_The_Epoch(t *testing.T) }) // STEP 1: wait for attester duties to be fetched for the next epoch - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: wait for attester duties to be executed currentSlot.Set(phase0.Slot(32)) duties, _ := dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedGenesisAttesterDuties(handler, duties) + expected := expectedExecutedDutiesAttesterGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -924,15 +903,14 @@ func TestScheduler_Attester_Genesis_Start_In_The_End_Of_The_Epoch(t *testing.T) func TestScheduler_Attester_Genesis_Fetch_Execute_Next_Epoch_Duty(t *testing.T) { var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch + handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) + currentSlot = &SafeValue[phase0.Slot]{} + dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + forkEpoch = goclient.FarFutureEpoch ) - currentSlot.Set(phase0.Slot(13)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterGenesisDutiesMock(scheduler, dutiesMap, waitForDuties) + currentSlot.Set(phase0.Slot(14)) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockAttesterGenesis(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ @@ -944,28 +922,27 @@ func TestScheduler_Attester_Genesis_Fetch_Execute_Next_Epoch_Duty(t *testing.T) }) // STEP 1: wait for no action to be taken - mockTicker.Send(currentSlot.Get()) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: wait for no action to be taken - currentSlot.Set(phase0.Slot(14)) - mockTicker.Send(currentSlot.Get()) + currentSlot.Set(phase0.Slot(15)) + ticker.Send(currentSlot.Get()) waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - // STEP 2: wait for duties to be fetched for the next epoch - currentSlot.Set(phase0.Slot(15)) - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // STEP 3: wait for duties to be fetched for the next epoch + currentSlot.Set(phase0.Slot(16)) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - // STEP 3: wait for attester duties to be executed + // STEP 4: wait for attester duties to be executed currentSlot.Set(phase0.Slot(32)) duties, _ := dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedGenesisAttesterDuties(handler, duties) + expected := expectedExecutedDutiesAttesterGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) - mockTicker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() diff --git a/operator/duties/attester_test.go b/operator/duties/attester_test.go deleted file mode 100644 index cd89fcdce6..0000000000 --- a/operator/duties/attester_test.go +++ /dev/null @@ -1,967 +0,0 @@ -package duties - -import ( - "context" - "testing" - "time" - - eth2apiv1 "github.com/attestantio/go-eth2-client/api/v1" - "github.com/attestantio/go-eth2-client/spec/phase0" - spectypes "github.com/ssvlabs/ssv-spec/types" - "github.com/stretchr/testify/require" - "go.uber.org/mock/gomock" - - "github.com/ssvlabs/ssv/utils/hashmap" - - "github.com/ssvlabs/ssv/operator/duties/dutystore" - "github.com/ssvlabs/ssv/protocol/v2/types" -) - -func setupAttesterDutiesMock( - s *Scheduler, - dutiesMap *hashmap.Map[phase0.Epoch, []*eth2apiv1.AttesterDuty], - waitForDuties *SafeValue[bool], -) (chan struct{}, chan []*spectypes.ValidatorDuty) { - fetchDutiesCall := make(chan struct{}) - executeDutiesCall := make(chan []*spectypes.ValidatorDuty) - - s.beaconNode.(*MockBeaconNode).EXPECT().AttesterDuties(gomock.Any(), gomock.Any(), gomock.Any()).DoAndReturn( - func(ctx context.Context, epoch phase0.Epoch, indices []phase0.ValidatorIndex) ([]*eth2apiv1.AttesterDuty, error) { - if waitForDuties.Get() { - fetchDutiesCall <- struct{}{} - } - duties, _ := dutiesMap.Get(epoch) - return duties, nil - }).AnyTimes() - - getShares := func(epoch phase0.Epoch) []*types.SSVShare { - uniqueIndices := make(map[phase0.ValidatorIndex]bool) - - duties, _ := dutiesMap.Get(epoch) - for _, d := range duties { - uniqueIndices[d.ValidatorIndex] = true - } - - shares := make([]*types.SSVShare, 0, len(uniqueIndices)) - for index := range uniqueIndices { - share := &types.SSVShare{ - Share: spectypes.Share{ - ValidatorIndex: index, - }, - } - shares = append(shares, share) - } - - return shares - } - s.validatorProvider.(*MockValidatorProvider).EXPECT().SelfParticipatingValidators(gomock.Any()).DoAndReturn(getShares).AnyTimes() - s.validatorProvider.(*MockValidatorProvider).EXPECT().ParticipatingValidators(gomock.Any()).DoAndReturn(getShares).AnyTimes() - - s.beaconNode.(*MockBeaconNode).EXPECT().SubmitBeaconCommitteeSubscriptions(gomock.Any(), gomock.Any()).Return(nil).AnyTimes() - - return fetchDutiesCall, executeDutiesCall -} - -func expectedExecutedAttesterDuties(handler *AttesterHandler, duties []*eth2apiv1.AttesterDuty) []*spectypes.ValidatorDuty { - expectedDuties := make([]*spectypes.ValidatorDuty, 0) - for _, d := range duties { - expectedDuties = append(expectedDuties, handler.toSpecDuty(d, spectypes.BNRoleAggregator)) - } - return expectedDuties -} - -func TestScheduler_Attester_Same_Slot(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(1), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - currentSlot.Set(phase0.Slot(1)) - - scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedAttesterDuties(handler, duties) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -func TestScheduler_Attester_Diff_Slots(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(2), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - currentSlot.Set(phase0.Slot(0)) - - scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - ticker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - currentSlot.Set(phase0.Slot(1)) - ticker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - currentSlot.Set(phase0.Slot(2)) - duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedAttesterDuties(handler, duties) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -func TestScheduler_Attester_Indices_Changed(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - currentSlot.Set(phase0.Slot(0)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - // STEP 1: wait for no action to be taken - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: trigger a change in active indices - scheduler.indicesChg <- struct{}{} - // no execution should happen in slot 0 - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(0), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - { - PubKey: phase0.BLSPubKey{1, 2, 4}, - Slot: phase0.Slot(1), - ValidatorIndex: phase0.ValidatorIndex(2), - }, - { - PubKey: phase0.BLSPubKey{1, 2, 5}, - Slot: phase0.Slot(2), - ValidatorIndex: phase0.ValidatorIndex(3), - }, - }) - - // STEP 3: wait for attester duties to be fetched again - currentSlot.Set(phase0.Slot(1)) - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - // no execution should happen in slot 1 - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 4: wait for attester duties to be executed - currentSlot.Set(phase0.Slot(2)) - duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedAttesterDuties(handler, []*eth2apiv1.AttesterDuty{duties[2]}) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - mockTicker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -func TestScheduler_Attester_Multiple_Indices_Changed_Same_Slot(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - currentSlot.Set(phase0.Slot(0)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - // STEP 1: wait for no action to be taken - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: wait for no action to be taken - currentSlot.Set(phase0.Slot(1)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: trigger a change in active indices - scheduler.indicesChg <- struct{}{} - duties, _ := dutiesMap.Get(phase0.Epoch(0)) - dutiesMap.Set(phase0.Epoch(0), append(duties, ð2apiv1.AttesterDuty{ - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(3), - ValidatorIndex: phase0.ValidatorIndex(1), - })) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 4: trigger a change in active indices in the same slot - scheduler.indicesChg <- struct{}{} - duties, _ = dutiesMap.Get(phase0.Epoch(0)) - dutiesMap.Set(phase0.Epoch(0), append(duties, ð2apiv1.AttesterDuty{ - PubKey: phase0.BLSPubKey{1, 2, 4}, - Slot: phase0.Slot(4), - ValidatorIndex: phase0.ValidatorIndex(2), - })) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 5: wait for attester duties to be fetched - currentSlot.Set(phase0.Slot(2)) - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 6: wait for attester duties to be executed - currentSlot.Set(phase0.Slot(3)) - duties, _ = dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedAttesterDuties(handler, []*eth2apiv1.AttesterDuty{duties[0]}) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - mockTicker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // STEP 7: wait for attester duties to be executed - currentSlot.Set(phase0.Slot(4)) - duties, _ = dutiesMap.Get(phase0.Epoch(0)) - expected = expectedExecutedAttesterDuties(handler, []*eth2apiv1.AttesterDuty{duties[1]}) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - mockTicker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -// reorg previous dependent root changed -func TestScheduler_Attester_Reorg_Previous_Epoch_Transition(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - currentSlot.Set(phase0.Slot(63)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - dutiesMap.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(66), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - - // STEP 1: wait for attester duties to be fetched for next epoch - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: trigger head event - e := ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - CurrentDutyDependentRoot: phase0.Root{0x01}, - PreviousDutyDependentRoot: phase0.Root{0x01}, - }, - } - scheduler.HandleHeadEvent(logger)(e) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: Ticker with no action - currentSlot.Set(phase0.Slot(64)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 4: trigger reorg on epoch transition - e = ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - PreviousDutyDependentRoot: phase0.Root{0x02}, - }, - } - dutiesMap.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(67), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - scheduler.HandleHeadEvent(logger)(e) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 5: wait for attester duties to be fetched again for the current epoch - currentSlot.Set(phase0.Slot(65)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 6: The first assigned duty should not be executed - currentSlot.Set(phase0.Slot(66)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 7: The second assigned duty should be executed - currentSlot.Set(phase0.Slot(67)) - duties, _ := dutiesMap.Get(phase0.Epoch(2)) - expected := expectedExecutedAttesterDuties(handler, duties) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - mockTicker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -// reorg previous dependent root changed and the indices changed as well -func TestScheduler_Attester_Reorg_Previous_Epoch_Transition_Indices_Changed(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - currentSlot.Set(phase0.Slot(63)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - dutiesMap.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(66), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - - // STEP 1: wait for attester duties to be fetched for next epoch - mockTicker.Send(currentSlot.Get()) - waitForDuties.Set(true) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: trigger head event - e := ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - CurrentDutyDependentRoot: phase0.Root{0x01}, - PreviousDutyDependentRoot: phase0.Root{0x01}, - }, - } - scheduler.HandleHeadEvent(logger)(e) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: Ticker with no action - currentSlot.Set(phase0.Slot(64)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 4: trigger reorg on epoch transition - e = ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - PreviousDutyDependentRoot: phase0.Root{0x02}, - }, - } - dutiesMap.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(67), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - scheduler.HandleHeadEvent(logger)(e) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 5: trigger indices change - scheduler.indicesChg <- struct{}{} - duties, _ := dutiesMap.Get(phase0.Epoch(2)) - dutiesMap.Set(phase0.Epoch(2), append(duties, ð2apiv1.AttesterDuty{ - PubKey: phase0.BLSPubKey{1, 2, 4}, - Slot: phase0.Slot(67), - ValidatorIndex: phase0.ValidatorIndex(2), - })) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 6: wait for attester duties to be fetched again for the current epoch - currentSlot.Set(phase0.Slot(65)) - mockTicker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 7: The first assigned duty should not be executed - currentSlot.Set(phase0.Slot(66)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 8: The second assigned duty should be executed - currentSlot.Set(phase0.Slot(67)) - duties, _ = dutiesMap.Get(phase0.Epoch(2)) - expected := expectedExecutedAttesterDuties(handler, duties) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - mockTicker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -// reorg previous dependent root changed -func TestScheduler_Attester_Reorg_Previous(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(35), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - currentSlot.Set(phase0.Slot(32)) - - // STEP 1: wait for attester duties to be fetched (handle initial duties) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: trigger head event - e := ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - PreviousDutyDependentRoot: phase0.Root{0x01}, - }, - } - scheduler.HandleHeadEvent(logger)(e) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: Ticker with no action - currentSlot.Set(phase0.Slot(33)) - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 4: trigger reorg - e = ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - PreviousDutyDependentRoot: phase0.Root{0x02}, - }, - } - dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(36), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - scheduler.HandleHeadEvent(logger)(e) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 5: wait for no action to be taken - currentSlot.Set(phase0.Slot(34)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 6: The first assigned duty should not be executed - currentSlot.Set(phase0.Slot(35)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 7: The second assigned duty should be executed - currentSlot.Set(phase0.Slot(36)) - duties, _ := dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedAttesterDuties(handler, duties) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - mockTicker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -// reorg previous dependent root changed and the indices changed the same slot -func TestScheduler_Attester_Reorg_Previous_Indices_Change_Same_Slot(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(35), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - currentSlot.Set(phase0.Slot(32)) - - // STEP 1: wait for attester duties to be fetched (handle initial duties) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: trigger head event - e := ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - PreviousDutyDependentRoot: phase0.Root{0x01}, - }, - } - scheduler.HandleHeadEvent(logger)(e) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: Ticker with no action - currentSlot.Set(phase0.Slot(33)) - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 4: trigger reorg - e = ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - PreviousDutyDependentRoot: phase0.Root{0x02}, - }, - } - dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(36), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - scheduler.HandleHeadEvent(logger)(e) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 5: trigger indices change - scheduler.indicesChg <- struct{}{} - duties, _ := dutiesMap.Get(phase0.Epoch(1)) - dutiesMap.Set(phase0.Epoch(1), append(duties, ð2apiv1.AttesterDuty{ - PubKey: phase0.BLSPubKey{1, 2, 4}, - Slot: phase0.Slot(36), - ValidatorIndex: phase0.ValidatorIndex(2), - })) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 6: wait for attester duties to be fetched again for the current epoch - currentSlot.Set(phase0.Slot(34)) - mockTicker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 7: The first assigned duty should not be executed - currentSlot.Set(phase0.Slot(35)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 8: The second and new from indices change assigned duties should be executed - currentSlot.Set(phase0.Slot(36)) - duties, _ = dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedAttesterDuties(handler, duties) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - mockTicker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -// reorg current dependent root changed -func TestScheduler_Attester_Reorg_Current(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - currentSlot.Set(phase0.Slot(48)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - dutiesMap.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(64), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - - // STEP 1: wait for attester duties to be fetched for next epoch - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: trigger head event - e := ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - CurrentDutyDependentRoot: phase0.Root{0x01}, - }, - } - scheduler.HandleHeadEvent(logger)(e) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: Ticker with no action - currentSlot.Set(phase0.Slot(49)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 4: trigger reorg - e = ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - CurrentDutyDependentRoot: phase0.Root{0x02}, - }, - } - dutiesMap.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(65), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - scheduler.HandleHeadEvent(logger)(e) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 5: wait for attester duties to be fetched again for the current epoch - currentSlot.Set(phase0.Slot(50)) - mockTicker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 6: skip to the next epoch - currentSlot.Set(phase0.Slot(51)) - for slot := currentSlot.Get(); slot < 64; slot++ { - mockTicker.Send(slot) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - currentSlot.Set(slot + 1) - } - - // STEP 7: The first assigned duty should not be executed - // slot = 64 - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 8: The second assigned duty should be executed - currentSlot.Set(phase0.Slot(65)) - duties, _ := dutiesMap.Get(phase0.Epoch(2)) - expected := expectedExecutedAttesterDuties(handler, duties) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - mockTicker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -// reorg current dependent root changed including indices change in the same slot -func TestScheduler_Attester_Reorg_Current_Indices_Changed(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - currentSlot.Set(phase0.Slot(48)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - dutiesMap.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(64), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - - // STEP 1: wait for attester duties to be fetched for next epoch - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: trigger head event - e := ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - CurrentDutyDependentRoot: phase0.Root{0x01}, - }, - } - scheduler.HandleHeadEvent(logger)(e) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: Ticker with no action - currentSlot.Set(phase0.Slot(49)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 4: trigger reorg - e = ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - CurrentDutyDependentRoot: phase0.Root{0x02}, - }, - } - dutiesMap.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(65), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - scheduler.HandleHeadEvent(logger)(e) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 5: trigger indices change - scheduler.indicesChg <- struct{}{} - duties, _ := dutiesMap.Get(phase0.Epoch(2)) - dutiesMap.Set(phase0.Epoch(2), append(duties, ð2apiv1.AttesterDuty{ - PubKey: phase0.BLSPubKey{1, 2, 4}, - Slot: phase0.Slot(65), - ValidatorIndex: phase0.ValidatorIndex(2), - })) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 6: wait for attester duties to be fetched again for the next epoch due to indices change - currentSlot.Set(phase0.Slot(50)) - mockTicker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 7: skip to the next epoch - currentSlot.Set(phase0.Slot(51)) - for slot := currentSlot.Get(); slot < 64; slot++ { - mockTicker.Send(slot) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - currentSlot.Set(slot + 1) - } - - // STEP 8: The first assigned duty should not be executed - // slot = 64 - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 9: The second assigned duty should be executed - currentSlot.Set(phase0.Slot(65)) - duties, _ = dutiesMap.Get(phase0.Epoch(2)) - expected := expectedExecutedAttesterDuties(handler, duties) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - mockTicker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -func TestScheduler_Attester_Early_Block(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(2), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - currentSlot.Set(phase0.Slot(0)) - - // STEP 1: wait for attester duties to be fetched (handle initial duties) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: wait for no action to be taken - currentSlot.Set(phase0.Slot(1)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: wait for attester duties to be executed faster than 1/3 of the slot duration - startTime := time.Now() - currentSlot.Set(phase0.Slot(2)) - mockTicker.Send(currentSlot.Get()) - duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedAttesterDuties(handler, duties) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - // STEP 4: trigger head event (block arrival) - e := ð2apiv1.Event{ - Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), - }, - } - scheduler.HandleHeadEvent(logger)(e) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - require.Less(t, time.Since(startTime), scheduler.network.Beacon.SlotDurationSec()/3) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -func TestScheduler_Attester_Start_In_The_End_Of_The_Epoch(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - currentSlot.Set(phase0.Slot(31)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(32), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - - // STEP 1: wait for attester duties to be fetched for the next epoch - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: wait for attester duties to be executed - currentSlot.Set(phase0.Slot(32)) - duties, _ := dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedAttesterDuties(handler, duties) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - mockTicker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -func TestScheduler_Attester_Fetch_Execute_Next_Epoch_Duty(t *testing.T) { - var ( - handler = NewAttesterHandler(dutystore.NewDuties[eth2apiv1.AttesterDuty]()) - currentSlot = &SafeValue[phase0.Slot]{} - dutiesMap = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - ) - currentSlot.Set(phase0.Slot(13)) - scheduler, logger, mockTicker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupAttesterDutiesMock(scheduler, dutiesMap, waitForDuties) - startFn() - - dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(32), - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - - // STEP 1: wait for no action to be taken - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: wait for no action to be taken - currentSlot.Set(phase0.Slot(14)) - mockTicker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: wait for duties to be fetched for the next epoch - currentSlot.Set(phase0.Slot(15)) - waitForDuties.Set(true) - mockTicker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: wait for attester duties to be executed - currentSlot.Set(phase0.Slot(32)) - duties, _ := dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedAttesterDuties(handler, duties) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - mockTicker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} diff --git a/operator/duties/committee.go b/operator/duties/committee.go index 1ada1e7a29..0a0b625f25 100644 --- a/operator/duties/committee.go +++ b/operator/duties/committee.go @@ -2,14 +2,14 @@ package duties import ( "context" - "fmt" + "sync" eth2apiv1 "github.com/attestantio/go-eth2-client/api/v1" "github.com/attestantio/go-eth2-client/spec/phase0" spectypes "github.com/ssvlabs/ssv-spec/types" "go.uber.org/zap" - "github.com/ssvlabs/ssv/operator/duties/dutystore" + "github.com/ssvlabs/ssv/logging/fields" ) type validatorCommitteeDutyMap map[phase0.ValidatorIndex]*committeeDuty @@ -18,8 +18,10 @@ type committeeDutiesMap map[spectypes.CommitteeID]*committeeDuty type CommitteeHandler struct { baseHandler - attDuties *dutystore.Duties[eth2apiv1.AttesterDuty] - syncDuties *dutystore.SyncCommitteeDuties + attHandler *AttesterHandler + syncHandler *SyncCommitteeHandler + + firstRun bool } type committeeDuty struct { @@ -28,17 +30,18 @@ type committeeDuty struct { operatorIDs []spectypes.OperatorID } -func NewCommitteeHandler(attDuties *dutystore.Duties[eth2apiv1.AttesterDuty], syncDuties *dutystore.SyncCommitteeDuties) *CommitteeHandler { +func NewCommitteeHandler(attHandler *AttesterHandler, syncHandler *SyncCommitteeHandler) *CommitteeHandler { h := &CommitteeHandler{ - attDuties: attDuties, - syncDuties: syncDuties, + attHandler: attHandler, + syncHandler: syncHandler, + firstRun: true, } return h } func (h *CommitteeHandler) Name() string { - return "CLUSTER" + return "COMMITTEE" } func (h *CommitteeHandler) HandleDuties(ctx context.Context) { @@ -56,37 +59,110 @@ func (h *CommitteeHandler) HandleDuties(ctx context.Context) { next = h.ticker.Next() epoch := h.network.Beacon.EstimatedEpochAtSlot(slot) period := h.network.Beacon.EstimatedSyncCommitteePeriodAtEpoch(epoch) - buildStr := fmt.Sprintf("p%v-e%v-s%v-#%v", period, epoch, slot, slot%32+1) + tickerID := fields.FormatSlotTickerCommitteeID(period, epoch, slot) if !h.network.PastAlanForkAtEpoch(epoch) { + if h.firstRun { + h.firstRun = false + } h.logger.Debug("🛠 ticker event", - zap.String("period_epoch_slot_pos", buildStr), + fields.SlotTickerID(tickerID), zap.String("status", "alan not forked yet"), ) continue } + h.logger.Debug("🛠 ticker event", fields.SlotTickerID(tickerID)) - h.logger.Debug("🛠 ticker event", zap.String("period_epoch_slot_pos", buildStr)) + if h.firstRun { + h.processFirstRun(ctx, period, epoch, slot) + } h.processExecution(period, epoch, slot) + if h.indicesChanged { + h.processIndicesChange(period, epoch, slot) + } + h.processFetching(ctx, period, epoch, slot) + h.processSlotTransition(period, epoch, slot) - case <-h.reorg: - // do nothing + case reorgEvent := <-h.reorg: + epoch := h.network.Beacon.EstimatedEpochAtSlot(reorgEvent.Slot) + period := h.network.Beacon.EstimatedSyncCommitteePeriodAtEpoch(epoch) + tickerID := fields.FormatSlotTickerCommitteeID(period, epoch, reorgEvent.Slot) + h.logger.Info("🔀 reorg event received", fields.SlotTickerID(tickerID), zap.Any("event", reorgEvent)) + + h.attHandler.processReorg(ctx, epoch, reorgEvent) + h.syncHandler.processReorg(period, reorgEvent) case <-h.indicesChange: - // do nothing + slot := h.network.Beacon.EstimatedCurrentSlot() + epoch := h.network.Beacon.EstimatedEpochAtSlot(slot) + period := h.network.Beacon.EstimatedSyncCommitteePeriodAtEpoch(epoch) + tickerID := fields.FormatSlotTickerCommitteeID(period, epoch, slot) + h.logger.Info("🔁 indices change received", fields.SlotTickerID(tickerID)) + + h.indicesChanged = true } } } +func (h *CommitteeHandler) processFirstRun(ctx context.Context, period uint64, epoch phase0.Epoch, slot phase0.Slot) { + var wg sync.WaitGroup + wg.Add(2) + + go func() { + defer wg.Done() + h.attHandler.processFirstRun(ctx, epoch, slot) + }() + + go func() { + defer wg.Done() + h.syncHandler.processFirstRun(ctx, period, slot) + }() + + wg.Wait() + h.firstRun = false +} + func (h *CommitteeHandler) processExecution(period uint64, epoch phase0.Epoch, slot phase0.Slot) { - attDuties := h.attDuties.CommitteeSlotDuties(epoch, slot) - syncDuties := h.syncDuties.CommitteePeriodDuties(period) + attDuties := h.attHandler.duties.CommitteeSlotDuties(epoch, slot) + syncDuties := h.syncHandler.duties.CommitteePeriodDuties(period) if attDuties == nil && syncDuties == nil { return } - committeeMap := h.buildCommitteeDuties(attDuties, syncDuties, epoch, slot) - h.dutiesExecutor.ExecuteCommitteeDuties(h.logger, committeeMap) + committeeDuties := h.buildCommitteeDuties(attDuties, syncDuties, epoch, slot) + h.dutiesExecutor.ExecuteCommitteeDuties(h.logger, committeeDuties) + + aggregationDuties := h.buildAggregationDuties(attDuties, syncDuties, slot) + h.dutiesExecutor.ExecuteDuties(h.logger, aggregationDuties) + +} + +func (h *CommitteeHandler) processFetching(ctx context.Context, period uint64, epoch phase0.Epoch, slot phase0.Slot) { + var wg sync.WaitGroup + wg.Add(2) + + go func() { + defer wg.Done() + h.attHandler.processFetching(ctx, epoch, slot) + }() + + go func() { + defer wg.Done() + h.syncHandler.processFetching(ctx, period, slot, true) + }() + + wg.Wait() +} + +func (h *CommitteeHandler) processIndicesChange(period uint64, epoch phase0.Epoch, slot phase0.Slot) { + h.attHandler.processIndicesChange(epoch, slot) + h.syncHandler.processIndicesChange(period, slot) + h.indicesChanged = false +} + +func (h *CommitteeHandler) processSlotTransition(period uint64, epoch phase0.Epoch, slot phase0.Slot) { + h.attHandler.processSlotTransition(epoch, slot) + h.syncHandler.processSlotTransition(period, slot) } func (h *CommitteeHandler) buildCommitteeDuties(attDuties []*eth2apiv1.AttesterDuty, syncDuties []*eth2apiv1.SyncCommitteeDuty, epoch phase0.Epoch, slot phase0.Slot) committeeDutiesMap { @@ -103,15 +179,15 @@ func (h *CommitteeHandler) buildCommitteeDuties(attDuties []*eth2apiv1.AttesterD } for _, d := range attDuties { - if h.shouldExecuteAtt(d) { - specDuty := h.toSpecAttDuty(d, spectypes.BNRoleAttester) + if h.attHandler.shouldExecute(d) { + specDuty := h.attHandler.toSpecDuty(d, spectypes.BNRoleAttester) h.appendBeaconDuty(validatorCommitteeMap, committeeMap, specDuty) } } for _, d := range syncDuties { - if h.shouldExecuteSync(d, slot) { - specDuty := h.toSpecSyncDuty(d, slot, spectypes.BNRoleSyncCommittee) + if h.syncHandler.shouldExecute(d, slot) { + specDuty := h.syncHandler.toSpecDuty(d, slot, spectypes.BNRoleSyncCommittee) h.appendBeaconDuty(validatorCommitteeMap, committeeMap, specDuty) } } @@ -119,6 +195,23 @@ func (h *CommitteeHandler) buildCommitteeDuties(attDuties []*eth2apiv1.AttesterD return committeeMap } +func (h *CommitteeHandler) buildAggregationDuties(attDuties []*eth2apiv1.AttesterDuty, syncDuties []*eth2apiv1.SyncCommitteeDuty, slot phase0.Slot) []*spectypes.ValidatorDuty { + // aggregator and contribution duties + toExecute := make([]*spectypes.ValidatorDuty, 0, len(attDuties)+len(syncDuties)) + for _, d := range attDuties { + if h.attHandler.shouldExecute(d) { + toExecute = append(toExecute, h.attHandler.toSpecDuty(d, spectypes.BNRoleAggregator)) + } + } + for _, d := range syncDuties { + if h.syncHandler.shouldExecute(d, slot) { + toExecute = append(toExecute, h.syncHandler.toSpecDuty(d, slot, spectypes.BNRoleSyncCommitteeContribution)) + } + } + + return toExecute +} + func (h *CommitteeHandler) appendBeaconDuty(vc validatorCommitteeDutyMap, c committeeDutiesMap, beaconDuty *spectypes.ValidatorDuty) { if beaconDuty == nil { h.logger.Error("received nil beaconDuty") @@ -146,57 +239,3 @@ func (h *CommitteeHandler) appendBeaconDuty(vc validatorCommitteeDutyMap, c comm cd.duty.ValidatorDuties = append(c[committee.id].duty.ValidatorDuties, beaconDuty) } - -func (h *CommitteeHandler) toSpecAttDuty(duty *eth2apiv1.AttesterDuty, role spectypes.BeaconRole) *spectypes.ValidatorDuty { - return &spectypes.ValidatorDuty{ - Type: role, - PubKey: duty.PubKey, - Slot: duty.Slot, - ValidatorIndex: duty.ValidatorIndex, - CommitteeIndex: duty.CommitteeIndex, - CommitteeLength: duty.CommitteeLength, - CommitteesAtSlot: duty.CommitteesAtSlot, - ValidatorCommitteeIndex: duty.ValidatorCommitteeIndex, - } -} - -func (h *CommitteeHandler) toSpecSyncDuty(duty *eth2apiv1.SyncCommitteeDuty, slot phase0.Slot, role spectypes.BeaconRole) *spectypes.ValidatorDuty { - indices := make([]uint64, len(duty.ValidatorSyncCommitteeIndices)) - for i, index := range duty.ValidatorSyncCommitteeIndices { - indices[i] = uint64(index) - } - return &spectypes.ValidatorDuty{ - Type: role, - PubKey: duty.PubKey, - Slot: slot, // in order for the duty scheduler to execute - ValidatorIndex: duty.ValidatorIndex, - ValidatorSyncCommitteeIndices: indices, - } -} - -func (h *CommitteeHandler) shouldExecuteAtt(duty *eth2apiv1.AttesterDuty) bool { - currentSlot := h.network.Beacon.EstimatedCurrentSlot() - // execute task if slot already began and not pass 1 epoch - var attestationPropagationSlotRange = phase0.Slot(h.network.Beacon.SlotsPerEpoch()) - if currentSlot >= duty.Slot && currentSlot-duty.Slot <= attestationPropagationSlotRange { - return true - } - if currentSlot+1 == duty.Slot { - h.warnMisalignedSlotAndDuty(duty.String()) - return true - } - return false -} - -func (h *CommitteeHandler) shouldExecuteSync(duty *eth2apiv1.SyncCommitteeDuty, slot phase0.Slot) bool { - currentSlot := h.network.Beacon.EstimatedCurrentSlot() - // execute task if slot already began and not pass 1 slot - if currentSlot == slot { - return true - } - if currentSlot+1 == slot { - h.warnMisalignedSlotAndDuty(duty.String()) - return true - } - return false -} diff --git a/operator/duties/committee_test.go b/operator/duties/committee_test.go index 5c9154c6a7..541a4047a7 100644 --- a/operator/duties/committee_test.go +++ b/operator/duties/committee_test.go @@ -18,12 +18,11 @@ import ( ssvtypes "github.com/ssvlabs/ssv/protocol/v2/types" ) -func setupCommitteeDutiesMock( +func setupDutiesMockCommittee( s *Scheduler, activeShares []*ssvtypes.SSVShare, attDuties *hashmap.Map[phase0.Epoch, []*eth2apiv1.AttesterDuty], syncDuties *hashmap.Map[uint64, []*eth2apiv1.SyncCommitteeDuty], - waitForDuties *SafeValue[bool], ) (chan struct{}, chan committeeDutiesMap) { fetchDutiesCall := make(chan struct{}) executeDutiesCall := make(chan committeeDutiesMap) @@ -57,18 +56,14 @@ func setupCommitteeDutiesMock( s.beaconNode.(*MockBeaconNode).EXPECT().AttesterDuties(gomock.Any(), gomock.Any(), gomock.Any()).DoAndReturn( func(ctx context.Context, epoch phase0.Epoch, indices []phase0.ValidatorIndex) ([]*eth2apiv1.AttesterDuty, error) { - if waitForDuties.Get() { - fetchDutiesCall <- struct{}{} - } + fetchDutiesCall <- struct{}{} duties, _ := attDuties.Get(epoch) return duties, nil }).AnyTimes() s.beaconNode.(*MockBeaconNode).EXPECT().SyncCommitteeDuties(gomock.Any(), gomock.Any(), gomock.Any()).DoAndReturn( func(ctx context.Context, epoch phase0.Epoch, indices []phase0.ValidatorIndex) ([]*eth2apiv1.SyncCommitteeDuty, error) { - if waitForDuties.Get() { - fetchDutiesCall <- struct{}{} - } + fetchDutiesCall <- struct{}{} period := s.network.Beacon.EstimatedSyncCommitteePeriodAtEpoch(epoch) duties, _ := syncDuties.Get(period) return duties, nil @@ -93,10 +88,9 @@ func TestScheduler_Committee_Same_Slot_Attester_Only(t *testing.T) { dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(0) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() activeShares = []*ssvtypes.SSVShare{{ @@ -118,18 +112,20 @@ func TestScheduler_Committee_Same_Slot_Attester_Only(t *testing.T) { currentSlot.Set(phase0.Slot(1)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() // STEP 1: wait for attester duties to be fetched and executed at the same slot duties, _ := attDuties.Get(phase0.Epoch(0)) committeeMap := commHandler.buildCommitteeDuties(duties, nil, 0, currentSlot.Get()) - setExecuteDutyFuncs(scheduler, executeDutiesCall, len(committeeMap)) startTime := time.Now() ticker.Send(currentSlot.Get()) - + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) // validate the 1/3 of the slot waiting time @@ -145,10 +141,9 @@ func TestScheduler_Committee_Same_Slot_SyncCommittee_Only(t *testing.T) { dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(0) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() activeShares = []*ssvtypes.SSVShare{{ @@ -169,18 +164,20 @@ func TestScheduler_Committee_Same_Slot_SyncCommittee_Only(t *testing.T) { currentSlot.Set(phase0.Slot(1)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() // STEP 1: wait for attester duties to be fetched and executed at the same slot duties, _ := syncDuties.Get(0) committeeMap := commHandler.buildCommitteeDuties(nil, duties, 0, currentSlot.Get()) - setExecuteDutyFuncs(scheduler, executeDutiesCall, len(committeeMap)) startTime := time.Now() ticker.Send(currentSlot.Get()) - + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) // validate the 1/3 of the slot waiting time @@ -196,10 +193,9 @@ func TestScheduler_Committee_Same_Slot(t *testing.T) { dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(0) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() activeShares = []*ssvtypes.SSVShare{{ @@ -227,19 +223,21 @@ func TestScheduler_Committee_Same_Slot(t *testing.T) { currentSlot.Set(phase0.Slot(1)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() // STEP 1: wait for attester duties to be fetched and executed at the same slot aDuties, _ := attDuties.Get(0) sDuties, _ := syncDuties.Get(0) committeeMap := commHandler.buildCommitteeDuties(aDuties, sDuties, 0, currentSlot.Get()) - setExecuteDutyFuncs(scheduler, executeDutiesCall, len(committeeMap)) startTime := time.Now() ticker.Send(currentSlot.Get()) - + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) // validate the 1/3 of the slot waiting time @@ -255,10 +253,9 @@ func TestScheduler_Committee_Diff_Slot_Attester_Only(t *testing.T) { dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(0) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() activeShares = []*ssvtypes.SSVShare{{ @@ -278,15 +275,15 @@ func TestScheduler_Committee_Diff_Slot_Attester_Only(t *testing.T) { }, }) - // STEP 1: wait for attester duties to be fetched using handle initial duties scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() - // STEP 2: wait for no action to be taken + // STEP 1: wait for committee duties to be fetched currentSlot.Set(phase0.Slot(1)) ticker.Send(currentSlot.Get()) - waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 3: wait for committee duties to be executed currentSlot.Set(phase0.Slot(2)) @@ -307,15 +304,111 @@ func TestScheduler_Committee_Diff_Slot_Attester_Only(t *testing.T) { require.NoError(t, schedulerPool.Wait()) } +func TestScheduler_Committee_Current_Next_Periods(t *testing.T) { + var ( + dutyStore = dutystore.New() + attHandler = NewAttesterHandler(dutyStore.Attester) + syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) + alanForkEpoch = phase0.Epoch(0) + currentSlot = &SafeValue[phase0.Slot]{} + attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{ + { + Share: spectypes.Share{ + Committee: []*spectypes.ShareMember{ + {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + }, + ValidatorIndex: 1, + }, + }, + { + Share: spectypes.Share{ + Committee: []*spectypes.ShareMember{ + {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + }, + ValidatorIndex: 2, + }, + }, + } + ) + attDuties.Set(phase0.Epoch(254), []*eth2apiv1.AttesterDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 3}, + Slot: phase0.Slot(256*32 - 49), + ValidatorIndex: phase0.ValidatorIndex(1), + }, + }) + attDuties.Set(phase0.Epoch(255), []*eth2apiv1.AttesterDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 4}, + Slot: phase0.Slot(254 * 32), + ValidatorIndex: phase0.ValidatorIndex(2), + }, + }) + syncDuties.Set(0, []*eth2apiv1.SyncCommitteeDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 3}, + ValidatorIndex: phase0.ValidatorIndex(1), + }, + }) + syncDuties.Set(1, []*eth2apiv1.SyncCommitteeDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 4}, + ValidatorIndex: phase0.ValidatorIndex(2), + }, + }) + + currentSlot.Set(phase0.Slot(256*32 - 49)) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) + startFn() + + // STEP 1: wait for committee duties to be fetched and executed + aDuties, _ := attDuties.Get(phase0.Epoch(254)) + sDuties, _ := syncDuties.Get(0) + committeeMap := commHandler.buildCommitteeDuties(aDuties, sDuties, 2, currentSlot.Get()) + setExecuteDutyFuncs(scheduler, executeDutiesCall, len(committeeMap)) + + ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) + + // STEP 2: wait for committee duty to be executed + currentSlot.Set(phase0.Slot(256*32 - 48)) + aDuties, _ = attDuties.Get(0) + sDuties, _ = syncDuties.Get(0) + committeeMap = commHandler.buildCommitteeDuties(aDuties, sDuties, 0, currentSlot.Get()) + setExecuteDutyFuncs(scheduler, executeDutiesCall, len(committeeMap)) + + startTime := time.Now() + ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) + + // Validate execution within 1/3 of the slot time + require.Less(t, scheduler.network.Beacon.SlotDurationSec()/3, time.Since(startTime)) + + // Stop scheduler & wait for graceful exit + cancel() + require.NoError(t, schedulerPool.Wait()) +} + func TestScheduler_Committee_Indices_Changed_Attester_Only(t *testing.T) { var ( dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(0) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() activeShares = []*ssvtypes.SSVShare{ @@ -347,13 +440,15 @@ func TestScheduler_Committee_Indices_Changed_Attester_Only(t *testing.T) { ) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() // STEP 1: wait for no action to be taken ticker.Send(currentSlot.Get()) - // no execution should happen in slot 0 - waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger a change in active indices scheduler.indicesChg <- struct{}{} @@ -378,12 +473,7 @@ func TestScheduler_Committee_Indices_Changed_Attester_Only(t *testing.T) { // STEP 3: wait for attester duties to be fetched currentSlot.Set(phase0.Slot(1)) - waitForDuties.Set(true) ticker.Send(currentSlot.Get()) - // Wait for the slot ticker to be triggered in the attester, sync committee, and cluster handlers. - // This ensures that no attester duties are fetched before the cluster ticker is triggered, - // preventing a scenario where the cluster handler executes duties in the same slot as the attester fetching them. - time.Sleep(10 * time.Millisecond) // wait for attester duties to be fetched waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) @@ -415,10 +505,9 @@ func TestScheduler_Committee_Indices_Changed_Attester_Only_2(t *testing.T) { dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(0) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() activeShares = []*ssvtypes.SSVShare{ @@ -450,13 +539,15 @@ func TestScheduler_Committee_Indices_Changed_Attester_Only_2(t *testing.T) { ) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() // STEP 1: wait for no action to be taken ticker.Send(currentSlot.Get()) - // no execution should happen in slot 0 - waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger a change in active indices scheduler.indicesChg <- struct{}{} @@ -481,12 +572,7 @@ func TestScheduler_Committee_Indices_Changed_Attester_Only_2(t *testing.T) { // STEP 3: wait for attester duties to be fetched currentSlot.Set(phase0.Slot(1)) - waitForDuties.Set(true) ticker.Send(currentSlot.Get()) - // Wait for the slot ticker to be triggered in the attester, sync committee, and cluster handlers. - // This ensures that no attester duties are fetched before the cluster ticker is triggered, - // preventing a scenario where the cluster handler executes duties in the same slot as the attester fetching them. - time.Sleep(10 * time.Millisecond) // wait for attester duties to be fetched waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) @@ -518,10 +604,9 @@ func TestScheduler_Committee_Indices_Changed_Attester_Only_3(t *testing.T) { dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(0) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() activeShares = []*ssvtypes.SSVShare{ @@ -551,13 +636,17 @@ func TestScheduler_Committee_Indices_Changed_Attester_Only_3(t *testing.T) { }, }) - // STEP 1: wait for attester duties to be fetched using handle initial duties + // STEP 1: wait for attester duties to be fetched scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() // STEP 1: wait for no action to be taken ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // no execution should happen in slot 0 waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) @@ -574,12 +663,7 @@ func TestScheduler_Committee_Indices_Changed_Attester_Only_3(t *testing.T) { // STEP 3: wait for attester duties to be fetched currentSlot.Set(phase0.Slot(1)) - waitForDuties.Set(true) ticker.Send(currentSlot.Get()) - // Wait for the slot ticker to be triggered in the attester, sync committee, and cluster handlers. - // This ensures that no attester duties are fetched before the cluster ticker is triggered, - // preventing a scenario where the cluster handler executes duties in the same slot as the attester fetching them. - time.Sleep(10 * time.Millisecond) // wait for attester duties to be fetched waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) @@ -607,15 +691,14 @@ func TestScheduler_Committee_Indices_Changed_Attester_Only_3(t *testing.T) { } // reorg previous dependent root changed -func TestScheduler_Committee_Reorg_Previous_Epoch_Transition_Attester_only(t *testing.T) { +func TestScheduler_Committee_Reorg_Previous_Epoch_Transition_Attester_Only(t *testing.T) { var ( dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(0) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() activeShares = []*ssvtypes.SSVShare{ @@ -632,7 +715,7 @@ func TestScheduler_Committee_Reorg_Previous_Epoch_Transition_Attester_only(t *te currentSlot.Set(phase0.Slot(63)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() attDuties.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ @@ -644,10 +727,13 @@ func TestScheduler_Committee_Reorg_Previous_Epoch_Transition_Attester_only(t *te }) // STEP 1: wait for attester duties to be fetched for next epoch - waitForDuties.Set(true) ticker.Send(currentSlot.Get()) // wait for attester duties to be fetched waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for next epoch attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger head event e := ð2apiv1.Event{ @@ -705,15 +791,14 @@ func TestScheduler_Committee_Reorg_Previous_Epoch_Transition_Attester_only(t *te } // reorg previous dependent root changed and the indices changed as well -func TestScheduler_Committee_Reorg_Previous_Epoch_Transition_Indices_Changed_Attester_only(t *testing.T) { +func TestScheduler_Committee_Reorg_Previous_Epoch_Transition_Indices_Changed_Attester_Only(t *testing.T) { var ( dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(0) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() activeShares = []*ssvtypes.SSVShare{ @@ -738,7 +823,7 @@ func TestScheduler_Committee_Reorg_Previous_Epoch_Transition_Indices_Changed_Att currentSlot.Set(phase0.Slot(63)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() attDuties.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ @@ -750,10 +835,13 @@ func TestScheduler_Committee_Reorg_Previous_Epoch_Transition_Indices_Changed_Att }) // STEP 1: wait for attester duties to be fetched for next epoch - waitForDuties.Set(true) ticker.Send(currentSlot.Get()) // wait for attester duties to be fetched waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for next epoch attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger head event e := ð2apiv1.Event{ @@ -799,10 +887,6 @@ func TestScheduler_Committee_Reorg_Previous_Epoch_Transition_Indices_Changed_Att // STEP 6: wait for attester duties to be fetched again for the current epoch currentSlot.Set(phase0.Slot(65)) ticker.Send(currentSlot.Get()) - // Wait for the slot ticker to be triggered in the attester, sync committee, and cluster handlers. - // This ensures that no attester duties are fetched before the cluster ticker is triggered, - // preventing a scenario where the cluster handler executes duties in the same slot as the attester fetching them. - time.Sleep(10 * time.Millisecond) // wait for attester duties to be fetched waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) @@ -825,15 +909,14 @@ func TestScheduler_Committee_Reorg_Previous_Epoch_Transition_Indices_Changed_Att } // reorg previous dependent root changed -func TestScheduler_Committee_Reorg_Previous_Attester_only(t *testing.T) { +func TestScheduler_Committee_Reorg_Previous_Attester_Only(t *testing.T) { var ( dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(0) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() activeShares = []*ssvtypes.SSVShare{ @@ -847,7 +930,6 @@ func TestScheduler_Committee_Reorg_Previous_Attester_only(t *testing.T) { }, } ) - attDuties.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ { PubKey: phase0.BLSPubKey{1, 2, 3}, @@ -856,13 +938,19 @@ func TestScheduler_Committee_Reorg_Previous_Attester_only(t *testing.T) { }, }) - // STEP 1: wait for attester duties to be fetched using handle initial duties + // STEP 1: wait for attester duties to be fetched currentSlot.Set(phase0.Slot(32)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() - waitForDuties.Set(true) + ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // STEP 2: trigger head event e := ð2apiv1.Event{ Data: ð2apiv1.HeadEvent{ @@ -895,8 +983,6 @@ func TestScheduler_Committee_Reorg_Previous_Attester_only(t *testing.T) { scheduler.HandleHeadEvent(logger)(e) // wait for attester duties to be fetched again for the current epoch waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - // no execution should happen in slot 33 - waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 5: Ticker with no action currentSlot.Set(phase0.Slot(34)) @@ -922,108 +1008,472 @@ func TestScheduler_Committee_Reorg_Previous_Attester_only(t *testing.T) { require.NoError(t, schedulerPool.Wait()) } -func TestScheduler_Committee_Early_Block_Attester_Only(t *testing.T) { +// reorg previous dependent root changed and the indices changed the same slot +func TestScheduler_Committee_Reorg_Previous_Indices_Changed_Attester_Only(t *testing.T) { var ( dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(0) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{{ - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + activeShares = []*ssvtypes.SSVShare{ + { + Share: spectypes.Share{ + Committee: []*spectypes.ShareMember{ + {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + }, + ValidatorIndex: 1, }, - ValidatorIndex: 1, }, - }} + { + Share: spectypes.Share{ + Committee: []*spectypes.ShareMember{ + {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + }, + ValidatorIndex: 2, + }, + }, + } ) - attDuties.Set(phase0.Epoch(0), []*eth2apiv1.AttesterDuty{ + attDuties.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ { PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(2), + Slot: phase0.Slot(35), ValidatorIndex: phase0.ValidatorIndex(1), }, }) - // STEP 1: wait for attester duties to be fetched (handle initial duties) - currentSlot.Set(phase0.Slot(0)) + + // STEP 1: wait for attester duties to be fetched + currentSlot.Set(phase0.Slot(32)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - // STEP 2: wait for no action to be taken - currentSlot.Set(phase0.Slot(1)) - ticker.Send(currentSlot.Get()) + // STEP 2: trigger head event + e := ð2apiv1.Event{ + Data: ð2apiv1.HeadEvent{ + Slot: currentSlot.Get(), + PreviousDutyDependentRoot: phase0.Root{0x01}, + }, + } + scheduler.HandleHeadEvent(logger)(e) waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - // STEP 3: wait for attester duties to be executed faster than 1/3 of the slot duration - startTime := time.Now() - currentSlot.Set(phase0.Slot(2)) + // STEP 3: Ticker with no action + currentSlot.Set(phase0.Slot(33)) ticker.Send(currentSlot.Get()) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - aDuties, _ := attDuties.Get(0) - committeeMap := commHandler.buildCommitteeDuties(aDuties, nil, 0, currentSlot.Get()) - setExecuteDutyFuncs(scheduler, executeDutiesCall, len(committeeMap)) - - // STEP 4: trigger head event (block arrival) - e := ð2apiv1.Event{ + // STEP 4: trigger reorg + e = ð2apiv1.Event{ Data: ð2apiv1.HeadEvent{ - Slot: currentSlot.Get(), + Slot: currentSlot.Get(), + PreviousDutyDependentRoot: phase0.Root{0x02}, }, } + attDuties.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 3}, + Slot: phase0.Slot(36), + ValidatorIndex: phase0.ValidatorIndex(1), + }, + }) scheduler.HandleHeadEvent(logger)(e) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 5: trigger indices change + scheduler.indicesChg <- struct{}{} + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + aDuties, _ := attDuties.Get(phase0.Epoch(1)) + attDuties.Set(phase0.Epoch(1), append(aDuties, ð2apiv1.AttesterDuty{ + PubKey: phase0.BLSPubKey{1, 2, 4}, + Slot: phase0.Slot(36), + ValidatorIndex: phase0.ValidatorIndex(2), + })) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 6: wait for attester duties to be fetched again for the current epoch + currentSlot.Set(phase0.Slot(34)) + ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 7: The first assigned duty should not be executed + currentSlot.Set(phase0.Slot(35)) + ticker.Send(currentSlot.Get()) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 8: The second and new from indices change assigned duties should be executed + currentSlot.Set(phase0.Slot(36)) + aDuties, _ = attDuties.Get(phase0.Epoch(1)) + committeeMap := commHandler.buildCommitteeDuties(aDuties, nil, 0, currentSlot.Get()) + setExecuteDutyFuncs(scheduler, executeDutiesCall, len(committeeMap)) + + ticker.Send(currentSlot.Get()) waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) - require.Less(t, time.Since(startTime), scheduler.network.Beacon.SlotDurationSec()/3) // Stop scheduler & wait for graceful exit. cancel() require.NoError(t, schedulerPool.Wait()) } -func TestScheduler_Committee_Early_Block(t *testing.T) { +// reorg current dependent root changed +func TestScheduler_Committee_Reorg_Current_Attester_Only(t *testing.T) { var ( dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(0) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{{ - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + activeShares = []*ssvtypes.SSVShare{ + { + Share: spectypes.Share{ + Committee: []*spectypes.ShareMember{ + {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + }, + ValidatorIndex: 1, }, - ValidatorIndex: 1, }, - }} + } ) - attDuties.Set(phase0.Epoch(0), []*eth2apiv1.AttesterDuty{ + currentSlot.Set(phase0.Slot(48)) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) + startFn() + + attDuties.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ { PubKey: phase0.BLSPubKey{1, 2, 3}, - Slot: phase0.Slot(1), + Slot: phase0.Slot(64), ValidatorIndex: phase0.ValidatorIndex(1), }, }) - syncDuties.Set(0, []*eth2apiv1.SyncCommitteeDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, + + // STEP 1: wait for attester duties to be fetched for next epoch + ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for next epoch attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 2: trigger head event + e := ð2apiv1.Event{ + Data: ð2apiv1.HeadEvent{ + Slot: currentSlot.Get(), + CurrentDutyDependentRoot: phase0.Root{0x01}, + }, + } + scheduler.HandleHeadEvent(logger)(e) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 3: Ticker with no action + currentSlot.Set(phase0.Slot(49)) + ticker.Send(currentSlot.Get()) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 4: trigger reorg + e = ð2apiv1.Event{ + Data: ð2apiv1.HeadEvent{ + Slot: currentSlot.Get(), + CurrentDutyDependentRoot: phase0.Root{0x02}, + }, + } + attDuties.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 3}, + Slot: phase0.Slot(65), + ValidatorIndex: phase0.ValidatorIndex(1), + }, + }) + scheduler.HandleHeadEvent(logger)(e) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 5: wait for attester duties to be fetched again for the current epoch + currentSlot.Set(phase0.Slot(50)) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 6: skip to the next epoch + currentSlot.Set(phase0.Slot(51)) + for slot := currentSlot.Get(); slot < 64; slot++ { + ticker.Send(slot) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + currentSlot.Set(slot + 1) + } + + // STEP 7: The first assigned duty should not be executed + // slot = 64 + ticker.Send(currentSlot.Get()) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 8: The second assigned duty should be executed + currentSlot.Set(phase0.Slot(65)) + aDuties, _ := attDuties.Get(phase0.Epoch(2)) + committeeMap := commHandler.buildCommitteeDuties(aDuties, nil, 0, currentSlot.Get()) + setExecuteDutyFuncs(scheduler, executeDutiesCall, len(committeeMap)) + + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) + + // Stop scheduler & wait for graceful exit. + cancel() + require.NoError(t, schedulerPool.Wait()) +} + +// reorg current dependent root changed including indices change in the same slot +func TestScheduler_Committee_Reorg_Current_Indices_Changed_Attester_Only(t *testing.T) { + var ( + dutyStore = dutystore.New() + attHandler = NewAttesterHandler(dutyStore.Attester) + syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) + alanForkEpoch = phase0.Epoch(0) + currentSlot = &SafeValue[phase0.Slot]{} + attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{ + { + Share: spectypes.Share{ + Committee: []*spectypes.ShareMember{ + {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + }, + ValidatorIndex: 1, + }, + }, + { + Share: spectypes.Share{ + Committee: []*spectypes.ShareMember{ + {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + }, + ValidatorIndex: 2, + }, + }, + } + ) + currentSlot.Set(phase0.Slot(48)) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) + startFn() + + attDuties.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 3}, + Slot: phase0.Slot(48), + ValidatorIndex: phase0.ValidatorIndex(1), + }, + }) + + // STEP 1: wait for attester duties to be fetched for next epoch + ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for next epoch attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 2: trigger head event + e := ð2apiv1.Event{ + Data: ð2apiv1.HeadEvent{ + Slot: currentSlot.Get(), + CurrentDutyDependentRoot: phase0.Root{0x01}, + }, + } + scheduler.HandleHeadEvent(logger)(e) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 3: Ticker with no action + currentSlot.Set(phase0.Slot(49)) + ticker.Send(currentSlot.Get()) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 4: trigger reorg + e = ð2apiv1.Event{ + Data: ð2apiv1.HeadEvent{ + Slot: currentSlot.Get(), + CurrentDutyDependentRoot: phase0.Root{0x02}, + }, + } + attDuties.Set(phase0.Epoch(2), []*eth2apiv1.AttesterDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 3}, + Slot: phase0.Slot(65), + ValidatorIndex: phase0.ValidatorIndex(1), + }, + }) + scheduler.HandleHeadEvent(logger)(e) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 5: trigger indices change + scheduler.indicesChg <- struct{}{} + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + aDuties, _ := attDuties.Get(phase0.Epoch(2)) + attDuties.Set(phase0.Epoch(2), append(aDuties, ð2apiv1.AttesterDuty{ + PubKey: phase0.BLSPubKey{1, 2, 4}, + Slot: phase0.Slot(65), + ValidatorIndex: phase0.ValidatorIndex(2), + })) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 6: wait for attester duties to be fetched again for the next epoch due to indices change + currentSlot.Set(phase0.Slot(50)) + ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched for current epoch + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for attester duties to be fetched for next epoch + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched for current period + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 7: skip to the next epoch + currentSlot.Set(phase0.Slot(51)) + for slot := currentSlot.Get(); slot < 64; slot++ { + ticker.Send(slot) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + currentSlot.Set(slot + 1) + } + + // STEP 8: The first assigned duty should not be executed + // slot = 64 + ticker.Send(currentSlot.Get()) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 9: The second assigned duty should be executed + currentSlot.Set(phase0.Slot(65)) + aDuties, _ = attDuties.Get(phase0.Epoch(2)) + committeeMap := commHandler.buildCommitteeDuties(aDuties, nil, 0, currentSlot.Get()) + setExecuteDutyFuncs(scheduler, executeDutiesCall, len(committeeMap)) + + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) + + // Stop scheduler & wait for graceful exit. + cancel() + require.NoError(t, schedulerPool.Wait()) +} + +func TestScheduler_Committee_Early_Block_Attester_Only(t *testing.T) { + var ( + dutyStore = dutystore.New() + attHandler = NewAttesterHandler(dutyStore.Attester) + syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) + alanForkEpoch = phase0.Epoch(0) + currentSlot = &SafeValue[phase0.Slot]{} + attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{{ + Share: spectypes.Share{ + Committee: []*spectypes.ShareMember{ + {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + }, + ValidatorIndex: 1, + }, + }} + ) + attDuties.Set(phase0.Epoch(0), []*eth2apiv1.AttesterDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 3}, + Slot: phase0.Slot(2), ValidatorIndex: phase0.ValidatorIndex(1), }, }) - // STEP 1: wait for attester & sync committee duties to be fetched (handle initial duties) + // STEP 1: wait for attester duties to be fetched + currentSlot.Set(phase0.Slot(0)) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) + startFn() + + ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 2: wait for no action to be taken + currentSlot.Set(phase0.Slot(1)) + ticker.Send(currentSlot.Get()) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 3: wait for attester duties to be executed faster than 1/3 of the slot duration + startTime := time.Now() + currentSlot.Set(phase0.Slot(2)) + ticker.Send(currentSlot.Get()) + + aDuties, _ := attDuties.Get(0) + committeeMap := commHandler.buildCommitteeDuties(aDuties, nil, 0, currentSlot.Get()) + setExecuteDutyFuncs(scheduler, executeDutiesCall, len(committeeMap)) + + // STEP 4: trigger head event (block arrival) + e := ð2apiv1.Event{ + Data: ð2apiv1.HeadEvent{ + Slot: currentSlot.Get(), + }, + } + scheduler.HandleHeadEvent(logger)(e) + waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) + require.Less(t, time.Since(startTime), scheduler.network.Beacon.SlotDurationSec()/3) + + // Stop scheduler & wait for graceful exit. + cancel() + require.NoError(t, schedulerPool.Wait()) +} + +func TestScheduler_Committee_Early_Block(t *testing.T) { + var ( + dutyStore = dutystore.New() + attHandler = NewAttesterHandler(dutyStore.Attester) + syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) + alanForkEpoch = phase0.Epoch(0) + currentSlot = &SafeValue[phase0.Slot]{} + attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{{ + Share: spectypes.Share{ + Committee: []*spectypes.ShareMember{ + {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + }, + ValidatorIndex: 1, + }, + }} + ) + attDuties.Set(phase0.Epoch(0), []*eth2apiv1.AttesterDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 3}, + Slot: phase0.Slot(1), + ValidatorIndex: phase0.ValidatorIndex(1), + }, + }) + syncDuties.Set(0, []*eth2apiv1.SyncCommitteeDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 3}, + ValidatorIndex: phase0.ValidatorIndex(1), + }, + }) + // STEP 1: wait for attester & sync committee duties to be fetched currentSlot.Set(phase0.Slot(1)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() // STEP 2: wait for committee duty to be executed @@ -1034,6 +1484,10 @@ func TestScheduler_Committee_Early_Block(t *testing.T) { startTime := time.Now() ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) // validate the 1/3 of the slot waiting time @@ -1062,15 +1516,134 @@ func TestScheduler_Committee_Early_Block(t *testing.T) { require.NoError(t, schedulerPool.Wait()) } -func TestScheduler_Committee_On_Fork_Attester_only(t *testing.T) { +func TestScheduler_Committee_Start_In_The_End_Of_The_Epoch_Attester_Only(t *testing.T) { + var ( + dutyStore = dutystore.New() + attHandler = NewAttesterHandler(dutyStore.Attester) + syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) + alanForkEpoch = phase0.Epoch(0) + currentSlot = &SafeValue[phase0.Slot]{} + attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{{ + Share: spectypes.Share{ + Committee: []*spectypes.ShareMember{ + {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + }, + ValidatorIndex: 1, + }, + }} + ) + currentSlot.Set(phase0.Slot(31)) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) + startFn() + + attDuties.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 3}, + Slot: phase0.Slot(32), + ValidatorIndex: phase0.ValidatorIndex(1), + }, + }) + + // STEP 1: wait for attester duties to be fetched for the next epoch + ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for next epoch attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 2: wait for attester duties to be executed + currentSlot.Set(phase0.Slot(32)) + aDuties, _ := attDuties.Get(1) + sDuties, _ := syncDuties.Get(1) + committeeMap := commHandler.buildCommitteeDuties(aDuties, sDuties, 0, currentSlot.Get()) + setExecuteDutyFuncs(scheduler, executeDutiesCall, len(committeeMap)) + + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) + + // Stop scheduler & wait for graceful exit. + cancel() + require.NoError(t, schedulerPool.Wait()) +} + +func TestScheduler_Committee_Fetch_Execute_Next_Epoch_Duty(t *testing.T) { var ( dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) + alanForkEpoch = phase0.Epoch(0) + currentSlot = &SafeValue[phase0.Slot]{} + attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() + syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{{ + Share: spectypes.Share{ + Committee: []*spectypes.ShareMember{ + {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, + }, + ValidatorIndex: 1, + }, + }} + ) + attDuties.Set(phase0.Epoch(1), []*eth2apiv1.AttesterDuty{ + { + PubKey: phase0.BLSPubKey{1, 2, 3}, + Slot: phase0.Slot(32), + ValidatorIndex: phase0.ValidatorIndex(1), + }, + }) + + currentSlot.Set(phase0.Slot(14)) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) + startFn() + + // STEP 1: wait for duties to be fetched + ticker.Send(currentSlot.Get()) + // wait for attester duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + // wait for sync committee duties to be fetched + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 2: wait for no action to be taken + currentSlot.Set(phase0.Slot(15)) + ticker.Send(currentSlot.Get()) + waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 3: wait for duties to be fetched for the next epoch + currentSlot.Set(phase0.Slot(16)) + ticker.Send(currentSlot.Get()) + waitForDutiesFetchCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + + // STEP 4: wait for attester duties to be executed + currentSlot.Set(phase0.Slot(32)) + aDuties, _ := attDuties.Get(1) + sDuties, _ := syncDuties.Get(1) + committeeMap := commHandler.buildCommitteeDuties(aDuties, sDuties, 0, currentSlot.Get()) + setExecuteDutyFuncs(scheduler, executeDutiesCall, len(committeeMap)) + + ticker.Send(currentSlot.Get()) + waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) + + // Stop scheduler & wait for graceful exit. + cancel() + require.NoError(t, schedulerPool.Wait()) +} + +func TestScheduler_Committee_On_Fork_Attester_Only(t *testing.T) { + var ( + dutyStore = dutystore.New() + attHandler = NewAttesterHandler(dutyStore.Attester) + syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(2) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() activeShares = []*ssvtypes.SSVShare{{ @@ -1082,6 +1655,13 @@ func TestScheduler_Committee_On_Fork_Attester_only(t *testing.T) { }, }} ) + currentSlot.Set(phase0.Slot(1)) + scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) + fetchDutiesCallAttester, executeDutiesCallAttester := setupDutiesMockAttesterGenesis(scheduler, attDuties) + fetchDutiesCallSyncCommittee, executeDutiesCallSyncCommittee := setupGenesisDutiesMockSyncCommittee(scheduler, activeShares, syncDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) + startFn() + attDuties.Set(phase0.Epoch(0), []*eth2apiv1.AttesterDuty{ { PubKey: phase0.BLSPubKey{1, 2, 3}, @@ -1097,20 +1677,15 @@ func TestScheduler_Committee_On_Fork_Attester_only(t *testing.T) { }, }) - currentSlot.Set(phase0.Slot(1)) - scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchAttesterDutiesCall, executeAttesterDutiesCall := setupAttesterGenesisDutiesMock(scheduler, attDuties, waitForDuties) - _, _ = setupSyncCommitteeDutiesMock(scheduler, activeShares, syncDuties, waitForDuties) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) - startFn() - aDuties, _ := attDuties.Get(0) - aExpected := expectedExecutedGenesisAttesterDuties(attHandler, aDuties) - setExecuteGenesisDutyFunc(scheduler, executeAttesterDutiesCall, len(aExpected)) + aExpected := expectedExecutedDutiesAttesterGenesis(attHandler, aDuties) + setExecuteGenesisDutyFunc(scheduler, executeDutiesCallAttester, len(aExpected)) startTime := time.Now() ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchAttesterDutiesCall, executeAttesterDutiesCall, timeout, aExpected) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCallAttester, executeDutiesCallAttester, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCallSyncCommittee, executeDutiesCallSyncCommittee, timeout) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCallAttester, executeDutiesCallAttester, timeout, aExpected) waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // validate the 1/3 of the slot waiting time @@ -1120,16 +1695,15 @@ func TestScheduler_Committee_On_Fork_Attester_only(t *testing.T) { currentSlot.Set(phase0.Slot(2)) for slot := currentSlot.Get(); slot < 48; slot++ { ticker.Send(slot) - waitForNoActionGenesis(t, logger, fetchAttesterDutiesCall, executeAttesterDutiesCall, timeout) + waitForNoActionGenesis(t, logger, fetchDutiesCallAttester, executeDutiesCallAttester, timeout) waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) currentSlot.Set(slot + 1) } // wait for duties to be fetched for the next fork epoch currentSlot.Set(phase0.Slot(48)) - waitForDuties.Set(true) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchAttesterDutiesCall, executeAttesterDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCallAttester, executeDutiesCallAttester, timeout) currentSlot.Set(phase0.Slot(64)) aDuties, _ = attDuties.Get(2) @@ -1138,8 +1712,7 @@ func TestScheduler_Committee_On_Fork_Attester_only(t *testing.T) { startTime = time.Now() ticker.Send(currentSlot.Get()) - waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) - waitForNoActionGenesis(t, logger, fetchAttesterDutiesCall, executeAttesterDutiesCall, timeout) + waitForNoActionGenesis(t, logger, fetchDutiesCallAttester, executeDutiesCallAttester, timeout) // validate the 1/3 of the slot waiting time require.Less(t, scheduler.network.Beacon.SlotDurationSec()/3, time.Since(startTime)) @@ -1154,10 +1727,9 @@ func TestScheduler_Committee_On_Fork(t *testing.T) { dutyStore = dutystore.New() attHandler = NewAttesterHandler(dutyStore.Attester) syncHandler = NewSyncCommitteeHandler(dutyStore.SyncCommittee) - commHandler = NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee) + commHandler = NewCommitteeHandler(attHandler, syncHandler) alanForkEpoch = phase0.Epoch(256) currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} attDuties = hashmap.New[phase0.Epoch, []*eth2apiv1.AttesterDuty]() syncDuties = hashmap.New[uint64, []*eth2apiv1.SyncCommitteeDuty]() activeShares = []*ssvtypes.SSVShare{{ @@ -1194,28 +1766,36 @@ func TestScheduler_Committee_On_Fork(t *testing.T) { currentSlot.Set(phase0.Slot(lastPeriodEpoch * 32)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{attHandler, syncHandler, commHandler}, currentSlot, alanForkEpoch) - fetchAttesterDutiesCall, executeAttesterDutiesCall := setupAttesterGenesisDutiesMock(scheduler, attDuties, waitForDuties) - _, _ = setupSyncCommitteeDutiesMock(scheduler, activeShares, syncDuties, waitForDuties) - fetchDutiesCall, executeDutiesCall := setupCommitteeDutiesMock(scheduler, activeShares, attDuties, syncDuties, waitForDuties) + fetchDutiesCallAttester, executeDutiesCallAttester := setupDutiesMockAttesterGenesis(scheduler, attDuties) + fetchDutiesCallSyncCommittee, executeDutiesCallSyncCommittee := setupGenesisDutiesMockSyncCommittee(scheduler, activeShares, syncDuties) + fetchDutiesCall, executeDutiesCall := setupDutiesMockCommittee(scheduler, activeShares, attDuties, syncDuties) startFn() aDuties, _ := attDuties.Get(lastPeriodEpoch) - aExpected := expectedExecutedGenesisAttesterDuties(attHandler, aDuties) - setExecuteGenesisDutyFunc(scheduler, executeAttesterDutiesCall, len(aExpected)) + aExpected := expectedExecutedDutiesAttesterGenesis(attHandler, aDuties) + setExecuteGenesisDutyFunc(scheduler, executeDutiesCallAttester, len(aExpected)) ticker.Send(currentSlot.Get()) - waitForNoActionGenesis(t, logger, fetchAttesterDutiesCall, executeAttesterDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCallAttester, executeDutiesCallAttester, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCallSyncCommittee, executeDutiesCallSyncCommittee, timeout) + // next period + waitForDutiesFetchGenesis(t, logger, fetchDutiesCallSyncCommittee, executeDutiesCallSyncCommittee, timeout) waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) currentSlot.Set(phase0.Slot(lastPeriodEpoch*32 + 1)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchAttesterDutiesCall, executeAttesterDutiesCall, timeout, aExpected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCallAttester, executeDutiesCallAttester, timeout, aExpected) waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) currentSlot.Set(phase0.Slot(lastPeriodEpoch*32 + 2)) for slot := currentSlot.Get(); slot < 256*32; slot++ { ticker.Send(slot) - waitForNoActionGenesis(t, logger, fetchAttesterDutiesCall, executeAttesterDutiesCall, timeout) + if uint64(slot)%32 == scheduler.network.SlotsPerEpoch()/2 { + waitForDutiesFetchGenesis(t, logger, fetchDutiesCallAttester, executeDutiesCallAttester, timeout) + } else { + waitForNoActionGenesis(t, logger, fetchDutiesCallAttester, executeDutiesCallAttester, timeout) + waitForNoActionGenesis(t, logger, fetchDutiesCallSyncCommittee, executeDutiesCallSyncCommittee, timeout) + } waitForNoActionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout) currentSlot.Set(slot + 1) } @@ -1228,7 +1808,7 @@ func TestScheduler_Committee_On_Fork(t *testing.T) { ticker.Send(currentSlot.Get()) waitForDutiesExecutionCommittee(t, logger, fetchDutiesCall, executeDutiesCall, timeout, committeeMap) - waitForNoActionGenesis(t, logger, fetchAttesterDutiesCall, executeAttesterDutiesCall, timeout) + waitForNoActionGenesis(t, logger, fetchDutiesCallAttester, executeDutiesCallAttester, timeout) // Stop scheduler & wait for graceful exit. cancel() diff --git a/operator/duties/dutystore/duties.go b/operator/duties/dutystore/duties.go index 175d230ca0..9a73e46954 100644 --- a/operator/duties/dutystore/duties.go +++ b/operator/duties/dutystore/duties.go @@ -90,7 +90,7 @@ func (d *Duties[D]) Set(epoch phase0.Epoch, duties []StoreDuty[D]) { d.m[epoch] = mapped } -func (d *Duties[D]) ResetEpoch(epoch phase0.Epoch) { +func (d *Duties[D]) Reset(epoch phase0.Epoch) { d.mu.Lock() defer d.mu.Unlock() diff --git a/operator/duties/proposer.go b/operator/duties/proposer.go index dfeb2cfea4..05a392aad9 100644 --- a/operator/duties/proposer.go +++ b/operator/duties/proposer.go @@ -34,24 +34,7 @@ func (h *ProposerHandler) Name() string { return spectypes.BNRoleProposer.String() } -// HandleDuties manages the duty lifecycle, handling different cases: -// -// On First Run: -// 1. Fetch duties for the current epoch. -// 2. Execute duties. -// -// On Re-org (current dependent root changed): -// 1. Fetch duties for the current epoch. -// 2. Execute duties. -// -// On Indices Change: -// 1. Execute duties. -// 2. ResetEpoch duties for the current epoch. -// 3. Fetch duties for the current epoch. -// -// On Ticker event: -// 1. Execute duties. -// 2. If necessary, fetch duties for the current epoch. +// HandleDuties manages the duty lifecycle func (h *ProposerHandler) HandleDuties(ctx context.Context) { h.logger.Info("starting duty handler") defer h.logger.Info("duty handler exited") @@ -65,47 +48,47 @@ func (h *ProposerHandler) HandleDuties(ctx context.Context) { case <-next: slot := h.ticker.Slot() next = h.ticker.Next() - currentEpoch := h.network.Beacon.EstimatedEpochAtSlot(slot) - buildStr := fmt.Sprintf("e%v-s%v-#%v", currentEpoch, slot, slot%32+1) - h.logger.Debug("🛠 ticker event", zap.String("epoch_slot_pos", buildStr)) + epoch := h.network.Beacon.EstimatedEpochAtSlot(slot) + tickerID := fields.FormatSlotTickerID(epoch, slot) + h.logger.Debug("🛠 ticker event", fields.SlotTickerID(tickerID)) ctx, cancel := context.WithDeadline(ctx, h.network.Beacon.GetSlotStartTime(slot+1).Add(100*time.Millisecond)) if h.fetchFirst { h.fetchFirst = false h.indicesChanged = false - h.processFetching(ctx, currentEpoch) - h.processExecution(currentEpoch, slot) + h.processFetching(ctx, epoch) + h.processExecution(epoch, slot) } else { - h.processExecution(currentEpoch, slot) + h.processExecution(epoch, slot) if h.indicesChanged { h.indicesChanged = false - h.processFetching(ctx, currentEpoch) + h.processFetching(ctx, epoch) } } cancel() // last slot of epoch if uint64(slot)%h.network.Beacon.SlotsPerEpoch() == h.network.Beacon.SlotsPerEpoch()-1 { - h.duties.ResetEpoch(currentEpoch - 1) + h.duties.Reset(epoch - 1) h.fetchFirst = true } case reorgEvent := <-h.reorg: - currentEpoch := h.network.Beacon.EstimatedEpochAtSlot(reorgEvent.Slot) - buildStr := fmt.Sprintf("e%v-s%v-#%v", currentEpoch, reorgEvent.Slot, reorgEvent.Slot%32+1) - h.logger.Info("🔀 reorg event received", zap.String("epoch_slot_pos", buildStr), zap.Any("event", reorgEvent)) + epoch := h.network.Beacon.EstimatedEpochAtSlot(reorgEvent.Slot) + tickerID := fields.FormatSlotTickerID(epoch, reorgEvent.Slot) + h.logger.Info("🔀 reorg event received", fields.SlotTickerID(tickerID), zap.Any("event", reorgEvent)) // reset current epoch duties if reorgEvent.Current { - h.duties.ResetEpoch(currentEpoch) + h.duties.Reset(epoch) h.fetchFirst = true } case <-h.indicesChange: slot := h.network.Beacon.EstimatedCurrentSlot() - currentEpoch := h.network.Beacon.EstimatedEpochAtSlot(slot) - buildStr := fmt.Sprintf("e%v-s%v-#%v", currentEpoch, slot, slot%32+1) - h.logger.Info("🔁 indices change received", zap.String("epoch_slot_pos", buildStr)) + epoch := h.network.Beacon.EstimatedEpochAtSlot(slot) + tickerID := fields.FormatSlotTickerID(epoch, slot) + h.logger.Info("🔁 indices change received", fields.SlotTickerID(tickerID)) h.indicesChanged = true } @@ -178,7 +161,7 @@ func (h *ProposerHandler) fetchAndProcessDuties(ctx context.Context, epoch phase return fmt.Errorf("failed to fetch proposer duties: %w", err) } - h.duties.ResetEpoch(epoch) + h.duties.Reset(epoch) specDuties := make([]*spectypes.ValidatorDuty, 0, len(duties)) storeDuties := make([]dutystore.StoreDuty[eth2apiv1.ProposerDuty], 0, len(duties)) diff --git a/operator/duties/proposer_genesis_test.go b/operator/duties/proposer_genesis_test.go index c9125fafec..9a2e00c11d 100644 --- a/operator/duties/proposer_genesis_test.go +++ b/operator/duties/proposer_genesis_test.go @@ -19,7 +19,7 @@ import ( "github.com/ssvlabs/ssv/protocol/v2/types" ) -func setupProposerGenesisDutiesMock(s *Scheduler, dutiesMap *hashmap.Map[phase0.Epoch, []*eth2apiv1.ProposerDuty]) (chan struct{}, chan []*genesisspectypes.Duty) { +func setupDutiesMockProposerGenesis(s *Scheduler, dutiesMap *hashmap.Map[phase0.Epoch, []*eth2apiv1.ProposerDuty]) (chan struct{}, chan []*genesisspectypes.Duty) { fetchDutiesCall := make(chan struct{}) executeDutiesCall := make(chan []*genesisspectypes.Duty) @@ -57,7 +57,7 @@ func setupProposerGenesisDutiesMock(s *Scheduler, dutiesMap *hashmap.Map[phase0. return fetchDutiesCall, executeDutiesCall } -func expectedExecutedGenesisProposerDuties(handler *ProposerHandler, duties []*eth2apiv1.ProposerDuty) []*genesisspectypes.Duty { +func expectedExecutedDutiesProposerGenesis(handler *ProposerHandler, duties []*eth2apiv1.ProposerDuty) []*genesisspectypes.Duty { expectedDuties := make([]*genesisspectypes.Duty, 0) for _, d := range duties { expectedDuties = append(expectedDuties, handler.toGenesisSpecDuty(d, genesisspectypes.BNRoleProposer)) @@ -73,7 +73,7 @@ func TestScheduler_Proposer_Genesis_Same_Slot(t *testing.T) { ) currentSlot.Set(phase0.Slot(0)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, goclient.FarFutureEpoch) - fetchDutiesCall, executeDutiesCall := setupProposerGenesisDutiesMock(scheduler, dutiesMap) + fetchDutiesCall, executeDutiesCall := setupDutiesMockProposerGenesis(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.ProposerDuty{ @@ -86,12 +86,12 @@ func TestScheduler_Proposer_Genesis_Same_Slot(t *testing.T) { // STEP 1: wait for proposer duties to be fetched and executed at the same slot duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedGenesisProposerDuties(handler, duties) + expected := expectedExecutedDutiesProposerGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -106,7 +106,7 @@ func TestScheduler_Proposer_Genesis_Diff_Slots(t *testing.T) { ) currentSlot.Set(phase0.Slot(0)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, goclient.FarFutureEpoch) - fetchDutiesCall, executeDutiesCall := setupProposerGenesisDutiesMock(scheduler, dutiesMap) + fetchDutiesCall, executeDutiesCall := setupDutiesMockProposerGenesis(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.ProposerDuty{ @@ -119,7 +119,7 @@ func TestScheduler_Proposer_Genesis_Diff_Slots(t *testing.T) { // STEP 1: wait for proposer duties to be fetched ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: wait for no action to be taken currentSlot.Set(phase0.Slot(1)) @@ -129,11 +129,11 @@ func TestScheduler_Proposer_Genesis_Diff_Slots(t *testing.T) { // STEP 3: wait for proposer duties to be executed currentSlot.Set(phase0.Slot(2)) duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedGenesisProposerDuties(handler, duties) + expected := expectedExecutedDutiesProposerGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -149,7 +149,7 @@ func TestScheduler_Proposer_Genesis_Indices_Changed(t *testing.T) { ) currentSlot.Set(phase0.Slot(0)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, goclient.FarFutureEpoch) - fetchDutiesCall, executeDutiesCall := setupProposerGenesisDutiesMock(scheduler, dutiesMap) + fetchDutiesCall, executeDutiesCall := setupDutiesMockProposerGenesis(scheduler, dutiesMap) startFn() // STEP 1: wait for no action to be taken @@ -186,18 +186,18 @@ func TestScheduler_Proposer_Genesis_Indices_Changed(t *testing.T) { // STEP 4: wait for proposer duties to be fetched again currentSlot.Set(phase0.Slot(2)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // no execution should happen in slot 2 waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 4: wait for proposer duties to be executed currentSlot.Set(phase0.Slot(3)) duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedGenesisProposerDuties(handler, []*eth2apiv1.ProposerDuty{duties[2]}) + expected := expectedExecutedDutiesProposerGenesis(handler, []*eth2apiv1.ProposerDuty{duties[2]}) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -212,7 +212,7 @@ func TestScheduler_Proposer_Genesis_Multiple_Indices_Changed_Same_Slot(t *testin ) currentSlot.Set(phase0.Slot(0)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, goclient.FarFutureEpoch) - fetchDutiesCall, executeDutiesCall := setupProposerGenesisDutiesMock(scheduler, dutiesMap) + fetchDutiesCall, executeDutiesCall := setupDutiesMockProposerGenesis(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.ProposerDuty{ @@ -225,7 +225,7 @@ func TestScheduler_Proposer_Genesis_Multiple_Indices_Changed_Same_Slot(t *testin // STEP 1: wait for proposer duties to be fetched ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger a change in active indices scheduler.indicesChg <- struct{}{} @@ -250,34 +250,34 @@ func TestScheduler_Proposer_Genesis_Multiple_Indices_Changed_Same_Slot(t *testin // STEP 4: wait for proposer duties to be fetched again currentSlot.Set(phase0.Slot(1)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 5: wait for proposer duties to be executed currentSlot.Set(phase0.Slot(2)) duties, _ = dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedGenesisProposerDuties(handler, []*eth2apiv1.ProposerDuty{duties[0]}) + expected := expectedExecutedDutiesProposerGenesis(handler, []*eth2apiv1.ProposerDuty{duties[0]}) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // STEP 6: wait for proposer duties to be executed currentSlot.Set(phase0.Slot(3)) duties, _ = dutiesMap.Get(phase0.Epoch(0)) - expected = expectedExecutedGenesisProposerDuties(handler, []*eth2apiv1.ProposerDuty{duties[1]}) + expected = expectedExecutedDutiesProposerGenesis(handler, []*eth2apiv1.ProposerDuty{duties[1]}) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // STEP 7: wait for proposer duties to be executed currentSlot.Set(phase0.Slot(4)) duties, _ = dutiesMap.Get(phase0.Epoch(0)) - expected = expectedExecutedGenesisProposerDuties(handler, []*eth2apiv1.ProposerDuty{duties[2]}) + expected = expectedExecutedDutiesProposerGenesis(handler, []*eth2apiv1.ProposerDuty{duties[2]}) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -293,7 +293,7 @@ func TestScheduler_Proposer_Genesis_Reorg_Current(t *testing.T) { ) currentSlot.Set(phase0.Slot(34)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, goclient.FarFutureEpoch) - fetchDutiesCall, executeDutiesCall := setupProposerGenesisDutiesMock(scheduler, dutiesMap) + fetchDutiesCall, executeDutiesCall := setupDutiesMockProposerGenesis(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.ProposerDuty{ @@ -306,7 +306,7 @@ func TestScheduler_Proposer_Genesis_Reorg_Current(t *testing.T) { // STEP 1: wait for proposer duties to be fetched ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger head event e := ð2apiv1.Event{ @@ -344,16 +344,16 @@ func TestScheduler_Proposer_Genesis_Reorg_Current(t *testing.T) { // The first assigned duty should not be executed currentSlot.Set(phase0.Slot(36)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 7: The second assigned duty should be executed currentSlot.Set(phase0.Slot(37)) duties, _ := dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedGenesisProposerDuties(handler, duties) + expected := expectedExecutedDutiesProposerGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -369,7 +369,7 @@ func TestScheduler_Proposer_Genesis_Reorg_Current_Indices_Changed(t *testing.T) ) currentSlot.Set(phase0.Slot(34)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, goclient.FarFutureEpoch) - fetchDutiesCall, executeDutiesCall := setupProposerGenesisDutiesMock(scheduler, dutiesMap) + fetchDutiesCall, executeDutiesCall := setupDutiesMockProposerGenesis(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.ProposerDuty{ @@ -382,7 +382,7 @@ func TestScheduler_Proposer_Genesis_Reorg_Current_Indices_Changed(t *testing.T) // STEP 1: wait for proposer duties to be fetched ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger head event e := ð2apiv1.Event{ @@ -430,25 +430,25 @@ func TestScheduler_Proposer_Genesis_Reorg_Current_Indices_Changed(t *testing.T) // The first assigned duty should not be executed currentSlot.Set(phase0.Slot(36)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 7: The second assigned duty should be executed currentSlot.Set(phase0.Slot(37)) duties, _ = dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedGenesisProposerDuties(handler, []*eth2apiv1.ProposerDuty{duties[0]}) + expected := expectedExecutedDutiesProposerGenesis(handler, []*eth2apiv1.ProposerDuty{duties[0]}) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // STEP 8: The second assigned duty should be executed currentSlot.Set(phase0.Slot(38)) duties, _ = dutiesMap.Get(phase0.Epoch(1)) - expected = expectedExecutedGenesisProposerDuties(handler, []*eth2apiv1.ProposerDuty{duties[1]}) + expected = expectedExecutedDutiesProposerGenesis(handler, []*eth2apiv1.ProposerDuty{duties[1]}) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() diff --git a/operator/duties/proposer_test.go b/operator/duties/proposer_test.go index 8a13a3ce77..c3760752ae 100644 --- a/operator/duties/proposer_test.go +++ b/operator/duties/proposer_test.go @@ -17,7 +17,7 @@ import ( "github.com/ssvlabs/ssv/protocol/v2/types" ) -func setupProposerDutiesMock(s *Scheduler, dutiesMap *hashmap.Map[phase0.Epoch, []*eth2apiv1.ProposerDuty]) (chan struct{}, chan []*spectypes.ValidatorDuty) { +func setupDutiesMockProposer(s *Scheduler, dutiesMap *hashmap.Map[phase0.Epoch, []*eth2apiv1.ProposerDuty]) (chan struct{}, chan []*spectypes.ValidatorDuty) { fetchDutiesCall := make(chan struct{}) executeDutiesCall := make(chan []*spectypes.ValidatorDuty) @@ -55,7 +55,7 @@ func setupProposerDutiesMock(s *Scheduler, dutiesMap *hashmap.Map[phase0.Epoch, return fetchDutiesCall, executeDutiesCall } -func expectedExecutedProposerDuties(handler *ProposerHandler, duties []*eth2apiv1.ProposerDuty) []*spectypes.ValidatorDuty { +func expectedExecutedDutiesProposer(handler *ProposerHandler, duties []*eth2apiv1.ProposerDuty) []*spectypes.ValidatorDuty { expectedDuties := make([]*spectypes.ValidatorDuty, 0) for _, d := range duties { expectedDuties = append(expectedDuties, handler.toSpecDuty(d, spectypes.BNRoleProposer)) @@ -71,7 +71,7 @@ func TestScheduler_Proposer_Same_Slot(t *testing.T) { ) currentSlot.Set(phase0.Slot(0)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, 0) - fetchDutiesCall, executeDutiesCall := setupProposerDutiesMock(scheduler, dutiesMap) + fetchDutiesCall, executeDutiesCall := setupDutiesMockProposer(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.ProposerDuty{ @@ -84,7 +84,7 @@ func TestScheduler_Proposer_Same_Slot(t *testing.T) { // STEP 1: wait for proposer duties to be fetched and executed at the same slot duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedProposerDuties(handler, duties) + expected := expectedExecutedDutiesProposer(handler, duties) setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) @@ -104,7 +104,7 @@ func TestScheduler_Proposer_Diff_Slots(t *testing.T) { ) currentSlot.Set(phase0.Slot(0)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, 0) - fetchDutiesCall, executeDutiesCall := setupProposerDutiesMock(scheduler, dutiesMap) + fetchDutiesCall, executeDutiesCall := setupDutiesMockProposer(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.ProposerDuty{ @@ -127,7 +127,7 @@ func TestScheduler_Proposer_Diff_Slots(t *testing.T) { // STEP 3: wait for proposer duties to be executed currentSlot.Set(phase0.Slot(2)) duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedProposerDuties(handler, duties) + expected := expectedExecutedDutiesProposer(handler, duties) setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) @@ -147,7 +147,7 @@ func TestScheduler_Proposer_Indices_Changed(t *testing.T) { ) currentSlot.Set(phase0.Slot(0)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, 0) - fetchDutiesCall, executeDutiesCall := setupProposerDutiesMock(scheduler, dutiesMap) + fetchDutiesCall, executeDutiesCall := setupDutiesMockProposer(scheduler, dutiesMap) startFn() // STEP 1: wait for no action to be taken @@ -191,7 +191,7 @@ func TestScheduler_Proposer_Indices_Changed(t *testing.T) { // STEP 4: wait for proposer duties to be executed currentSlot.Set(phase0.Slot(3)) duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedProposerDuties(handler, []*eth2apiv1.ProposerDuty{duties[2]}) + expected := expectedExecutedDutiesProposer(handler, []*eth2apiv1.ProposerDuty{duties[2]}) setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) @@ -210,7 +210,7 @@ func TestScheduler_Proposer_Multiple_Indices_Changed_Same_Slot(t *testing.T) { ) currentSlot.Set(phase0.Slot(0)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, 0) - fetchDutiesCall, executeDutiesCall := setupProposerDutiesMock(scheduler, dutiesMap) + fetchDutiesCall, executeDutiesCall := setupDutiesMockProposer(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.ProposerDuty{ @@ -253,7 +253,7 @@ func TestScheduler_Proposer_Multiple_Indices_Changed_Same_Slot(t *testing.T) { // STEP 5: wait for proposer duties to be executed currentSlot.Set(phase0.Slot(2)) duties, _ = dutiesMap.Get(phase0.Epoch(0)) - expected := expectedExecutedProposerDuties(handler, []*eth2apiv1.ProposerDuty{duties[0]}) + expected := expectedExecutedDutiesProposer(handler, []*eth2apiv1.ProposerDuty{duties[0]}) setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) @@ -262,7 +262,7 @@ func TestScheduler_Proposer_Multiple_Indices_Changed_Same_Slot(t *testing.T) { // STEP 6: wait for proposer duties to be executed currentSlot.Set(phase0.Slot(3)) duties, _ = dutiesMap.Get(phase0.Epoch(0)) - expected = expectedExecutedProposerDuties(handler, []*eth2apiv1.ProposerDuty{duties[1]}) + expected = expectedExecutedDutiesProposer(handler, []*eth2apiv1.ProposerDuty{duties[1]}) setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) @@ -271,7 +271,7 @@ func TestScheduler_Proposer_Multiple_Indices_Changed_Same_Slot(t *testing.T) { // STEP 7: wait for proposer duties to be executed currentSlot.Set(phase0.Slot(4)) duties, _ = dutiesMap.Get(phase0.Epoch(0)) - expected = expectedExecutedProposerDuties(handler, []*eth2apiv1.ProposerDuty{duties[2]}) + expected = expectedExecutedDutiesProposer(handler, []*eth2apiv1.ProposerDuty{duties[2]}) setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) @@ -291,7 +291,7 @@ func TestScheduler_Proposer_Reorg_Current(t *testing.T) { ) currentSlot.Set(phase0.Slot(34)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, 0) - fetchDutiesCall, executeDutiesCall := setupProposerDutiesMock(scheduler, dutiesMap) + fetchDutiesCall, executeDutiesCall := setupDutiesMockProposer(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.ProposerDuty{ @@ -347,7 +347,7 @@ func TestScheduler_Proposer_Reorg_Current(t *testing.T) { // STEP 7: The second assigned duty should be executed currentSlot.Set(phase0.Slot(37)) duties, _ := dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedProposerDuties(handler, duties) + expected := expectedExecutedDutiesProposer(handler, duties) setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) @@ -367,7 +367,7 @@ func TestScheduler_Proposer_Reorg_Current_Indices_Changed(t *testing.T) { ) currentSlot.Set(phase0.Slot(34)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, 0) - fetchDutiesCall, executeDutiesCall := setupProposerDutiesMock(scheduler, dutiesMap) + fetchDutiesCall, executeDutiesCall := setupDutiesMockProposer(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(1), []*eth2apiv1.ProposerDuty{ @@ -433,7 +433,7 @@ func TestScheduler_Proposer_Reorg_Current_Indices_Changed(t *testing.T) { // STEP 7: The second assigned duty should be executed currentSlot.Set(phase0.Slot(37)) duties, _ = dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedProposerDuties(handler, []*eth2apiv1.ProposerDuty{duties[0]}) + expected := expectedExecutedDutiesProposer(handler, []*eth2apiv1.ProposerDuty{duties[0]}) setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) @@ -442,7 +442,7 @@ func TestScheduler_Proposer_Reorg_Current_Indices_Changed(t *testing.T) { // STEP 8: The second assigned duty should be executed currentSlot.Set(phase0.Slot(38)) duties, _ = dutiesMap.Get(phase0.Epoch(1)) - expected = expectedExecutedProposerDuties(handler, []*eth2apiv1.ProposerDuty{duties[1]}) + expected = expectedExecutedDutiesProposer(handler, []*eth2apiv1.ProposerDuty{duties[1]}) setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) @@ -462,8 +462,8 @@ func TestScheduler_Proposer_On_Fork(t *testing.T) { ) currentSlot.Set(phase0.Slot(0)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, alanForkEpoch) - fetchDutiesCallGenesis, executeDutiesCallGenesis := setupProposerGenesisDutiesMock(scheduler, dutiesMap) - _, executeDutiesCall := setupProposerDutiesMock(scheduler, dutiesMap) + fetchDutiesCallGenesis, executeDutiesCallGenesis := setupDutiesMockProposerGenesis(scheduler, dutiesMap) + _, executeDutiesCall := setupDutiesMockProposer(scheduler, dutiesMap) startFn() dutiesMap.Set(phase0.Epoch(0), []*eth2apiv1.ProposerDuty{ @@ -483,7 +483,7 @@ func TestScheduler_Proposer_On_Fork(t *testing.T) { // STEP 1: wait for proposer genesis duties to be fetched ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCallGenesis, executeDutiesCallGenesis, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCallGenesis, executeDutiesCallGenesis, timeout) // STEP 2: wait for no action to be taken currentSlot.Set(phase0.Slot(1)) @@ -493,11 +493,11 @@ func TestScheduler_Proposer_On_Fork(t *testing.T) { // STEP 3: wait for proposer duties to be executed currentSlot.Set(phase0.Slot(2)) duties, _ := dutiesMap.Get(phase0.Epoch(0)) - expectedGenesis := expectedExecutedGenesisProposerDuties(handler, duties) + expectedGenesis := expectedExecutedDutiesProposerGenesis(handler, duties) setExecuteGenesisDutyFunc(scheduler, executeDutiesCallGenesis, len(expectedGenesis)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCallGenesis, executeDutiesCallGenesis, timeout, expectedGenesis) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCallGenesis, executeDutiesCallGenesis, timeout, expectedGenesis) waitForNoAction(t, logger, fetchDutiesCallGenesis, executeDutiesCall, timeout) // skip to the next epoch @@ -511,7 +511,7 @@ func TestScheduler_Proposer_On_Fork(t *testing.T) { // fork epoch currentSlot.Set(phase0.Slot(32)) duties, _ = dutiesMap.Get(phase0.Epoch(1)) - expected := expectedExecutedProposerDuties(handler, duties) + expected := expectedExecutedDutiesProposer(handler, duties) setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) diff --git a/operator/duties/scheduler.go b/operator/duties/scheduler.go index 2c42b4c268..d4a42d3baa 100644 --- a/operator/duties/scheduler.go +++ b/operator/duties/scheduler.go @@ -136,6 +136,9 @@ func NewScheduler(opts *SchedulerOptions) *Scheduler { dutyStore = dutystore.New() } + attHandler := NewAttesterHandler(dutyStore.Attester) + syncCommHandler := NewSyncCommitteeHandler(dutyStore.SyncCommittee) + s := &Scheduler{ beaconNode: opts.BeaconNode, executionClient: opts.ExecutionClient, @@ -148,11 +151,11 @@ func NewScheduler(opts *SchedulerOptions) *Scheduler { blockPropagateDelay: blockPropagationDelay, handlers: []dutyHandler{ - NewAttesterHandler(dutyStore.Attester), + attHandler, NewProposerHandler(dutyStore.Proposer), - NewSyncCommitteeHandler(dutyStore.SyncCommittee), + syncCommHandler, NewVoluntaryExitHandler(dutyStore.VoluntaryExit, opts.ValidatorExitCh), - NewCommitteeHandler(dutyStore.Attester, dutyStore.SyncCommittee), + NewCommitteeHandler(attHandler, syncCommHandler), NewValidatorRegistrationHandler(), }, @@ -306,8 +309,8 @@ func (s *Scheduler) HandleHeadEvent(logger *zap.Logger) func(event *eth2apiv1.Ev // check for reorg epoch := s.network.Beacon.EstimatedEpochAtSlot(data.Slot) - buildStr := fmt.Sprintf("e%v-s%v-#%v", epoch, data.Slot, data.Slot%32+1) - logger := logger.With(zap.String("epoch_slot_pos", buildStr)) + tickerID := fields.FormatSlotTickerID(epoch, data.Slot) + logger := logger.With(fields.SlotTickerID(tickerID)) if s.lastBlockEpoch != 0 { if epoch > s.lastBlockEpoch { // Change of epoch. diff --git a/operator/duties/scheduler_test.go b/operator/duties/scheduler_test.go index e963e782ed..13d341ce9a 100644 --- a/operator/duties/scheduler_test.go +++ b/operator/duties/scheduler_test.go @@ -260,7 +260,7 @@ func waitForDutiesFetch(t *testing.T, logger *zap.Logger, fetchDutiesCall chan s } } -func waitForGenesisDutiesFetch(t *testing.T, logger *zap.Logger, fetchDutiesCall chan struct{}, executeDutiesCall chan []*genesisspectypes.Duty, timeout time.Duration) { +func waitForDutiesFetchGenesis(t *testing.T, logger *zap.Logger, fetchDutiesCall chan struct{}, executeDutiesCall chan []*genesisspectypes.Duty, timeout time.Duration) { select { case <-fetchDutiesCall: logger.Debug("duties fetched") @@ -317,7 +317,7 @@ func waitForDutiesExecution(t *testing.T, logger *zap.Logger, fetchDutiesCall ch } } -func waitForGenesisDutiesExecution(t *testing.T, logger *zap.Logger, fetchDutiesCall chan struct{}, executeDutiesCall chan []*genesisspectypes.Duty, timeout time.Duration, expectedDuties []*genesisspectypes.Duty) { +func waitForDutiesExecutionGenesis(t *testing.T, logger *zap.Logger, fetchDutiesCall chan struct{}, executeDutiesCall chan []*genesisspectypes.Duty, timeout time.Duration, expectedDuties []*genesisspectypes.Duty) { select { case <-fetchDutiesCall: require.FailNow(t, "unexpected duties call") diff --git a/operator/duties/sync_committee.go b/operator/duties/sync_committee.go index 13f2776e16..4f63b214c6 100644 --- a/operator/duties/sync_committee.go +++ b/operator/duties/sync_committee.go @@ -26,13 +26,15 @@ type SyncCommitteeHandler struct { // preparationSlots is the number of slots ahead of the sync committee // period change at which to prepare the relevant duties. preparationSlots uint64 + firstRun bool } func NewSyncCommitteeHandler(duties *dutystore.SyncCommitteeDuties) *SyncCommitteeHandler { h := &SyncCommitteeHandler{ - duties: duties, + duties: duties, + firstRun: true, } - h.fetchCurrentPeriod = true + return h } @@ -40,26 +42,7 @@ func (h *SyncCommitteeHandler) Name() string { return spectypes.BNRoleSyncCommittee.String() } -// HandleDuties manages the duty lifecycle, handling different cases: -// -// On First Run: -// 1. Fetch duties for the current period. -// 2. If necessary, fetch duties for the next period. -// 3. Execute duties. -// -// On Re-org: -// 1. Execute duties. -// 2. If necessary, fetch duties for the next period. -// -// On Indices Change: -// 1. Execute duties. -// 2. ResetEpoch duties for the current period. -// 3. Fetch duties for the current period. -// 4. If necessary, fetch duties for the next period. -// -// On Ticker event: -// 1. Execute duties. -// 2. If necessary, fetch duties for the next period. +// HandleDuties manages the duty lifecycle func (h *SyncCommitteeHandler) HandleDuties(ctx context.Context) { h.logger.Info("starting duty handler") defer h.logger.Info("duty handler exited") @@ -68,10 +51,6 @@ func (h *SyncCommitteeHandler) HandleDuties(ctx context.Context) { // The 1.5 epochs timing helps ensure setup occurs when the beacon node is likely less busy. h.preparationSlots = h.network.Beacon.SlotsPerEpoch() * 3 / 2 - if h.shouldFetchNextPeriod(h.network.Beacon.EstimatedCurrentSlot()) { - h.fetchNextPeriod = true - } - next := h.ticker.Next() for { select { @@ -83,66 +62,58 @@ func (h *SyncCommitteeHandler) HandleDuties(ctx context.Context) { next = h.ticker.Next() epoch := h.network.Beacon.EstimatedEpochAtSlot(slot) period := h.network.Beacon.EstimatedSyncCommitteePeriodAtEpoch(epoch) - buildStr := fmt.Sprintf("p%v-e%v-s%v-#%v", period, epoch, slot, slot%32+1) - h.logger.Debug("🛠 ticker event", zap.String("period_epoch_slot_pos", buildStr)) - - ctx, cancel := context.WithDeadline(ctx, h.network.Beacon.GetSlotStartTime(slot+1).Add(100*time.Millisecond)) - h.processExecution(period, slot) - h.processFetching(ctx, period, true) - cancel() - - // if we have reached the preparation slots -1, prepare the next period duties in the next slot. - periodSlots := h.slotsPerPeriod() - if uint64(slot)%periodSlots == periodSlots-h.preparationSlots-1 { - h.fetchNextPeriod = true - } + tickerID := fields.FormatSlotTickerCommitteeID(period, epoch, slot) + h.logger.Debug("🛠 ticker event", fields.SlotTickerID(tickerID)) - // last slot of period - if slot == h.network.Beacon.LastSlotOfSyncPeriod(period) { - h.duties.Reset(period - 1) + if !h.network.PastAlanForkAtEpoch(epoch) { + if h.firstRun { + h.processFirstRun(ctx, period, slot) + } + h.processExecution(period, slot) + if h.indicesChanged { + h.processIndicesChange(period, slot) + } + h.processFetching(ctx, period, slot, true) + h.processSlotTransition(period, slot) } case reorgEvent := <-h.reorg: epoch := h.network.Beacon.EstimatedEpochAtSlot(reorgEvent.Slot) period := h.network.Beacon.EstimatedSyncCommitteePeriodAtEpoch(epoch) + tickerID := fields.FormatSlotTickerCommitteeID(period, epoch, reorgEvent.Slot) + h.logger.Info("🔀 reorg event received", fields.SlotTickerID(tickerID), zap.Any("event", reorgEvent)) - buildStr := fmt.Sprintf("p%v-e%v-s%v-#%v", period, epoch, reorgEvent.Slot, reorgEvent.Slot%32+1) - h.logger.Info("🔀 reorg event received", zap.String("period_epoch_slot_pos", buildStr), zap.Any("event", reorgEvent)) - - // reset current epoch duties - if reorgEvent.Current && h.shouldFetchNextPeriod(reorgEvent.Slot) { - h.duties.Reset(period + 1) - h.fetchNextPeriod = true + if !h.network.PastAlanForkAtEpoch(epoch) { + h.processReorg(period, reorgEvent) } case <-h.indicesChange: slot := h.network.Beacon.EstimatedCurrentSlot() epoch := h.network.Beacon.EstimatedEpochAtSlot(slot) period := h.network.Beacon.EstimatedSyncCommitteePeriodAtEpoch(epoch) - buildStr := fmt.Sprintf("p%v-e%v-s%v-#%v", period, epoch, slot, slot%32+1) - h.logger.Info("🔁 indices change received", zap.String("period_epoch_slot_pos", buildStr)) - - h.fetchCurrentPeriod = true + tickerID := fields.FormatSlotTickerCommitteeID(period, epoch, slot) + h.logger.Info("🔁 indices change received", fields.SlotTickerID(tickerID)) - // reset next period duties if in appropriate slot range - if h.shouldFetchNextPeriod(slot) { - h.fetchNextPeriod = true + if !h.network.PastAlanForkAtEpoch(epoch) { + h.indicesChanged = true } } } } -func (h *SyncCommitteeHandler) HandleInitialDuties(ctx context.Context) { - ctx, cancel := context.WithTimeout(ctx, h.network.Beacon.SlotDurationSec()/2) - defer cancel() +func (h *SyncCommitteeHandler) processFirstRun(ctx context.Context, period uint64, slot phase0.Slot) { + h.fetchCurrentPeriod = true + h.processFetching(ctx, period, slot, false) - epoch := h.network.Beacon.EstimatedCurrentEpoch() - period := h.network.Beacon.EstimatedSyncCommitteePeriodAtEpoch(epoch) - h.processFetching(ctx, period, false) + periodSlots := h.slotsPerPeriod() + if uint64(slot)%periodSlots > periodSlots-h.preparationSlots-1 { + h.fetchNextPeriod = true + } + h.firstRun = false } -func (h *SyncCommitteeHandler) processFetching(ctx context.Context, period uint64, waitForInitial bool) { - ctx, cancel := context.WithCancel(ctx) +func (h *SyncCommitteeHandler) processFetching(ctx context.Context, period uint64, slot phase0.Slot, waitForInitial bool) { + ctx, cancel := context.WithDeadline(ctx, h.network.Beacon.GetSlotStartTime(slot+1).Add(100*time.Millisecond)) defer cancel() if h.fetchCurrentPeriod { @@ -163,33 +134,52 @@ func (h *SyncCommitteeHandler) processFetching(ctx context.Context, period uint6 } func (h *SyncCommitteeHandler) processExecution(period uint64, slot phase0.Slot) { - // range over duties and execute duties := h.duties.CommitteePeriodDuties(period) if duties == nil { return } - if !h.network.PastAlanForkAtEpoch(h.network.Beacon.EstimatedEpochAtSlot(slot)) { - toExecute := make([]*genesisspectypes.Duty, 0, len(duties)*2) - for _, d := range duties { - if h.shouldExecute(d, slot) { - toExecute = append(toExecute, h.toGenesisSpecDuty(d, slot, genesisspectypes.BNRoleSyncCommittee)) - toExecute = append(toExecute, h.toGenesisSpecDuty(d, slot, genesisspectypes.BNRoleSyncCommitteeContribution)) - } + toExecute := make([]*genesisspectypes.Duty, 0, len(duties)*2) + for _, d := range duties { + if h.shouldExecute(d, slot) { + toExecute = append(toExecute, h.toGenesisSpecDuty(d, slot, genesisspectypes.BNRoleSyncCommittee)) + toExecute = append(toExecute, h.toGenesisSpecDuty(d, slot, genesisspectypes.BNRoleSyncCommitteeContribution)) } + } - h.dutiesExecutor.ExecuteGenesisDuties(h.logger, toExecute) - return + h.dutiesExecutor.ExecuteGenesisDuties(h.logger, toExecute) +} + +func (h *SyncCommitteeHandler) processIndicesChange(period uint64, slot phase0.Slot) { + h.fetchCurrentPeriod = true + + // reset next period duties if in appropriate slot range + if h.shouldFetchNextPeriod(slot) { + h.duties.Reset(period + 1) + h.fetchNextPeriod = true } - toExecute := make([]*spectypes.ValidatorDuty, 0, len(duties)) - for _, d := range duties { - if h.shouldExecute(d, slot) { - toExecute = append(toExecute, h.toSpecDuty(d, slot, spectypes.BNRoleSyncCommitteeContribution)) - } + h.indicesChanged = false +} + +func (h *SyncCommitteeHandler) processReorg(period uint64, reorgEvent ReorgEvent) { + if reorgEvent.Current && h.shouldFetchNextPeriod(reorgEvent.Slot) { + h.duties.Reset(period + 1) + h.fetchNextPeriod = true } +} - h.dutiesExecutor.ExecuteDuties(h.logger, toExecute) +func (h *SyncCommitteeHandler) processSlotTransition(period uint64, slot phase0.Slot) { + // if we have reached the preparation slots -1, prepare the next period duties in the next slot. + periodSlots := h.slotsPerPeriod() + if uint64(slot)%periodSlots == periodSlots-h.preparationSlots-1 { + h.fetchNextPeriod = true + } + + // last slot of period + if slot == h.network.Beacon.LastSlotOfSyncPeriod(period) { + h.duties.Reset(period - 1) + } } func (h *SyncCommitteeHandler) fetchAndProcessDuties(ctx context.Context, period uint64, waitForInitial bool) error { @@ -324,7 +314,7 @@ func calculateSubscriptions(endEpoch phase0.Epoch, duties []*eth2apiv1.SyncCommi func (h *SyncCommitteeHandler) shouldFetchNextPeriod(slot phase0.Slot) bool { periodSlots := h.slotsPerPeriod() - return uint64(slot)%periodSlots > periodSlots-h.preparationSlots-2 + return uint64(slot)%periodSlots >= periodSlots-h.preparationSlots-1 } func (h *SyncCommitteeHandler) slotsPerPeriod() uint64 { diff --git a/operator/duties/sync_committee_genesis_test.go b/operator/duties/sync_committee_genesis_test.go index 1c1cfc4765..4a1a08795b 100644 --- a/operator/duties/sync_committee_genesis_test.go +++ b/operator/duties/sync_committee_genesis_test.go @@ -21,11 +21,10 @@ import ( ssvtypes "github.com/ssvlabs/ssv/protocol/v2/types" ) -func setupSyncCommitteeGenesisDutiesMock( +func setupGenesisDutiesMockSyncCommittee( s *Scheduler, activeShares []*ssvtypes.SSVShare, dutiesMap *hashmap.Map[uint64, []*v1.SyncCommitteeDuty], - waitForDuties *SafeValue[bool], ) (chan struct{}, chan []*genesisspectypes.Duty) { fetchDutiesCall := make(chan struct{}) executeDutiesCall := make(chan []*genesisspectypes.Duty) @@ -59,9 +58,7 @@ func setupSyncCommitteeGenesisDutiesMock( s.beaconNode.(*MockBeaconNode).EXPECT().SyncCommitteeDuties(gomock.Any(), gomock.Any(), gomock.Any()).DoAndReturn( func(ctx context.Context, epoch phase0.Epoch, indices []phase0.ValidatorIndex) ([]*v1.SyncCommitteeDuty, error) { - if waitForDuties.Get() { - fetchDutiesCall <- struct{}{} - } + fetchDutiesCall <- struct{}{} period := s.network.Beacon.EstimatedSyncCommitteePeriodAtEpoch(epoch) duties, _ := dutiesMap.Get(period) return duties, nil @@ -80,7 +77,7 @@ func setupSyncCommitteeGenesisDutiesMock( return fetchDutiesCall, executeDutiesCall } -func expectedExecutedGenesisSyncCommitteeDuties(handler *SyncCommitteeHandler, duties []*v1.SyncCommitteeDuty, slot phase0.Slot) []*genesisspectypes.Duty { +func expectedExecutedDutiesSyncCommitteeGenesis(handler *SyncCommitteeHandler, duties []*v1.SyncCommitteeDuty, slot phase0.Slot) []*genesisspectypes.Duty { expectedDuties := make([]*genesisspectypes.Duty, 0) for _, d := range duties { expectedDuties = append(expectedDuties, handler.toGenesisSpecDuty(d, slot, genesisspectypes.BNRoleSyncCommittee)) @@ -91,12 +88,11 @@ func expectedExecutedGenesisSyncCommitteeDuties(handler *SyncCommitteeHandler, d func TestScheduler_SyncCommittee_Genesis_Same_Period(t *testing.T) { var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{{ + handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) + currentSlot = &SafeValue[phase0.Slot]{} + forkEpoch = goclient.FarFutureEpoch + dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{{ Share: spectypes.Share{ Committee: []*spectypes.ShareMember{ {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, @@ -115,34 +111,35 @@ func TestScheduler_SyncCommittee_Genesis_Same_Period(t *testing.T) { // STEP 1: wait for sync committee duties to be fetched (handle initial duties) currentSlot.Set(phase0.Slot(1)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeGenesisDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupGenesisDutiesMockSyncCommittee(scheduler, activeShares, dutiesMap) startFn() // STEP 1: wait for sync committee duties to be fetched and executed at the same slot duties, _ := dutiesMap.Get(0) - expected := expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected := expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // STEP 2: expect sync committee duties to be executed at the same period currentSlot.Set(phase0.Slot(2)) duties, _ = dutiesMap.Get(0) - expected = expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected = expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // STEP 3: expect sync committee duties to be executed at the last slot of the period currentSlot.Set(scheduler.network.Beacon.LastSlotOfSyncPeriod(0)) duties, _ = dutiesMap.Get(0) - expected = expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected = expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // STEP 4: expect no action to be taken as we are in the next period firstSlotOfNextPeriod := scheduler.network.Beacon.GetEpochFirstSlot(scheduler.network.Beacon.FirstEpochOfSyncPeriod(1)) @@ -157,12 +154,11 @@ func TestScheduler_SyncCommittee_Genesis_Same_Period(t *testing.T) { func TestScheduler_SyncCommittee_Genesis_Current_Next_Periods(t *testing.T) { var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{ + handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) + currentSlot = &SafeValue[phase0.Slot]{} + forkEpoch = goclient.FarFutureEpoch + dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{ { Share: spectypes.Share{ Committee: []*spectypes.ShareMember{ @@ -197,44 +193,46 @@ func TestScheduler_SyncCommittee_Genesis_Current_Next_Periods(t *testing.T) { // STEP 1: wait for sync committee duties to be fetched (handle initial duties) currentSlot.Set(phase0.Slot(256*32 - 49)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeGenesisDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupGenesisDutiesMockSyncCommittee(scheduler, activeShares, dutiesMap) startFn() duties, _ := dutiesMap.Get(0) - expected := expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected := expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // STEP 2: wait for sync committee duties to be executed currentSlot.Set(phase0.Slot(256*32 - 48)) duties, _ = dutiesMap.Get(0) - expected = expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected = expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // STEP 3: wait for sync committee duties to be executed currentSlot.Set(phase0.Slot(256*32 - 47)) duties, _ = dutiesMap.Get(0) - expected = expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected = expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // ... // STEP 4: new period, wait for sync committee duties to be executed currentSlot.Set(phase0.Slot(256 * 32)) duties, _ = dutiesMap.Get(1) - expected = expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected = expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -243,12 +241,11 @@ func TestScheduler_SyncCommittee_Genesis_Current_Next_Periods(t *testing.T) { func TestScheduler_SyncCommittee_Genesis_Indices_Changed(t *testing.T) { var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{ + handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) + currentSlot = &SafeValue[phase0.Slot]{} + forkEpoch = goclient.FarFutureEpoch + dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{ { Share: spectypes.Share{ Committee: []*spectypes.ShareMember{ @@ -269,7 +266,7 @@ func TestScheduler_SyncCommittee_Genesis_Indices_Changed(t *testing.T) { ) currentSlot.Set(phase0.Slot(256*32 - 3)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeGenesisDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupGenesisDutiesMockSyncCommittee(scheduler, activeShares, dutiesMap) startFn() dutiesMap.Set(1, []*v1.SyncCommitteeDuty{ @@ -279,10 +276,10 @@ func TestScheduler_SyncCommittee_Genesis_Indices_Changed(t *testing.T) { }, }) - // STEP 1: wait for sync committee duties to be fetched for next period - waitForDuties.Set(true) + // STEP 1: wait for sync committee duties to be fetched for current and next period ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger a change in active indices scheduler.indicesChg <- struct{}{} @@ -296,8 +293,8 @@ func TestScheduler_SyncCommittee_Genesis_Indices_Changed(t *testing.T) { // STEP 3: wait for sync committee duties to be fetched again currentSlot.Set(phase0.Slot(256*32 - 2)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 4: no action should be taken currentSlot.Set(phase0.Slot(256*32 - 1)) @@ -307,11 +304,11 @@ func TestScheduler_SyncCommittee_Genesis_Indices_Changed(t *testing.T) { // STEP 5: execute duties currentSlot.Set(phase0.Slot(256 * 32)) duties, _ = dutiesMap.Get(1) - expected := expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected := expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -320,12 +317,11 @@ func TestScheduler_SyncCommittee_Genesis_Indices_Changed(t *testing.T) { func TestScheduler_SyncCommittee_Genesis_Multiple_Indices_Changed_Same_Slot(t *testing.T) { var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{ + handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) + currentSlot = &SafeValue[phase0.Slot]{} + forkEpoch = goclient.FarFutureEpoch + dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{ { Share: spectypes.Share{ Committee: []*spectypes.ShareMember{ @@ -346,12 +342,13 @@ func TestScheduler_SyncCommittee_Genesis_Multiple_Indices_Changed_Same_Slot(t *t ) currentSlot.Set(phase0.Slot(256*32 - 3)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeGenesisDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupGenesisDutiesMockSyncCommittee(scheduler, activeShares, dutiesMap) startFn() - // STEP 1: wait for no action to be taken + // STEP 1: wait for sync committee duties to be fetched for current and next period ticker.Send(currentSlot.Get()) - waitForNoActionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger a change in active indices scheduler.indicesChg <- struct{}{} @@ -374,10 +371,9 @@ func TestScheduler_SyncCommittee_Genesis_Multiple_Indices_Changed_Same_Slot(t *t // STEP 4: wait for sync committee duties to be fetched again currentSlot.Set(phase0.Slot(256*32 - 2)) - waitForDuties.Set(true) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 5: no action should be taken currentSlot.Set(phase0.Slot(256*32 - 1)) @@ -387,11 +383,11 @@ func TestScheduler_SyncCommittee_Genesis_Multiple_Indices_Changed_Same_Slot(t *t // STEP 6: The first assigned duty should not be executed, but the second one should currentSlot.Set(phase0.Slot(256 * 32)) duties, _ = dutiesMap.Get(1) - expected := expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected := expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -401,12 +397,11 @@ func TestScheduler_SyncCommittee_Genesis_Multiple_Indices_Changed_Same_Slot(t *t // reorg current dependent root changed func TestScheduler_SyncCommittee_Genesis_Reorg_Current(t *testing.T) { var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{ + handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) + currentSlot = &SafeValue[phase0.Slot]{} + forkEpoch = goclient.FarFutureEpoch + dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{ { Share: spectypes.Share{ Committee: []*spectypes.ShareMember{ @@ -427,7 +422,7 @@ func TestScheduler_SyncCommittee_Genesis_Reorg_Current(t *testing.T) { ) currentSlot.Set(phase0.Slot(256*32 - 3)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeGenesisDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupGenesisDutiesMockSyncCommittee(scheduler, activeShares, dutiesMap) startFn() dutiesMap.Set(1, []*v1.SyncCommitteeDuty{ @@ -438,9 +433,9 @@ func TestScheduler_SyncCommittee_Genesis_Reorg_Current(t *testing.T) { }) // STEP 1: wait for sync committee duties to be fetched and executed at the same slot - waitForDuties.Set(true) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger head event e := &v1.Event{ @@ -476,16 +471,16 @@ func TestScheduler_SyncCommittee_Genesis_Reorg_Current(t *testing.T) { // STEP 5: wait for sync committee duties to be fetched again for the current epoch currentSlot.Set(phase0.Slot(256*32 - 1)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 6: The first assigned duty should not be executed, but the second one should currentSlot.Set(phase0.Slot(256 * 32)) duties, _ := dutiesMap.Get(1) - expected := expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected := expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -495,12 +490,11 @@ func TestScheduler_SyncCommittee_Genesis_Reorg_Current(t *testing.T) { // reorg current dependent root changed including indices change in the same slot func TestScheduler_SyncCommittee_Genesis_Reorg_Current_Indices_Changed(t *testing.T) { var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{ + handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) + currentSlot = &SafeValue[phase0.Slot]{} + forkEpoch = goclient.FarFutureEpoch + dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{ { Share: spectypes.Share{ Committee: []*spectypes.ShareMember{ @@ -529,7 +523,7 @@ func TestScheduler_SyncCommittee_Genesis_Reorg_Current_Indices_Changed(t *testin ) currentSlot.Set(phase0.Slot(256*32 - 3)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeGenesisDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupGenesisDutiesMockSyncCommittee(scheduler, activeShares, dutiesMap) startFn() dutiesMap.Set(1, []*v1.SyncCommitteeDuty{ @@ -540,9 +534,9 @@ func TestScheduler_SyncCommittee_Genesis_Reorg_Current_Indices_Changed(t *testin }) // STEP 1: wait for sync committee duties to be fetched and executed at the same slot - waitForDuties.Set(true) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 2: trigger head event e := &v1.Event{ @@ -587,17 +581,17 @@ func TestScheduler_SyncCommittee_Genesis_Reorg_Current_Indices_Changed(t *testin // STEP 5: wait for sync committee duties to be fetched again for the current epoch currentSlot.Set(phase0.Slot(256*32 - 1)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - waitForGenesisDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) // STEP 6: The first assigned duty should not be executed, but the second and the new from indices change should currentSlot.Set(phase0.Slot(256 * 32)) duties, _ = dutiesMap.Get(1) - expected := expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected := expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // Stop scheduler & wait for graceful exit. cancel() @@ -606,12 +600,11 @@ func TestScheduler_SyncCommittee_Genesis_Reorg_Current_Indices_Changed(t *testin func TestScheduler_SyncCommittee_Genesis_Early_Block(t *testing.T) { var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = goclient.FarFutureEpoch - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{{ + handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) + currentSlot = &SafeValue[phase0.Slot]{} + forkEpoch = goclient.FarFutureEpoch + dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() + activeShares = []*ssvtypes.SSVShare{{ Share: spectypes.Share{ Committee: []*spectypes.ShareMember{ {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, @@ -629,31 +622,32 @@ func TestScheduler_SyncCommittee_Genesis_Early_Block(t *testing.T) { currentSlot.Set(phase0.Slot(0)) scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeGenesisDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) + fetchDutiesCall, executeDutiesCall := setupGenesisDutiesMockSyncCommittee(scheduler, activeShares, dutiesMap) startFn() duties, _ := dutiesMap.Get(0) - expected := expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected := expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) // STEP 1: wait for sync committee duties to be fetched and executed at the same slot ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesFetchGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // STEP 2: expect sync committee duties to be executed at the same period currentSlot.Set(phase0.Slot(1)) duties, _ = dutiesMap.Get(0) - expected = expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected = expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) // STEP 3: wait for sync committee duties to be executed faster than 1/3 of the slot duration startTime := time.Now() currentSlot.Set(phase0.Slot(2)) duties, _ = dutiesMap.Get(0) - expected = expectedExecutedGenesisSyncCommitteeDuties(handler, duties, currentSlot.Get()) + expected = expectedExecutedDutiesSyncCommitteeGenesis(handler, duties, currentSlot.Get()) setExecuteGenesisDutyFunc(scheduler, executeDutiesCall, len(expected)) ticker.Send(currentSlot.Get()) @@ -665,7 +659,7 @@ func TestScheduler_SyncCommittee_Genesis_Early_Block(t *testing.T) { }, } scheduler.HandleHeadEvent(logger)(e) - waitForGenesisDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) + waitForDutiesExecutionGenesis(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) require.Less(t, time.Since(startTime), scheduler.network.Beacon.SlotDurationSec()/3) // Stop scheduler & wait for graceful exit. diff --git a/operator/duties/sync_committee_test.go b/operator/duties/sync_committee_test.go deleted file mode 100644 index aa3e65270a..0000000000 --- a/operator/duties/sync_committee_test.go +++ /dev/null @@ -1,674 +0,0 @@ -package duties - -import ( - "context" - "testing" - "time" - - v1 "github.com/attestantio/go-eth2-client/api/v1" - "github.com/attestantio/go-eth2-client/spec/phase0" - "github.com/stretchr/testify/require" - "go.uber.org/mock/gomock" - - "github.com/ssvlabs/ssv/utils/hashmap" - - spectypes "github.com/ssvlabs/ssv-spec/types" - - "github.com/ssvlabs/ssv/operator/duties/dutystore" - mocknetwork "github.com/ssvlabs/ssv/protocol/v2/blockchain/beacon/mocks" - ssvtypes "github.com/ssvlabs/ssv/protocol/v2/types" -) - -func setupSyncCommitteeDutiesMock( - s *Scheduler, - activeShares []*ssvtypes.SSVShare, - dutiesMap *hashmap.Map[uint64, []*v1.SyncCommitteeDuty], - waitForDuties *SafeValue[bool], -) (chan struct{}, chan []*spectypes.ValidatorDuty) { - fetchDutiesCall := make(chan struct{}) - executeDutiesCall := make(chan []*spectypes.ValidatorDuty) - - s.network.Beacon.(*mocknetwork.MockBeaconNetwork).EXPECT().EstimatedSyncCommitteePeriodAtEpoch(gomock.Any()).DoAndReturn( - func(epoch phase0.Epoch) uint64 { - return uint64(epoch) / s.network.Beacon.EpochsPerSyncCommitteePeriod() - }, - ).AnyTimes() - - s.network.Beacon.(*mocknetwork.MockBeaconNetwork).EXPECT().FirstEpochOfSyncPeriod(gomock.Any()).DoAndReturn( - func(period uint64) phase0.Epoch { - return phase0.Epoch(period * s.network.Beacon.EpochsPerSyncCommitteePeriod()) - }, - ).AnyTimes() - - s.network.Beacon.(*mocknetwork.MockBeaconNetwork).EXPECT().LastSlotOfSyncPeriod(gomock.Any()).DoAndReturn( - func(period uint64) phase0.Slot { - lastEpoch := s.network.Beacon.FirstEpochOfSyncPeriod(period+1) - 1 - // If we are in the sync committee that ends at slot x we do not generate a message during slot x-1 - // as it will never be included, hence -1. - return s.network.Beacon.GetEpochFirstSlot(lastEpoch+1) - 2 - }, - ).AnyTimes() - - s.network.Beacon.(*mocknetwork.MockBeaconNetwork).EXPECT().GetEpochFirstSlot(gomock.Any()).DoAndReturn( - func(epoch phase0.Epoch) phase0.Slot { - return phase0.Slot(uint64(epoch) * s.network.Beacon.SlotsPerEpoch()) - }, - ).AnyTimes() - - s.beaconNode.(*MockBeaconNode).EXPECT().SyncCommitteeDuties(gomock.Any(), gomock.Any(), gomock.Any()).DoAndReturn( - func(ctx context.Context, epoch phase0.Epoch, indices []phase0.ValidatorIndex) ([]*v1.SyncCommitteeDuty, error) { - if waitForDuties.Get() { - fetchDutiesCall <- struct{}{} - } - period := s.network.Beacon.EstimatedSyncCommitteePeriodAtEpoch(epoch) - duties, _ := dutiesMap.Get(period) - return duties, nil - }).AnyTimes() - - s.validatorProvider.(*MockValidatorProvider).EXPECT().SelfParticipatingValidators(gomock.Any()).Return(activeShares).AnyTimes() - s.validatorProvider.(*MockValidatorProvider).EXPECT().ParticipatingValidators(gomock.Any()).Return(activeShares).AnyTimes() - - s.validatorController.(*MockValidatorController).EXPECT().AllActiveIndices(gomock.Any(), gomock.Any()).DoAndReturn( - func(epoch phase0.Epoch, afterInit bool) []phase0.ValidatorIndex { - return indicesFromShares(activeShares) - }).AnyTimes() - - s.beaconNode.(*MockBeaconNode).EXPECT().SubmitSyncCommitteeSubscriptions(gomock.Any(), gomock.Any()).Return(nil).AnyTimes() - - return fetchDutiesCall, executeDutiesCall -} - -func expectedExecutedSyncCommitteeDuties(handler *SyncCommitteeHandler, duties []*v1.SyncCommitteeDuty, slot phase0.Slot) []*spectypes.ValidatorDuty { - expectedDuties := make([]*spectypes.ValidatorDuty, 0) - for _, d := range duties { - if !handler.network.PastAlanForkAtEpoch(handler.network.Beacon.EstimatedEpochAtSlot(slot)) { - expectedDuties = append(expectedDuties, handler.toSpecDuty(d, slot, spectypes.BNRoleSyncCommittee)) - } - expectedDuties = append(expectedDuties, handler.toSpecDuty(d, slot, spectypes.BNRoleSyncCommitteeContribution)) - } - return expectedDuties -} - -func TestScheduler_SyncCommittee_Same_Period(t *testing.T) { - var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{{ - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 1, - }, - }} - ) - dutiesMap.Set(0, []*v1.SyncCommitteeDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - - // STEP 1: wait for sync committee duties to be fetched (handle initial duties) - currentSlot.Set(phase0.Slot(1)) - scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) - startFn() - - // STEP 1: wait for sync committee duties to be fetched and executed at the same slot - duties, _ := dutiesMap.Get(0) - expected := expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // STEP 2: expect sync committee duties to be executed at the same period - currentSlot.Set(phase0.Slot(2)) - duties, _ = dutiesMap.Get(0) - expected = expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // STEP 3: expect sync committee duties to be executed at the last slot of the period - currentSlot.Set(scheduler.network.Beacon.LastSlotOfSyncPeriod(0)) - duties, _ = dutiesMap.Get(0) - expected = expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // STEP 4: expect no action to be taken as we are in the next period - firstSlotOfNextPeriod := scheduler.network.Beacon.GetEpochFirstSlot(scheduler.network.Beacon.FirstEpochOfSyncPeriod(1)) - currentSlot.Set(firstSlotOfNextPeriod) - ticker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -func TestScheduler_SyncCommittee_Current_Next_Periods(t *testing.T) { - var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{ - { - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 1, - }, - }, - { - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 2, - }, - }, - } - ) - dutiesMap.Set(0, []*v1.SyncCommitteeDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - dutiesMap.Set(1, []*v1.SyncCommitteeDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 4}, - ValidatorIndex: phase0.ValidatorIndex(2), - }, - }) - - // STEP 1: wait for sync committee duties to be fetched (handle initial duties) - currentSlot.Set(phase0.Slot(256*32 - 49)) - scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) - startFn() - - duties, _ := dutiesMap.Get(0) - expected := expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // STEP 2: wait for sync committee duties to be executed - currentSlot.Set(phase0.Slot(256*32 - 48)) - duties, _ = dutiesMap.Get(0) - expected = expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // STEP 3: wait for sync committee duties to be executed - currentSlot.Set(phase0.Slot(256*32 - 47)) - duties, _ = dutiesMap.Get(0) - expected = expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // ... - - // STEP 4: new period, wait for sync committee duties to be executed - currentSlot.Set(phase0.Slot(256 * 32)) - duties, _ = dutiesMap.Get(1) - expected = expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -func TestScheduler_SyncCommittee_Indices_Changed(t *testing.T) { - var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{ - { - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 1, - }, - }, - { - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 2, - }, - }, - } - ) - currentSlot.Set(phase0.Slot(256*32 - 3)) - scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) - startFn() - - dutiesMap.Set(1, []*v1.SyncCommitteeDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - - // STEP 1: wait for sync committee duties to be fetched for next period - waitForDuties.Set(true) - ticker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: trigger a change in active indices - scheduler.indicesChg <- struct{}{} - duties, _ := dutiesMap.Get(1) - dutiesMap.Set(1, append(duties, &v1.SyncCommitteeDuty{ - PubKey: phase0.BLSPubKey{1, 2, 4}, - ValidatorIndex: phase0.ValidatorIndex(2), - })) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: wait for sync committee duties to be fetched again - currentSlot.Set(phase0.Slot(256*32 - 2)) - ticker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 4: no action should be taken - currentSlot.Set(phase0.Slot(256*32 - 1)) - ticker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 5: execute duties - currentSlot.Set(phase0.Slot(256 * 32)) - duties, _ = dutiesMap.Get(1) - expected := expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -func TestScheduler_SyncCommittee_Multiple_Indices_Changed_Same_Slot(t *testing.T) { - var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{ - { - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 1, - }, - }, - { - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 2, - }, - }, - } - ) - currentSlot.Set(phase0.Slot(256*32 - 3)) - scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) - startFn() - - // STEP 1: wait for no action to be taken - ticker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: trigger a change in active indices - scheduler.indicesChg <- struct{}{} - dutiesMap.Set(1, []*v1.SyncCommitteeDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: trigger a change in active indices - scheduler.indicesChg <- struct{}{} - duties, _ := dutiesMap.Get(1) - dutiesMap.Set(1, append(duties, &v1.SyncCommitteeDuty{ - PubKey: phase0.BLSPubKey{1, 2, 4}, - ValidatorIndex: phase0.ValidatorIndex(2), - })) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 4: wait for sync committee duties to be fetched again - currentSlot.Set(phase0.Slot(256*32 - 2)) - waitForDuties.Set(true) - ticker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 5: no action should be taken - currentSlot.Set(phase0.Slot(256*32 - 1)) - ticker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 6: The first assigned duty should not be executed, but the second one should - currentSlot.Set(phase0.Slot(256 * 32)) - duties, _ = dutiesMap.Get(1) - expected := expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -// reorg current dependent root changed -func TestScheduler_SyncCommittee_Reorg_Current(t *testing.T) { - var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{ - { - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 1, - }, - }, - { - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 2, - }, - }, - } - ) - currentSlot.Set(phase0.Slot(256*32 - 3)) - scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) - startFn() - - dutiesMap.Set(1, []*v1.SyncCommitteeDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - - // STEP 1: wait for sync committee duties to be fetched and executed at the same slot - waitForDuties.Set(true) - ticker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: trigger head event - e := &v1.Event{ - Data: &v1.HeadEvent{ - Slot: currentSlot.Get(), - CurrentDutyDependentRoot: phase0.Root{0x01}, - }, - } - scheduler.HandleHeadEvent(logger)(e) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: Ticker with no action - currentSlot.Set(phase0.Slot(256*32 - 2)) - ticker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 4: trigger reorg - e = &v1.Event{ - Data: &v1.HeadEvent{ - Slot: currentSlot.Get(), - CurrentDutyDependentRoot: phase0.Root{0x02}, - }, - } - dutiesMap.Set(1, []*v1.SyncCommitteeDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 4}, - ValidatorIndex: phase0.ValidatorIndex(2), - }, - }) - scheduler.HandleHeadEvent(logger)(e) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 5: wait for sync committee duties to be fetched again for the current epoch - currentSlot.Set(phase0.Slot(256*32 - 1)) - ticker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 6: The first assigned duty should not be executed, but the second one should - currentSlot.Set(phase0.Slot(256 * 32)) - duties, _ := dutiesMap.Get(1) - expected := expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -// reorg current dependent root changed including indices change in the same slot -func TestScheduler_SyncCommittee_Reorg_Current_Indices_Changed(t *testing.T) { - var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{ - { - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 1, - }, - }, - { - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 2, - }, - }, - { - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 3, - }, - }, - } - ) - currentSlot.Set(phase0.Slot(256*32 - 3)) - scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) - startFn() - - dutiesMap.Set(1, []*v1.SyncCommitteeDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - - // STEP 1: wait for sync committee duties to be fetched and executed at the same slot - waitForDuties.Set(true) - ticker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 2: trigger head event - e := &v1.Event{ - Data: &v1.HeadEvent{ - Slot: currentSlot.Get(), - CurrentDutyDependentRoot: phase0.Root{0x01}, - }, - } - scheduler.HandleHeadEvent(logger)(e) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: Ticker with no action - currentSlot.Set(phase0.Slot(256*32 - 2)) - ticker.Send(currentSlot.Get()) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 4: trigger reorg - e = &v1.Event{ - Data: &v1.HeadEvent{ - Slot: currentSlot.Get(), - CurrentDutyDependentRoot: phase0.Root{0x02}, - }, - } - dutiesMap.Set(1, []*v1.SyncCommitteeDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 4}, - ValidatorIndex: phase0.ValidatorIndex(2), - }, - }) - scheduler.HandleHeadEvent(logger)(e) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 3: trigger a change in active indices - scheduler.indicesChg <- struct{}{} - duties, _ := dutiesMap.Get(1) - dutiesMap.Set(1, append(duties, &v1.SyncCommitteeDuty{ - PubKey: phase0.BLSPubKey{1, 2, 5}, - ValidatorIndex: phase0.ValidatorIndex(3), - })) - waitForNoAction(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 5: wait for sync committee duties to be fetched again for the current epoch - currentSlot.Set(phase0.Slot(256*32 - 1)) - ticker.Send(currentSlot.Get()) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - waitForDutiesFetch(t, logger, fetchDutiesCall, executeDutiesCall, timeout) - - // STEP 6: The first assigned duty should not be executed, but the second and the new from indices change should - currentSlot.Set(phase0.Slot(256 * 32)) - duties, _ = dutiesMap.Get(1) - expected := expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} - -func TestScheduler_SyncCommittee_Early_Block(t *testing.T) { - var ( - handler = NewSyncCommitteeHandler(dutystore.NewSyncCommitteeDuties()) - currentSlot = &SafeValue[phase0.Slot]{} - waitForDuties = &SafeValue[bool]{} - forkEpoch = phase0.Epoch(0) - dutiesMap = hashmap.New[uint64, []*v1.SyncCommitteeDuty]() - activeShares = []*ssvtypes.SSVShare{{ - Share: spectypes.Share{ - Committee: []*spectypes.ShareMember{ - {Signer: 1}, {Signer: 2}, {Signer: 3}, {Signer: 4}, - }, - ValidatorIndex: 1, - }, - }} - ) - dutiesMap.Set(0, []*v1.SyncCommitteeDuty{ - { - PubKey: phase0.BLSPubKey{1, 2, 3}, - ValidatorIndex: phase0.ValidatorIndex(1), - }, - }) - - currentSlot.Set(phase0.Slot(0)) - scheduler, logger, ticker, timeout, cancel, schedulerPool, startFn := setupSchedulerAndMocks(t, []dutyHandler{handler}, currentSlot, forkEpoch) - fetchDutiesCall, executeDutiesCall := setupSyncCommitteeDutiesMock(scheduler, activeShares, dutiesMap, waitForDuties) - startFn() - - duties, _ := dutiesMap.Get(0) - expected := expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - // STEP 1: wait for sync committee duties to be fetched and executed at the same slot - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // STEP 2: expect sync committee duties to be executed at the same period - currentSlot.Set(phase0.Slot(1)) - duties, _ = dutiesMap.Get(0) - expected = expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - - // STEP 3: wait for sync committee duties to be executed faster than 1/3 of the slot duration - startTime := time.Now() - currentSlot.Set(phase0.Slot(2)) - duties, _ = dutiesMap.Get(0) - expected = expectedExecutedSyncCommitteeDuties(handler, duties, currentSlot.Get()) - setExecuteDutyFunc(scheduler, executeDutiesCall, len(expected)) - - ticker.Send(currentSlot.Get()) - - // STEP 4: trigger head event (block arrival) - e := &v1.Event{ - Data: &v1.HeadEvent{ - Slot: currentSlot.Get(), - }, - } - scheduler.HandleHeadEvent(logger)(e) - waitForDutiesExecution(t, logger, fetchDutiesCall, executeDutiesCall, timeout, expected) - require.Greater(t, time.Since(startTime), scheduler.network.Beacon.SlotDurationSec()/3) - - // Stop scheduler & wait for graceful exit. - cancel() - require.NoError(t, schedulerPool.Wait()) -} diff --git a/operator/duties/voluntary_exit_genesis_test.go b/operator/duties/voluntary_exit_genesis_test.go index e8719d8922..6e053388f5 100644 --- a/operator/duties/voluntary_exit_genesis_test.go +++ b/operator/duties/voluntary_exit_genesis_test.go @@ -91,7 +91,7 @@ func TestVoluntaryExitHandler_HandleGenesisDuties(t *testing.T) { t.Run("slot = 5, block = 1 - executing duty, fetching block number", func(t *testing.T) { currentSlot.Set(phase0.Slot(normalExit.BlockNumber) + voluntaryExitSlotsToPostpone) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, nil, executeDutiesCall, timeout, expectedDuties[:1]) + waitForDutiesExecutionGenesis(t, logger, nil, executeDutiesCall, timeout, expectedDuties[:1]) require.EqualValues(t, 2, blockByNumberCalls.Load()) }) @@ -100,7 +100,7 @@ func TestVoluntaryExitHandler_HandleGenesisDuties(t *testing.T) { t.Run("slot = 5, block = 1 - executing another duty, no block number fetch", func(t *testing.T) { currentSlot.Set(phase0.Slot(sameBlockExit.BlockNumber) + voluntaryExitSlotsToPostpone) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, nil, executeDutiesCall, timeout, expectedDuties[1:2]) + waitForDutiesExecutionGenesis(t, logger, nil, executeDutiesCall, timeout, expectedDuties[1:2]) require.EqualValues(t, 2, blockByNumberCalls.Load()) }) @@ -116,7 +116,7 @@ func TestVoluntaryExitHandler_HandleGenesisDuties(t *testing.T) { t.Run("slot = 6, block = 1 - executing new duty, fetching block number", func(t *testing.T) { currentSlot.Set(phase0.Slot(newBlockExit.BlockNumber) + voluntaryExitSlotsToPostpone) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, nil, executeDutiesCall, timeout, expectedDuties[2:3]) + waitForDutiesExecutionGenesis(t, logger, nil, executeDutiesCall, timeout, expectedDuties[2:3]) require.EqualValues(t, 3, blockByNumberCalls.Load()) }) @@ -125,7 +125,7 @@ func TestVoluntaryExitHandler_HandleGenesisDuties(t *testing.T) { t.Run("slot = 10, block = 5 - executing past duty, fetching block number", func(t *testing.T) { currentSlot.Set(phase0.Slot(pastBlockExit.BlockNumber) + voluntaryExitSlotsToPostpone + 1) ticker.Send(currentSlot.Get()) - waitForGenesisDutiesExecution(t, logger, nil, executeDutiesCall, timeout, expectedDuties[3:4]) + waitForDutiesExecutionGenesis(t, logger, nil, executeDutiesCall, timeout, expectedDuties[3:4]) require.EqualValues(t, 4, blockByNumberCalls.Load()) })