2023-12-07 02:36:25 +00:00
|
|
|
package verification
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"sync"
|
|
|
|
|
2024-02-15 05:46:47 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/blockchain/kzg"
|
|
|
|
forkchoicetypes "github.com/prysmaticlabs/prysm/v5/beacon-chain/forkchoice/types"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/startup"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/state"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/consensus-types/blocks"
|
|
|
|
"github.com/prysmaticlabs/prysm/v5/consensus-types/primitives"
|
2024-03-08 18:20:38 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v5/network/forks"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1"
|
2023-12-07 02:36:25 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Forkchoicer represents the forkchoice methods that the verifiers need.
|
|
|
|
// Note that forkchoice is used here in a lock-free fashion, assuming that a version of forkchoice
|
|
|
|
// is given that internally handles the details of locking the underlying store.
|
|
|
|
type Forkchoicer interface {
|
|
|
|
FinalizedCheckpoint() *forkchoicetypes.Checkpoint
|
|
|
|
HasNode([32]byte) bool
|
|
|
|
IsCanonical(root [32]byte) bool
|
|
|
|
Slot([32]byte) (primitives.Slot, error)
|
2024-01-07 03:24:07 +00:00
|
|
|
TargetRootForEpoch([32]byte, primitives.Epoch) ([32]byte, error)
|
2023-12-07 02:36:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// StateByRooter describes a stategen-ish type that can produce arbitrary states by their root
|
|
|
|
type StateByRooter interface {
|
|
|
|
StateByRoot(ctx context.Context, blockRoot [32]byte) (state.BeaconState, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// sharedResources provides access to resources that are required by different verification types.
|
|
|
|
// for example, sidecar verification and block verification share the block signature verification cache.
|
|
|
|
type sharedResources struct {
|
|
|
|
clock *startup.Clock
|
|
|
|
fc Forkchoicer
|
|
|
|
sc SignatureCache
|
|
|
|
pc ProposerCache
|
|
|
|
sr StateByRooter
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initializer is used to create different Verifiers.
|
|
|
|
// Verifiers require access to stateful data structures, like caches,
|
2024-02-14 20:58:51 +00:00
|
|
|
// and it is Initializer's job to provide access to those.
|
2023-12-07 02:36:25 +00:00
|
|
|
type Initializer struct {
|
|
|
|
shared *sharedResources
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewBlobVerifier creates a BlobVerifier for a single blob, with the given set of requirements.
|
2024-01-06 23:47:09 +00:00
|
|
|
func (ini *Initializer) NewBlobVerifier(b blocks.ROBlob, reqs []Requirement) *ROBlobVerifier {
|
|
|
|
return &ROBlobVerifier{
|
2023-12-07 02:36:25 +00:00
|
|
|
sharedResources: ini.shared,
|
|
|
|
blob: b,
|
|
|
|
results: newResults(reqs...),
|
2024-01-06 23:47:09 +00:00
|
|
|
verifyBlobCommitment: kzg.Verify,
|
2023-12-07 02:36:25 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// InitializerWaiter provides an Initializer once all dependent resources are ready
|
|
|
|
// via the WaitForInitializer method.
|
|
|
|
type InitializerWaiter struct {
|
|
|
|
sync.RWMutex
|
2024-03-08 18:20:38 +00:00
|
|
|
ready bool
|
|
|
|
cw startup.ClockWaiter
|
|
|
|
ini *Initializer
|
|
|
|
getFork forkLookup
|
|
|
|
}
|
|
|
|
|
|
|
|
type forkLookup func(targetEpoch primitives.Epoch) (*ethpb.Fork, error)
|
|
|
|
|
|
|
|
type InitializerOption func(waiter *InitializerWaiter)
|
|
|
|
|
|
|
|
// WithForkLookup allows tests to modify how Fork consensus type lookup works. Needed for spectests with weird Forks.
|
|
|
|
func WithForkLookup(fl forkLookup) InitializerOption {
|
|
|
|
return func(iw *InitializerWaiter) {
|
|
|
|
iw.getFork = fl
|
|
|
|
}
|
2023-12-07 02:36:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewInitializerWaiter creates an InitializerWaiter which can be used to obtain an Initializer once async dependencies are ready.
|
2024-03-08 18:20:38 +00:00
|
|
|
func NewInitializerWaiter(cw startup.ClockWaiter, fc Forkchoicer, sr StateByRooter, opts ...InitializerOption) *InitializerWaiter {
|
2023-12-07 20:14:01 +00:00
|
|
|
pc := newPropCache()
|
|
|
|
// signature cache is initialized in WaitForInitializer, since we need the genesis validators root, which can be obtained from startup.Clock.
|
2023-12-07 02:36:25 +00:00
|
|
|
shared := &sharedResources{
|
|
|
|
fc: fc,
|
|
|
|
pc: pc,
|
|
|
|
sr: sr,
|
|
|
|
}
|
2024-03-08 18:20:38 +00:00
|
|
|
iw := &InitializerWaiter{cw: cw, ini: &Initializer{shared: shared}}
|
|
|
|
for _, o := range opts {
|
|
|
|
o(iw)
|
|
|
|
}
|
|
|
|
if iw.getFork == nil {
|
|
|
|
iw.getFork = forks.Fork
|
|
|
|
}
|
|
|
|
return iw
|
2023-12-07 02:36:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// WaitForInitializer ensures that asynchronous initialization of the shared resources the initializer
|
|
|
|
// depends on has completed before the underlying Initializer is accessible by client code.
|
|
|
|
func (w *InitializerWaiter) WaitForInitializer(ctx context.Context) (*Initializer, error) {
|
|
|
|
if err := w.waitForReady(ctx); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-12-07 20:14:01 +00:00
|
|
|
// We wait until this point to initialize the signature cache because here we have access to the genesis validator root.
|
|
|
|
vr := w.ini.shared.clock.GenesisValidatorsRoot()
|
2024-03-08 18:20:38 +00:00
|
|
|
sc := newSigCache(vr[:], DefaultSignatureCacheSize, w.getFork)
|
2023-12-07 20:14:01 +00:00
|
|
|
w.ini.shared.sc = sc
|
2023-12-07 02:36:25 +00:00
|
|
|
return w.ini, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *InitializerWaiter) waitForReady(ctx context.Context) error {
|
|
|
|
w.Lock()
|
|
|
|
defer w.Unlock()
|
|
|
|
if w.ready {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
clock, err := w.cw.WaitForClock(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
w.ini.shared.clock = clock
|
|
|
|
w.ready = true
|
|
|
|
return nil
|
|
|
|
}
|