2023-05-19 10:19:48 +02:00
|
|
|
package transfer
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"math/big"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
|
|
"github.com/ethereum/go-ethereum/core/types"
|
|
|
|
"github.com/ethereum/go-ethereum/event"
|
|
|
|
"github.com/ethereum/go-ethereum/log"
|
|
|
|
"github.com/status-im/status-go/rpc/chain"
|
|
|
|
"github.com/status-im/status-go/services/wallet/async"
|
2023-06-02 17:08:45 -03:00
|
|
|
"github.com/status-im/status-go/services/wallet/token"
|
2023-05-19 10:19:48 +02:00
|
|
|
"github.com/status-im/status-go/services/wallet/walletevent"
|
|
|
|
)
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
type findNewBlocksCommand struct {
|
|
|
|
*findBlocksCommand
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *findNewBlocksCommand) Command() async.Command {
|
|
|
|
return async.InfiniteCommand{
|
|
|
|
Interval: 13 * time.Second, // TODO - make it configurable based on chain block mining time
|
|
|
|
Runable: c.Run,
|
|
|
|
}.Run
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *findNewBlocksCommand) Run(parent context.Context) (err error) {
|
|
|
|
log.Debug("start findNewBlocksCommand", "account", c.account, "chain", c.chainClient.ChainID, "noLimit", c.noLimit)
|
|
|
|
|
|
|
|
headNum, err := getHeadBlockNumber(parent, c.chainClient)
|
|
|
|
if err != nil {
|
|
|
|
// c.error = err
|
|
|
|
return err // Might need to retry a couple of times
|
|
|
|
}
|
|
|
|
|
|
|
|
blockRange, err := loadBlockRangeInfo(c.chainClient.ChainID, c.account, c.blockRangeDAO)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("findBlocksCommand loadBlockRangeInfo", "error", err)
|
|
|
|
// c.error = err
|
|
|
|
return err // Will keep spinning forever nomatter what
|
|
|
|
}
|
|
|
|
|
|
|
|
if blockRange != nil {
|
|
|
|
c.fromBlockNumber = new(big.Int).Add(blockRange.LastKnown, big.NewInt(1))
|
|
|
|
|
|
|
|
log.Debug("Launching new blocks command", "chainID", c.chainClient.ChainID, "account", c.account,
|
|
|
|
"from", c.fromBlockNumber, "headNum", headNum)
|
|
|
|
|
|
|
|
// In case interval between checks is set smaller than block mining time,
|
|
|
|
// we might need to wait for the next block to be mined
|
|
|
|
if c.fromBlockNumber.Cmp(headNum) > 0 {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
c.toBlockNumber = headNum
|
|
|
|
|
|
|
|
_ = c.findBlocksCommand.Run(parent)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-05-19 10:19:48 +02:00
|
|
|
// TODO NewFindBlocksCommand
|
|
|
|
type findBlocksCommand struct {
|
|
|
|
account common.Address
|
|
|
|
db *Database
|
2023-05-26 10:27:48 +02:00
|
|
|
blockRangeDAO *BlockRangeSequentialDAO
|
2023-05-19 10:19:48 +02:00
|
|
|
chainClient *chain.ClientWithFallback
|
|
|
|
balanceCache *balanceCache
|
|
|
|
feed *event.Feed
|
|
|
|
noLimit bool
|
|
|
|
transactionManager *TransactionManager
|
2023-05-26 10:27:48 +02:00
|
|
|
fromBlockNumber *big.Int
|
|
|
|
toBlockNumber *big.Int
|
2023-06-14 12:00:56 +02:00
|
|
|
blocksLoadedCh chan<- []*DBHeader
|
2023-05-26 10:27:48 +02:00
|
|
|
|
|
|
|
// Not to be set by the caller
|
|
|
|
resFromBlock *Block
|
|
|
|
startBlockNumber *big.Int
|
|
|
|
error error
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *findBlocksCommand) Command() async.Command {
|
|
|
|
return async.FiniteCommand{
|
|
|
|
Interval: 5 * time.Second,
|
|
|
|
Runable: c.Run,
|
|
|
|
}.Run
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *findBlocksCommand) Run(parent context.Context) (err error) {
|
2023-05-26 10:27:48 +02:00
|
|
|
log.Debug("start findBlocksCommand", "account", c.account, "chain", c.chainClient.ChainID, "noLimit", c.noLimit)
|
2023-05-19 10:19:48 +02:00
|
|
|
|
|
|
|
rangeSize := big.NewInt(DefaultNodeBlockChunkSize)
|
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
from, to := new(big.Int).Set(c.fromBlockNumber), new(big.Int).Set(c.toBlockNumber)
|
2023-06-01 15:09:50 +02:00
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
// Limit the range size to DefaultNodeBlockChunkSize
|
|
|
|
if new(big.Int).Sub(to, from).Cmp(rangeSize) > 0 {
|
2023-05-19 10:19:48 +02:00
|
|
|
from.Sub(to, rangeSize)
|
|
|
|
}
|
|
|
|
|
|
|
|
for {
|
|
|
|
headers, _ := c.checkRange(parent, from, to)
|
|
|
|
if c.error != nil {
|
2023-06-01 15:09:50 +02:00
|
|
|
log.Error("findBlocksCommand checkRange", "error", c.error, "account", c.account,
|
|
|
|
"chain", c.chainClient.ChainID, "from", from, "to", to)
|
2023-05-19 10:19:48 +02:00
|
|
|
break
|
|
|
|
}
|
|
|
|
|
2023-06-01 15:09:50 +02:00
|
|
|
if len(headers) > 0 {
|
|
|
|
log.Debug("findBlocksCommand saving headers", "len", len(headers), "lastBlockNumber", to,
|
|
|
|
"balance", c.balanceCache.ReadCachedBalance(c.account, to),
|
|
|
|
"nonce", c.balanceCache.ReadCachedNonce(c.account, to))
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-06-01 15:09:50 +02:00
|
|
|
err = c.db.SaveBlocks(c.chainClient.ChainID, c.account, headers)
|
|
|
|
if err != nil {
|
|
|
|
c.error = err
|
|
|
|
// return err
|
|
|
|
break
|
|
|
|
}
|
2023-06-14 12:00:56 +02:00
|
|
|
|
|
|
|
c.blocksFound(headers)
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
err = c.upsertBlockRange(&BlockRange{c.startBlockNumber, c.resFromBlock.Number, to})
|
|
|
|
if err != nil {
|
|
|
|
break
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
from, to = nextRange(c.resFromBlock.Number, c.fromBlockNumber)
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
if to.Cmp(c.fromBlockNumber) <= 0 || (c.startBlockNumber != nil &&
|
2023-05-19 10:19:48 +02:00
|
|
|
c.startBlockNumber.Cmp(big.NewInt(0)) > 0 && to.Cmp(c.startBlockNumber) <= 0) {
|
2023-05-26 10:27:48 +02:00
|
|
|
log.Debug("Checked all ranges, stop execution", "startBlock", c.startBlockNumber, "from", from, "to", to)
|
2023-05-19 10:19:48 +02:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
log.Debug("end findBlocksCommand", "account", c.account, "chain", c.chainClient.ChainID, "noLimit", c.noLimit)
|
2023-05-19 10:19:48 +02:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
func (c *findBlocksCommand) blocksFound(headers []*DBHeader) {
|
|
|
|
c.blocksLoadedCh <- headers
|
|
|
|
}
|
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
func (c *findBlocksCommand) upsertBlockRange(blockRange *BlockRange) error {
|
|
|
|
log.Debug("upsert block range", "Start", blockRange.Start, "FirstKnown", blockRange.FirstKnown, "LastKnown", blockRange.LastKnown,
|
|
|
|
"chain", c.chainClient.ChainID, "account", c.account)
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
err := c.blockRangeDAO.upsertRange(c.chainClient.ChainID, c.account, blockRange)
|
2023-05-19 10:19:48 +02:00
|
|
|
if err != nil {
|
|
|
|
c.error = err
|
2023-05-26 10:27:48 +02:00
|
|
|
log.Error("findBlocksCommand upsertRange", "error", err)
|
|
|
|
return err
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *findBlocksCommand) checkRange(parent context.Context, from *big.Int, to *big.Int) (
|
|
|
|
foundHeaders []*DBHeader, err error) {
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
fromBlock := &Block{Number: from}
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
newFromBlock, ethHeaders, startBlock, err := c.fastIndex(parent, c.balanceCache, fromBlock, to)
|
2023-05-19 10:19:48 +02:00
|
|
|
if err != nil {
|
2023-06-01 15:09:50 +02:00
|
|
|
log.Error("findBlocksCommand checkRange fastIndex", "err", err, "account", c.account,
|
|
|
|
"chain", c.chainClient.ChainID)
|
2023-05-19 10:19:48 +02:00
|
|
|
c.error = err
|
2023-05-26 10:27:48 +02:00
|
|
|
// return err // In case c.noLimit is true, hystrix "max concurrency" may be reached and we will not be able to index ETH transfers
|
2023-05-19 10:19:48 +02:00
|
|
|
return nil, nil
|
|
|
|
}
|
2023-06-14 12:00:56 +02:00
|
|
|
log.Debug("findBlocksCommand checkRange", "chainID", c.chainClient.ChainID, "account", c.account,
|
|
|
|
"startBlock", startBlock, "newFromBlock", newFromBlock.Number, "toBlockNumber", to, "noLimit", c.noLimit)
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-06-01 15:09:50 +02:00
|
|
|
// There could be incoming ERC20 transfers which don't change the balance
|
|
|
|
// and nonce of ETH account, so we keep looking for them
|
|
|
|
erc20Headers, err := c.fastIndexErc20(parent, newFromBlock.Number, to)
|
|
|
|
if err != nil {
|
2023-06-14 12:00:56 +02:00
|
|
|
log.Error("findBlocksCommand checkRange fastIndexErc20", "err", err, "account", c.account, "chain", c.chainClient.ChainID)
|
2023-06-01 15:09:50 +02:00
|
|
|
c.error = err
|
|
|
|
// return err
|
|
|
|
return nil, nil
|
|
|
|
}
|
2023-05-26 10:27:48 +02:00
|
|
|
|
2023-06-01 15:09:50 +02:00
|
|
|
allHeaders := append(ethHeaders, erc20Headers...)
|
2023-05-26 10:27:48 +02:00
|
|
|
|
2023-06-01 15:09:50 +02:00
|
|
|
if len(allHeaders) > 0 {
|
2023-06-02 17:08:45 -03:00
|
|
|
foundHeaders = uniqueHeaderPerBlockHash(allHeaders)
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
c.resFromBlock = newFromBlock
|
|
|
|
c.startBlockNumber = startBlock
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
log.Debug("end findBlocksCommand checkRange", "chainID", c.chainClient.ChainID, "account", c.account,
|
|
|
|
"c.startBlock", c.startBlockNumber, "newFromBlock", newFromBlock.Number,
|
2023-05-19 10:19:48 +02:00
|
|
|
"toBlockNumber", to, "c.resFromBlock", c.resFromBlock.Number)
|
|
|
|
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
func loadBlockRangeInfo(chainID uint64, account common.Address, blockDAO *BlockRangeSequentialDAO) (
|
|
|
|
*BlockRange, error) {
|
|
|
|
|
|
|
|
blockRange, err := blockDAO.getBlockRange(chainID, account)
|
2023-05-19 10:19:48 +02:00
|
|
|
if err != nil {
|
2023-05-26 10:27:48 +02:00
|
|
|
log.Error("failed to load block ranges from database", "chain", chainID, "account", account,
|
|
|
|
"error", err)
|
2023-05-19 10:19:48 +02:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
return blockRange, nil
|
|
|
|
}
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-06-01 15:09:50 +02:00
|
|
|
// Returns if all blocks are loaded, which means that start block (beginning of account history)
|
|
|
|
// has been found and all block headers saved to the DB
|
2023-05-26 10:27:48 +02:00
|
|
|
func areAllHistoryBlocksLoaded(blockInfo *BlockRange) bool {
|
|
|
|
if blockInfo == nil {
|
|
|
|
return false
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
if blockInfo.FirstKnown != nil && blockInfo.Start != nil &&
|
|
|
|
blockInfo.Start.Cmp(blockInfo.FirstKnown) >= 0 {
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-05-26 10:27:48 +02:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
|
|
|
|
2023-06-01 15:09:50 +02:00
|
|
|
func areAllHistoryBlocksLoadedForAddress(blockRangeDAO *BlockRangeSequentialDAO, chainID uint64,
|
|
|
|
address common.Address) (bool, error) {
|
|
|
|
|
|
|
|
blockRange, err := blockRangeDAO.getBlockRange(chainID, address)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("findBlocksCommand getBlockRange", "error", err)
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return areAllHistoryBlocksLoaded(blockRange), nil
|
|
|
|
}
|
|
|
|
|
2023-05-19 10:19:48 +02:00
|
|
|
// run fast indexing for every accont up to canonical chain head minus safety depth.
|
|
|
|
// every account will run it from last synced header.
|
|
|
|
func (c *findBlocksCommand) fastIndex(ctx context.Context, bCache *balanceCache,
|
|
|
|
fromBlock *Block, toBlockNumber *big.Int) (resultingFrom *Block, headers []*DBHeader,
|
|
|
|
startBlock *big.Int, err error) {
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
log.Debug("fast index started", "chainID", c.chainClient.ChainID, "account", c.account,
|
|
|
|
"from", fromBlock.Number, "to", toBlockNumber)
|
2023-05-19 10:19:48 +02:00
|
|
|
|
|
|
|
start := time.Now()
|
|
|
|
group := async.NewGroup(ctx)
|
|
|
|
|
|
|
|
command := ðHistoricalCommand{
|
|
|
|
chainClient: c.chainClient,
|
|
|
|
balanceCache: bCache,
|
|
|
|
address: c.account,
|
2023-06-01 15:09:50 +02:00
|
|
|
feed: c.feed,
|
|
|
|
from: fromBlock,
|
|
|
|
to: toBlockNumber,
|
|
|
|
noLimit: c.noLimit,
|
|
|
|
threadLimit: SequentialThreadLimit,
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
|
|
|
group.Add(command.Command())
|
|
|
|
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
err = ctx.Err()
|
|
|
|
log.Info("fast indexer ctx Done", "error", err)
|
|
|
|
return
|
|
|
|
case <-group.WaitAsync():
|
|
|
|
if command.error != nil {
|
|
|
|
err = command.error
|
|
|
|
return
|
|
|
|
}
|
|
|
|
resultingFrom = &Block{Number: command.resultingFrom}
|
|
|
|
headers = command.foundHeaders
|
|
|
|
startBlock = command.startBlock
|
2023-06-14 12:00:56 +02:00
|
|
|
log.Debug("fast indexer finished", "chainID", c.chainClient.ChainID, "account", c.account, "in", time.Since(start),
|
|
|
|
"startBlock", command.startBlock, "resultingFrom", resultingFrom.Number, "headers", len(headers))
|
2023-05-19 10:19:48 +02:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// run fast indexing for every accont up to canonical chain head minus safety depth.
|
|
|
|
// every account will run it from last synced header.
|
|
|
|
func (c *findBlocksCommand) fastIndexErc20(ctx context.Context, fromBlockNumber *big.Int,
|
|
|
|
toBlockNumber *big.Int) ([]*DBHeader, error) {
|
|
|
|
|
|
|
|
start := time.Now()
|
|
|
|
group := async.NewGroup(ctx)
|
|
|
|
|
|
|
|
erc20 := &erc20HistoricalCommand{
|
|
|
|
erc20: NewERC20TransfersDownloader(c.chainClient, []common.Address{c.account}, types.NewLondonSigner(c.chainClient.ToBigInt())),
|
|
|
|
chainClient: c.chainClient,
|
|
|
|
feed: c.feed,
|
|
|
|
address: c.account,
|
|
|
|
from: fromBlockNumber,
|
|
|
|
to: toBlockNumber,
|
|
|
|
foundHeaders: []*DBHeader{},
|
|
|
|
}
|
|
|
|
group.Add(erc20.Command())
|
|
|
|
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return nil, ctx.Err()
|
|
|
|
case <-group.WaitAsync():
|
|
|
|
headers := erc20.foundHeaders
|
2023-06-14 12:00:56 +02:00
|
|
|
log.Debug("fast indexer Erc20 finished", "chainID", c.chainClient.ChainID, "account", c.account,
|
|
|
|
"in", time.Since(start), "headers", len(headers))
|
2023-05-19 10:19:48 +02:00
|
|
|
return headers, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
func loadTransfersLoop(ctx context.Context, account common.Address, blockDAO *BlockDAO, db *Database,
|
|
|
|
chainClient *chain.ClientWithFallback, transactionManager *TransactionManager, tokenManager *token.Manager,
|
|
|
|
feed *event.Feed, blocksLoadedCh <-chan []*DBHeader) {
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
log.Debug("loadTransfersLoop start", "chain", chainClient.ChainID, "account", account)
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
log.Info("loadTransfersLoop error", "chain", chainClient.ChainID, "account", account, "error", ctx.Err())
|
|
|
|
return
|
|
|
|
case dbHeaders := <-blocksLoadedCh:
|
|
|
|
log.Debug("loadTransfersOnDemand transfers received", "chain", chainClient.ChainID, "account", account, "headers", len(dbHeaders))
|
|
|
|
|
|
|
|
blockNums := make([]*big.Int, len(dbHeaders))
|
|
|
|
for i, dbHeader := range dbHeaders {
|
|
|
|
blockNums[i] = dbHeader.Number
|
2023-06-01 15:09:50 +02:00
|
|
|
}
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
blocksByAddress := map[common.Address][]*big.Int{account: blockNums}
|
|
|
|
go func() {
|
|
|
|
_ = loadTransfers(ctx, []common.Address{account}, blockDAO, db, chainClient, noBlockLimit,
|
|
|
|
blocksByAddress, transactionManager, tokenManager, feed)
|
|
|
|
}()
|
2023-06-01 15:09:50 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
func newLoadBlocksAndTransfersCommand(account common.Address, db *Database,
|
2023-06-01 15:09:50 +02:00
|
|
|
blockDAO *BlockDAO, chainClient *chain.ClientWithFallback, feed *event.Feed,
|
2023-06-02 17:08:45 -03:00
|
|
|
transactionManager *TransactionManager, tokenManager *token.Manager) *loadBlocksAndTransfersCommand {
|
2023-06-01 15:09:50 +02:00
|
|
|
|
|
|
|
return &loadBlocksAndTransfersCommand{
|
2023-06-14 12:00:56 +02:00
|
|
|
account: account,
|
2023-06-01 15:09:50 +02:00
|
|
|
db: db,
|
|
|
|
blockRangeDAO: &BlockRangeSequentialDAO{db.client},
|
|
|
|
blockDAO: blockDAO,
|
|
|
|
chainClient: chainClient,
|
|
|
|
feed: feed,
|
|
|
|
errorsCount: 0,
|
|
|
|
transactionManager: transactionManager,
|
2023-06-02 17:08:45 -03:00
|
|
|
tokenManager: tokenManager,
|
2023-06-14 12:00:56 +02:00
|
|
|
blocksLoadedCh: make(chan []*DBHeader),
|
2023-06-01 15:09:50 +02:00
|
|
|
}
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
type loadBlocksAndTransfersCommand struct {
|
2023-06-14 12:00:56 +02:00
|
|
|
account common.Address
|
2023-05-19 10:19:48 +02:00
|
|
|
db *Database
|
|
|
|
blockRangeDAO *BlockRangeSequentialDAO
|
|
|
|
blockDAO *BlockDAO
|
|
|
|
chainClient *chain.ClientWithFallback
|
|
|
|
feed *event.Feed
|
|
|
|
balanceCache *balanceCache
|
|
|
|
errorsCount int
|
|
|
|
// nonArchivalRPCNode bool // TODO Make use of it
|
|
|
|
transactionManager *TransactionManager
|
2023-06-02 17:08:45 -03:00
|
|
|
tokenManager *token.Manager
|
2023-06-14 12:00:56 +02:00
|
|
|
blocksLoadedCh chan []*DBHeader
|
2023-06-01 15:09:50 +02:00
|
|
|
|
|
|
|
// Not to be set by the caller
|
2023-06-14 12:00:56 +02:00
|
|
|
transfersLoaded bool // For event RecentHistoryReady to be sent only once per account during app lifetime
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (c *loadBlocksAndTransfersCommand) Run(parent context.Context) error {
|
2023-06-14 12:00:56 +02:00
|
|
|
log.Debug("start load all transfers command", "chain", c.chainClient.ChainID, "account", c.account)
|
2023-05-19 10:19:48 +02:00
|
|
|
|
|
|
|
ctx := parent
|
|
|
|
|
|
|
|
if c.balanceCache == nil {
|
|
|
|
c.balanceCache = newBalanceCache() // TODO - need to keep balanceCache in memory??? What about sharing it with other packages?
|
|
|
|
}
|
|
|
|
|
|
|
|
group := async.NewGroup(ctx)
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
err := c.fetchTransfersForLoadedBlocks(group)
|
|
|
|
for err != nil {
|
|
|
|
return err
|
2023-05-26 10:27:48 +02:00
|
|
|
}
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
c.startTransfersLoop(ctx)
|
2023-05-26 10:27:48 +02:00
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
err = c.fetchHistoryBlocks(parent, group, c.blocksLoadedCh)
|
|
|
|
for err != nil {
|
|
|
|
group.Stop()
|
|
|
|
group.Wait()
|
|
|
|
return err
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
c.startFetchingNewBlocks(group, c.account, c.blocksLoadedCh)
|
2023-05-19 10:19:48 +02:00
|
|
|
|
|
|
|
select {
|
|
|
|
case <-ctx.Done():
|
|
|
|
return ctx.Err()
|
|
|
|
case <-group.WaitAsync():
|
2023-06-14 12:00:56 +02:00
|
|
|
log.Debug("end loadBlocksAndTransfers command", "chain", c.chainClient.ChainID, "account", c.account)
|
2023-05-19 10:19:48 +02:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *loadBlocksAndTransfersCommand) Command() async.Command {
|
|
|
|
return async.InfiniteCommand{
|
2023-06-14 12:00:56 +02:00
|
|
|
Interval: 5 * time.Second,
|
2023-05-19 10:19:48 +02:00
|
|
|
Runable: c.Run,
|
|
|
|
}.Run
|
|
|
|
}
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
func (c *loadBlocksAndTransfersCommand) startTransfersLoop(ctx context.Context) {
|
|
|
|
go loadTransfersLoop(ctx, c.account, c.blockDAO, c.db, c.chainClient, c.transactionManager, c.tokenManager,
|
|
|
|
c.feed, c.blocksLoadedCh)
|
|
|
|
}
|
2023-05-26 10:27:48 +02:00
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
func (c *loadBlocksAndTransfersCommand) fetchHistoryBlocks(ctx context.Context, group *async.Group, blocksLoadedCh chan []*DBHeader) error {
|
2023-05-26 10:27:48 +02:00
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
log.Debug("fetchHistoryBlocks start", "chainID", c.chainClient.ChainID, "account", c.account)
|
|
|
|
|
|
|
|
headNum, err := getHeadBlockNumber(ctx, c.chainClient)
|
|
|
|
if err != nil {
|
|
|
|
// c.error = err
|
|
|
|
return err // Might need to retry a couple of times
|
2023-05-26 10:27:48 +02:00
|
|
|
}
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
blockRange, err := loadBlockRangeInfo(c.chainClient.ChainID, c.account, c.blockRangeDAO)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("findBlocksCommand loadBlockRangeInfo", "error", err)
|
|
|
|
// c.error = err
|
|
|
|
return err // Will keep spinning forever nomatter what
|
|
|
|
}
|
2023-05-26 10:27:48 +02:00
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
allHistoryLoaded := areAllHistoryBlocksLoaded(blockRange)
|
|
|
|
to := getToHistoryBlockNumber(headNum, blockRange, allHistoryLoaded)
|
2023-05-26 10:27:48 +02:00
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
log.Debug("fetchHistoryBlocks", "chainID", c.chainClient.ChainID, "account", c.account, "to", to, "allHistoryLoaded", allHistoryLoaded)
|
2023-06-01 15:09:50 +02:00
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
if !allHistoryLoaded {
|
|
|
|
fbc := &findBlocksCommand{
|
|
|
|
account: c.account,
|
|
|
|
db: c.db,
|
|
|
|
blockRangeDAO: c.blockRangeDAO,
|
|
|
|
chainClient: c.chainClient,
|
|
|
|
balanceCache: c.balanceCache,
|
|
|
|
feed: c.feed,
|
|
|
|
noLimit: false,
|
|
|
|
fromBlockNumber: big.NewInt(0),
|
|
|
|
toBlockNumber: to,
|
|
|
|
transactionManager: c.transactionManager,
|
|
|
|
blocksLoadedCh: blocksLoadedCh,
|
|
|
|
}
|
|
|
|
group.Add(fbc.Command())
|
|
|
|
} else {
|
|
|
|
if !c.transfersLoaded {
|
|
|
|
transfersLoaded, err := c.areAllTransfersLoaded()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if transfersLoaded {
|
|
|
|
c.transfersLoaded = true
|
|
|
|
c.notifyHistoryReady()
|
|
|
|
}
|
|
|
|
}
|
2023-05-26 10:27:48 +02:00
|
|
|
}
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
log.Debug("fetchHistoryBlocks end", "chainID", c.chainClient.ChainID, "account", c.account)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *loadBlocksAndTransfersCommand) startFetchingNewBlocks(group *async.Group, address common.Address, blocksLoadedCh chan<- []*DBHeader) {
|
|
|
|
|
|
|
|
log.Debug("startFetchingNewBlocks", "chainID", c.chainClient.ChainID, "account", address)
|
|
|
|
|
|
|
|
newBlocksCmd := &findNewBlocksCommand{
|
|
|
|
findBlocksCommand: &findBlocksCommand{
|
|
|
|
account: address,
|
|
|
|
db: c.db,
|
|
|
|
blockRangeDAO: c.blockRangeDAO,
|
|
|
|
chainClient: c.chainClient,
|
|
|
|
balanceCache: c.balanceCache,
|
|
|
|
feed: c.feed,
|
|
|
|
noLimit: false,
|
|
|
|
transactionManager: c.transactionManager,
|
|
|
|
blocksLoadedCh: blocksLoadedCh,
|
|
|
|
},
|
2023-05-26 10:27:48 +02:00
|
|
|
}
|
|
|
|
group.Add(newBlocksCmd.Command())
|
|
|
|
}
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
func (c *loadBlocksAndTransfersCommand) fetchTransfersForLoadedBlocks(group *async.Group) error {
|
|
|
|
|
|
|
|
log.Debug("fetchTransfers start", "chainID", c.chainClient.ChainID, "account", c.account)
|
|
|
|
|
|
|
|
blocks, err := c.blockDAO.GetBlocksToLoadByAddress(c.chainClient.ChainID, c.account, numberOfBlocksCheckedPerIteration)
|
|
|
|
if err != nil {
|
|
|
|
log.Error("loadBlocksAndTransfersCommand GetBlocksToLoadByAddress", "error", err)
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
blocksMap := make(map[common.Address][]*big.Int)
|
|
|
|
blocksMap[c.account] = blocks
|
|
|
|
|
|
|
|
txCommand := &loadTransfersCommand{
|
|
|
|
accounts: []common.Address{c.account},
|
2023-05-26 10:27:48 +02:00
|
|
|
db: c.db,
|
|
|
|
blockDAO: c.blockDAO,
|
|
|
|
chainClient: c.chainClient,
|
|
|
|
transactionManager: c.transactionManager,
|
2023-06-14 12:00:56 +02:00
|
|
|
tokenManager: c.tokenManager,
|
|
|
|
blocksByAddress: blocksMap,
|
2023-06-01 15:09:50 +02:00
|
|
|
feed: c.feed,
|
2023-05-26 10:27:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
group.Add(txCommand.Command())
|
2023-06-14 12:00:56 +02:00
|
|
|
|
|
|
|
return nil
|
2023-05-26 10:27:48 +02:00
|
|
|
}
|
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
func (c *loadBlocksAndTransfersCommand) notifyHistoryReady() {
|
2023-06-01 15:09:50 +02:00
|
|
|
if c.feed != nil {
|
|
|
|
c.feed.Send(walletevent.Event{
|
|
|
|
Type: EventRecentHistoryReady,
|
2023-06-14 12:00:56 +02:00
|
|
|
Accounts: []common.Address{c.account},
|
2023-06-01 15:09:50 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
func (c *loadBlocksAndTransfersCommand) areAllTransfersLoaded() (bool, error) {
|
|
|
|
allBlocksLoaded, err := areAllHistoryBlocksLoadedForAddress(c.blockRangeDAO, c.chainClient.ChainID, c.account)
|
2023-06-01 15:09:50 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Error("loadBlockAndTransfersCommand allHistoryBlocksLoaded", "error", err)
|
|
|
|
return false, err
|
|
|
|
}
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-06-01 15:09:50 +02:00
|
|
|
if allBlocksLoaded {
|
2023-06-14 12:00:56 +02:00
|
|
|
firstHeader, err := c.blockDAO.GetFirstSavedBlock(c.chainClient.ChainID, c.account)
|
2023-06-01 15:09:50 +02:00
|
|
|
if err != nil {
|
|
|
|
log.Error("loadBlocksAndTransfersCommand GetFirstSavedBlock", "error", err)
|
|
|
|
return false, err
|
|
|
|
}
|
2023-05-19 10:19:48 +02:00
|
|
|
|
2023-06-01 15:09:50 +02:00
|
|
|
// If first block is Loaded, we have fetched all the transfers
|
|
|
|
if firstHeader != nil && firstHeader.Loaded {
|
|
|
|
return true, nil
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-06-01 15:09:50 +02:00
|
|
|
return false, nil
|
2023-05-19 10:19:48 +02:00
|
|
|
}
|
2023-05-26 10:27:48 +02:00
|
|
|
|
2023-06-14 12:00:56 +02:00
|
|
|
// TODO - make it a common method for every service that wants head block number, that will cache the latest block
|
|
|
|
// and updates it on timeout
|
2023-05-26 10:27:48 +02:00
|
|
|
func getHeadBlockNumber(parent context.Context, chainClient *chain.ClientWithFallback) (*big.Int, error) {
|
|
|
|
ctx, cancel := context.WithTimeout(parent, 3*time.Second)
|
|
|
|
head, err := chainClient.HeaderByNumber(ctx, nil)
|
|
|
|
cancel()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return head.Number, err
|
|
|
|
}
|
|
|
|
|
|
|
|
func nextRange(from *big.Int, zeroBlockNumber *big.Int) (*big.Int, *big.Int) {
|
|
|
|
log.Debug("next range start", "from", from, "zeroBlockNumber", zeroBlockNumber)
|
|
|
|
|
|
|
|
rangeSize := big.NewInt(DefaultNodeBlockChunkSize)
|
|
|
|
|
|
|
|
to := new(big.Int).Sub(from, big.NewInt(1)) // it won't hit the cache, but we wont load the transfers twice
|
|
|
|
if to.Cmp(rangeSize) > 0 {
|
|
|
|
from.Sub(to, rangeSize)
|
|
|
|
} else {
|
|
|
|
from = new(big.Int).Set(zeroBlockNumber)
|
|
|
|
}
|
|
|
|
|
|
|
|
log.Debug("next range end", "from", from, "to", to, "zeroBlockNumber", zeroBlockNumber)
|
|
|
|
|
|
|
|
return from, to
|
|
|
|
}
|
|
|
|
|
|
|
|
func getToHistoryBlockNumber(headNum *big.Int, blockRange *BlockRange, allHistoryLoaded bool) *big.Int {
|
|
|
|
var toBlockNum *big.Int
|
|
|
|
if blockRange != nil {
|
|
|
|
if !allHistoryLoaded {
|
|
|
|
toBlockNum = blockRange.FirstKnown
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
toBlockNum = headNum
|
|
|
|
}
|
|
|
|
|
|
|
|
return toBlockNum
|
|
|
|
}
|