2019-02-04 13:47:17 +00:00
|
|
|
package db
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"math/big"
|
2019-03-12 04:05:55 +00:00
|
|
|
"sort"
|
2019-02-04 13:47:17 +00:00
|
|
|
|
|
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
|
|
"github.com/prometheus/client_golang/prometheus/promauto"
|
2019-03-03 17:31:29 +00:00
|
|
|
pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1"
|
2019-02-04 13:47:17 +00:00
|
|
|
"github.com/sirupsen/logrus"
|
2019-02-28 03:55:47 +00:00
|
|
|
"go.opencensus.io/trace"
|
2019-02-04 13:47:17 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
2019-03-12 04:05:55 +00:00
|
|
|
pendingDepositsCount = promauto.NewGauge(prometheus.GaugeOpts{
|
2019-02-04 13:47:17 +00:00
|
|
|
Name: "beacondb_pending_deposits",
|
|
|
|
Help: "The number of pending deposits in the beaconDB in-memory database",
|
|
|
|
})
|
|
|
|
)
|
|
|
|
|
|
|
|
// Container object for holding the deposit and a reference to the block in
|
|
|
|
// which the deposit transaction was included in the proof of work chain.
|
|
|
|
type depositContainer struct {
|
|
|
|
deposit *pb.Deposit
|
|
|
|
block *big.Int
|
|
|
|
}
|
|
|
|
|
|
|
|
// InsertPendingDeposit into the database. If deposit or block number are nil
|
|
|
|
// then this method does nothing.
|
|
|
|
func (db *BeaconDB) InsertPendingDeposit(ctx context.Context, d *pb.Deposit, blockNum *big.Int) {
|
2019-02-28 03:55:47 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "BeaconDB.InsertPendingDeposit")
|
|
|
|
defer span.End()
|
2019-02-04 13:47:17 +00:00
|
|
|
if d == nil || blockNum == nil {
|
|
|
|
log.WithFields(logrus.Fields{
|
|
|
|
"block": blockNum,
|
|
|
|
"deposit": d,
|
|
|
|
}).Debug("Ignoring nil deposit insertion")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
db.depositsLock.Lock()
|
|
|
|
defer db.depositsLock.Unlock()
|
2019-03-12 04:05:55 +00:00
|
|
|
db.pendingDeposits = append(db.pendingDeposits, &depositContainer{deposit: d, block: blockNum})
|
|
|
|
pendingDepositsCount.Inc()
|
2019-02-04 13:47:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// PendingDeposits returns a list of deposits until the given block number
|
|
|
|
// (inclusive). If no block is specified then this method returns all pending
|
|
|
|
// deposits.
|
|
|
|
func (db *BeaconDB) PendingDeposits(ctx context.Context, beforeBlk *big.Int) []*pb.Deposit {
|
2019-02-28 03:55:47 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "BeaconDB.PendingDeposits")
|
|
|
|
defer span.End()
|
2019-02-04 13:47:17 +00:00
|
|
|
db.depositsLock.RLock()
|
|
|
|
defer db.depositsLock.RUnlock()
|
|
|
|
|
|
|
|
var deposits []*pb.Deposit
|
2019-03-12 04:05:55 +00:00
|
|
|
for _, ctnr := range db.pendingDeposits {
|
2019-02-05 17:25:09 +00:00
|
|
|
if beforeBlk == nil || beforeBlk.Cmp(ctnr.block) > -1 {
|
2019-02-04 13:47:17 +00:00
|
|
|
deposits = append(deposits, ctnr.deposit)
|
|
|
|
}
|
|
|
|
}
|
2019-03-12 04:05:55 +00:00
|
|
|
// Sort the deposits by Merkle index.
|
|
|
|
sort.SliceStable(deposits, func(i, j int) bool {
|
|
|
|
return deposits[i].MerkleTreeIndex < deposits[j].MerkleTreeIndex
|
|
|
|
})
|
2019-02-04 13:47:17 +00:00
|
|
|
return deposits
|
|
|
|
}
|
|
|
|
|
|
|
|
// RemovePendingDeposit from the database. The deposit is indexed by the
|
|
|
|
// MerkleTreeIndex. This method does nothing if deposit ptr is nil.
|
|
|
|
func (db *BeaconDB) RemovePendingDeposit(ctx context.Context, d *pb.Deposit) {
|
2019-02-28 03:55:47 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "BeaconDB.RemovePendingDeposit")
|
|
|
|
defer span.End()
|
2019-02-04 13:47:17 +00:00
|
|
|
|
|
|
|
if d == nil {
|
|
|
|
log.Debug("Ignoring nil deposit removal")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
db.depositsLock.Lock()
|
|
|
|
defer db.depositsLock.Unlock()
|
|
|
|
|
|
|
|
idx := -1
|
2019-03-12 04:05:55 +00:00
|
|
|
for i, ctnr := range db.pendingDeposits {
|
2019-02-04 13:47:17 +00:00
|
|
|
if ctnr.deposit.MerkleTreeIndex == d.MerkleTreeIndex {
|
|
|
|
idx = i
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if idx >= 0 {
|
2019-03-12 04:05:55 +00:00
|
|
|
db.pendingDeposits = append(db.pendingDeposits[:idx], db.pendingDeposits[idx+1:]...)
|
|
|
|
pendingDepositsCount.Dec()
|
2019-02-04 13:47:17 +00:00
|
|
|
}
|
|
|
|
}
|
2019-03-27 16:15:29 +00:00
|
|
|
|
2019-04-14 08:17:39 +00:00
|
|
|
// PrunePendingDeposits removes any deposit which is older than the given deposit merkle tree index.
|
|
|
|
func (db *BeaconDB) PrunePendingDeposits(ctx context.Context, merkleTreeIndex uint64) {
|
2019-03-27 16:15:29 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "BeaconDB.PrunePendingDeposits")
|
|
|
|
defer span.End()
|
|
|
|
|
2019-04-14 08:17:39 +00:00
|
|
|
if merkleTreeIndex == 0 {
|
|
|
|
log.Debug("Ignoring 0 deposit removal")
|
2019-03-27 16:15:29 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
db.depositsLock.Lock()
|
|
|
|
defer db.depositsLock.Unlock()
|
|
|
|
|
2019-04-14 08:17:39 +00:00
|
|
|
var cleanDeposits []*depositContainer
|
|
|
|
for _, dp := range db.pendingDeposits {
|
|
|
|
if dp.deposit.MerkleTreeIndex >= merkleTreeIndex {
|
|
|
|
cleanDeposits = append(cleanDeposits, dp)
|
2019-03-27 16:15:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-04-14 08:17:39 +00:00
|
|
|
db.pendingDeposits = cleanDeposits
|
2019-04-05 02:28:00 +00:00
|
|
|
pendingDepositsCount.Set(float64(len(db.pendingDeposits)))
|
2019-03-27 16:15:29 +00:00
|
|
|
}
|