2023-09-07 03:19:32 +00:00
|
|
|
package depositsnapshot
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"encoding/hex"
|
|
|
|
"sort"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
|
|
"github.com/pkg/errors"
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
|
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
2024-02-15 05:46:47 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v5/encoding/bytesutil"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1"
|
2023-09-07 03:19:32 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
"go.opencensus.io/trace"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
historicalDepositsCount = promauto.NewCounter(prometheus.CounterOpts{
|
|
|
|
Name: "beacondb_all_deposits_eip4881",
|
|
|
|
Help: "The number of total deposits in memory",
|
|
|
|
})
|
|
|
|
log = logrus.WithField("prefix", "cache")
|
|
|
|
)
|
|
|
|
|
|
|
|
// InsertDeposit into the database. If deposit or block number are nil
|
|
|
|
// then this method does nothing.
|
|
|
|
func (c *Cache) InsertDeposit(ctx context.Context, d *ethpb.Deposit, blockNum uint64, index int64, depositRoot [32]byte) error {
|
|
|
|
ctx, span := trace.StartSpan(ctx, "Cache.InsertDeposit")
|
|
|
|
defer span.End()
|
2023-12-15 16:49:27 +00:00
|
|
|
if ctx.Err() != nil {
|
|
|
|
return ctx.Err()
|
|
|
|
}
|
2023-09-07 03:19:32 +00:00
|
|
|
if d == nil {
|
|
|
|
log.WithFields(logrus.Fields{
|
2024-02-22 22:40:36 +00:00
|
|
|
"block": blockNum,
|
|
|
|
"deposit": d,
|
|
|
|
"index": index,
|
|
|
|
"depositRoot": hex.EncodeToString(depositRoot[:]),
|
2023-09-07 03:19:32 +00:00
|
|
|
}).Warn("Ignoring nil deposit insertion")
|
|
|
|
return errors.New("nil deposit inserted into the cache")
|
|
|
|
}
|
|
|
|
c.depositsLock.Lock()
|
|
|
|
defer c.depositsLock.Unlock()
|
|
|
|
|
|
|
|
if int(index) != len(c.deposits) {
|
|
|
|
return errors.Errorf("wanted deposit with index %d to be inserted but received %d", len(c.deposits), index)
|
|
|
|
}
|
|
|
|
// Keep the slice sorted on insertion in order to avoid costly sorting on retrieval.
|
|
|
|
heightIdx := sort.Search(len(c.deposits), func(i int) bool { return c.deposits[i].Index >= index })
|
|
|
|
depCtr := ðpb.DepositContainer{Deposit: d, Eth1BlockHeight: blockNum, DepositRoot: depositRoot[:], Index: index}
|
|
|
|
newDeposits := append(
|
|
|
|
[]*ethpb.DepositContainer{depCtr},
|
|
|
|
c.deposits[heightIdx:]...)
|
|
|
|
c.deposits = append(c.deposits[:heightIdx], newDeposits...)
|
|
|
|
// Append the deposit to our map, in the event no deposits
|
|
|
|
// exist for the pubkey , it is simply added to the map.
|
|
|
|
pubkey := bytesutil.ToBytes48(d.Data.PublicKey)
|
|
|
|
c.depositsByKey[pubkey] = append(c.depositsByKey[pubkey], depCtr)
|
|
|
|
historicalDepositsCount.Inc()
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// InsertDepositContainers inserts a set of deposit containers into our deposit cache.
|
|
|
|
func (c *Cache) InsertDepositContainers(ctx context.Context, ctrs []*ethpb.DepositContainer) {
|
2023-12-15 16:49:27 +00:00
|
|
|
_, span := trace.StartSpan(ctx, "Cache.InsertDepositContainers")
|
2023-09-07 03:19:32 +00:00
|
|
|
defer span.End()
|
|
|
|
c.depositsLock.Lock()
|
|
|
|
defer c.depositsLock.Unlock()
|
|
|
|
|
|
|
|
// Initialize slice if nil object provided.
|
|
|
|
if ctrs == nil {
|
|
|
|
ctrs = make([]*ethpb.DepositContainer, 0)
|
|
|
|
}
|
|
|
|
sort.SliceStable(ctrs, func(i int, j int) bool { return ctrs[i].Index < ctrs[j].Index })
|
|
|
|
c.deposits = ctrs
|
|
|
|
for _, ctr := range ctrs {
|
|
|
|
// Use a new value, as the reference
|
|
|
|
// changes in the next iteration.
|
|
|
|
newPtr := ctr
|
|
|
|
pKey := bytesutil.ToBytes48(newPtr.Deposit.Data.PublicKey)
|
|
|
|
c.depositsByKey[pKey] = append(c.depositsByKey[pKey], newPtr)
|
|
|
|
}
|
|
|
|
historicalDepositsCount.Add(float64(len(ctrs)))
|
|
|
|
}
|
|
|
|
|
|
|
|
// InsertFinalizedDeposits inserts deposits up to eth1DepositIndex (inclusive) into the finalized deposits cache.
|
|
|
|
func (c *Cache) InsertFinalizedDeposits(ctx context.Context, eth1DepositIndex int64,
|
|
|
|
executionHash common.Hash, executionNumber uint64) error {
|
|
|
|
ctx, span := trace.StartSpan(ctx, "Cache.InsertFinalizedDeposits")
|
|
|
|
defer span.End()
|
|
|
|
c.depositsLock.Lock()
|
|
|
|
defer c.depositsLock.Unlock()
|
|
|
|
|
2023-12-15 16:49:27 +00:00
|
|
|
if ctx.Err() != nil {
|
|
|
|
return ctx.Err()
|
|
|
|
}
|
|
|
|
|
2023-09-07 03:19:32 +00:00
|
|
|
depositTrie := c.finalizedDeposits.depositTree
|
|
|
|
insertIndex := int(c.finalizedDeposits.MerkleTrieIndex() + 1)
|
|
|
|
|
|
|
|
// Don't insert into finalized trie if there is no deposit to
|
|
|
|
// insert.
|
|
|
|
if len(c.deposits) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
// In the event we have less deposits than we need to
|
|
|
|
// finalize we finalize till the index on which we do have it.
|
|
|
|
if len(c.deposits) <= int(eth1DepositIndex) {
|
|
|
|
eth1DepositIndex = int64(len(c.deposits)) - 1
|
|
|
|
}
|
|
|
|
// If we finalize to some lower deposit index, we
|
|
|
|
// ignore it.
|
|
|
|
if int(eth1DepositIndex) < insertIndex {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
currIdx := int64(depositTrie.depositCount) - 1
|
|
|
|
|
|
|
|
// Insert deposits into deposit trie.
|
|
|
|
for _, ctr := range c.deposits {
|
|
|
|
if ctr.Index > currIdx && ctr.Index <= eth1DepositIndex {
|
|
|
|
rt, err := ctr.Deposit.Data.HashTreeRoot()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := depositTrie.Insert(rt[:], int(ctr.Index)); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := depositTrie.Finalize(eth1DepositIndex, executionHash, executionNumber); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
c.finalizedDeposits = finalizedDepositsContainer{
|
|
|
|
depositTree: depositTrie,
|
|
|
|
merkleTrieIndex: eth1DepositIndex,
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|