mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-15 06:28:20 +00:00
221 lines
6.0 KiB
Go
221 lines
6.0 KiB
Go
package initialsync
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"sort"
|
|
"time"
|
|
|
|
"github.com/libp2p/go-libp2p/core/peer"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/blocks"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
prysmTime "github.com/prysmaticlabs/prysm/v4/time"
|
|
"github.com/prysmaticlabs/prysm/v4/time/slots"
|
|
)
|
|
|
|
const (
|
|
stateNew stateID = iota
|
|
stateScheduled
|
|
stateDataParsed
|
|
stateSkipped
|
|
stateSent
|
|
)
|
|
|
|
const (
|
|
eventTick eventID = iota
|
|
eventDataReceived
|
|
)
|
|
|
|
// stateID is unique handle for a state.
|
|
type stateID uint8
|
|
|
|
// eventID is unique handle for an event.
|
|
type eventID uint8
|
|
|
|
// stateMachineManager is a collection of managed FSMs.
|
|
type stateMachineManager struct {
|
|
keys []primitives.Slot
|
|
machines map[primitives.Slot]*stateMachine
|
|
handlers map[stateID]map[eventID]eventHandlerFn
|
|
}
|
|
|
|
// stateMachine holds a state of a single block processing FSM.
|
|
// Each FSM allows deterministic state transitions: State(S) x Event(E) -> Actions (A), State(S').
|
|
type stateMachine struct {
|
|
smm *stateMachineManager
|
|
start primitives.Slot
|
|
state stateID
|
|
pid peer.ID
|
|
bwb []blocks.BlockWithVerifiedBlobs
|
|
updated time.Time
|
|
}
|
|
|
|
// eventHandlerFn is an event handler function's signature.
|
|
type eventHandlerFn func(m *stateMachine, data interface{}) (newState stateID, err error)
|
|
|
|
// newStateMachineManager returns fully initialized state machine manager.
|
|
func newStateMachineManager() *stateMachineManager {
|
|
return &stateMachineManager{
|
|
keys: make([]primitives.Slot, 0, lookaheadSteps),
|
|
machines: make(map[primitives.Slot]*stateMachine, lookaheadSteps),
|
|
handlers: make(map[stateID]map[eventID]eventHandlerFn),
|
|
}
|
|
}
|
|
|
|
// addHandler attaches an event handler to a state event.
|
|
func (smm *stateMachineManager) addEventHandler(event eventID, state stateID, fn eventHandlerFn) {
|
|
if _, ok := smm.handlers[state]; !ok {
|
|
smm.handlers[state] = make(map[eventID]eventHandlerFn)
|
|
}
|
|
if _, ok := smm.handlers[state][event]; !ok {
|
|
smm.handlers[state][event] = fn
|
|
}
|
|
}
|
|
|
|
// addStateMachine allocates memory for new FSM.
|
|
func (smm *stateMachineManager) addStateMachine(startSlot primitives.Slot) *stateMachine {
|
|
smm.machines[startSlot] = &stateMachine{
|
|
smm: smm,
|
|
start: startSlot,
|
|
state: stateNew,
|
|
bwb: []blocks.BlockWithVerifiedBlobs{},
|
|
updated: prysmTime.Now(),
|
|
}
|
|
smm.recalculateMachineAttribs()
|
|
return smm.machines[startSlot]
|
|
}
|
|
|
|
// removeStateMachine frees memory of a processed/finished FSM.
|
|
func (smm *stateMachineManager) removeStateMachine(startSlot primitives.Slot) error {
|
|
if _, ok := smm.machines[startSlot]; !ok {
|
|
return fmt.Errorf("state for machine %v is not found", startSlot)
|
|
}
|
|
smm.machines[startSlot].bwb = nil
|
|
delete(smm.machines, startSlot)
|
|
smm.recalculateMachineAttribs()
|
|
return nil
|
|
}
|
|
|
|
// removeAllStateMachines removes all managed machines.
|
|
func (smm *stateMachineManager) removeAllStateMachines() error {
|
|
for _, key := range smm.keys {
|
|
if err := smm.removeStateMachine(key); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
smm.recalculateMachineAttribs()
|
|
return nil
|
|
}
|
|
|
|
// recalculateMachineAttribs updates cached attributes, which are used for efficiency.
|
|
func (smm *stateMachineManager) recalculateMachineAttribs() {
|
|
keys := make([]primitives.Slot, 0, lookaheadSteps)
|
|
for key := range smm.machines {
|
|
keys = append(keys, key)
|
|
}
|
|
sort.Slice(keys, func(i, j int) bool {
|
|
return keys[i] < keys[j]
|
|
})
|
|
smm.keys = keys
|
|
}
|
|
|
|
// findStateMachine returns a state machine for a given start slot (if exists).
|
|
func (smm *stateMachineManager) findStateMachine(startSlot primitives.Slot) (*stateMachine, bool) {
|
|
fsm, ok := smm.machines[startSlot]
|
|
return fsm, ok
|
|
}
|
|
|
|
// highestStartSlot returns the start slot for the latest known state machine.
|
|
func (smm *stateMachineManager) highestStartSlot() (primitives.Slot, error) {
|
|
if len(smm.keys) == 0 {
|
|
return 0, errors.New("no state machine exist")
|
|
}
|
|
key := smm.keys[len(smm.keys)-1]
|
|
return smm.machines[key].start, nil
|
|
}
|
|
|
|
// allMachinesInState checks whether all registered state machines are in the same state.
|
|
func (smm *stateMachineManager) allMachinesInState(state stateID) bool {
|
|
if len(smm.machines) == 0 {
|
|
return false
|
|
}
|
|
for _, fsm := range smm.machines {
|
|
if fsm.state != state {
|
|
return false
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
// String returns human readable representation of a FSM collection.
|
|
func (smm *stateMachineManager) String() string {
|
|
return fmt.Sprintf("%v", smm.machines)
|
|
}
|
|
|
|
// setState updates the current state of a given state machine.
|
|
func (m *stateMachine) setState(name stateID) {
|
|
if m.state == name {
|
|
return
|
|
}
|
|
m.state = name
|
|
m.updated = prysmTime.Now()
|
|
}
|
|
|
|
// trigger invokes the event handler on a given state machine.
|
|
func (m *stateMachine) trigger(event eventID, data interface{}) error {
|
|
handlers, ok := m.smm.handlers[m.state]
|
|
if !ok {
|
|
return fmt.Errorf("no event handlers registered for event: %v, state: %v", event, m.state)
|
|
}
|
|
if handlerFn, ok := handlers[event]; ok {
|
|
state, err := handlerFn(m, data)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
m.setState(state)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// isFirst checks whether a given machine has the lowest start slot.
|
|
func (m *stateMachine) isFirst() bool {
|
|
return m.start == m.smm.keys[0]
|
|
}
|
|
|
|
// isLast checks whether a given machine has the highest start slot.
|
|
func (m *stateMachine) isLast() bool {
|
|
return m.start == m.smm.keys[len(m.smm.keys)-1]
|
|
}
|
|
|
|
// String returns human-readable representation of a FSM state.
|
|
func (m *stateMachine) String() string {
|
|
return fmt.Sprintf("{%d:%s}", slots.ToEpoch(m.start), m.state)
|
|
}
|
|
|
|
// String returns human-readable representation of a state.
|
|
func (s stateID) String() string {
|
|
states := map[stateID]string{
|
|
stateNew: "new",
|
|
stateScheduled: "scheduled",
|
|
stateDataParsed: "dataParsed",
|
|
stateSkipped: "skipped",
|
|
stateSent: "sent",
|
|
}
|
|
if _, ok := states[s]; !ok {
|
|
return "stateUnknown"
|
|
}
|
|
return states[s]
|
|
}
|
|
|
|
// String returns human-readable representation of an event.
|
|
func (e eventID) String() string {
|
|
events := map[eventID]string{
|
|
eventTick: "tick",
|
|
eventDataReceived: "dataReceived",
|
|
}
|
|
if _, ok := events[e]; !ok {
|
|
return "eventUnknown"
|
|
}
|
|
return events[e]
|
|
}
|