mirror of
https://github.com/ethereum/go-ethereum.git
synced 2026-05-23 08:19:27 +00:00
This PR implements the serving side of the eth71 BAL exchange messages.
Until commit 4cd7092 also contained the requesting side, but since that
part still needs more work, I'm splitting it out into a separate PR.
The test injects BALs directly into rawdb. This can be removed once BAL
generation is integrated into the chain maker.
---------
Co-authored-by: Felix Lange <fjl@twurst.com>
566 lines
20 KiB
Go
566 lines
20 KiB
Go
// Copyright 2021 The go-ethereum Authors
|
|
// This file is part of the go-ethereum library.
|
|
//
|
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Lesser General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// The go-ethereum library is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public License
|
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
package core
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"math/big"
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
"github.com/ethereum/go-ethereum/consensus"
|
|
"github.com/ethereum/go-ethereum/consensus/misc/eip4844"
|
|
"github.com/ethereum/go-ethereum/core/rawdb"
|
|
"github.com/ethereum/go-ethereum/core/state"
|
|
"github.com/ethereum/go-ethereum/core/state/snapshot"
|
|
"github.com/ethereum/go-ethereum/core/types"
|
|
"github.com/ethereum/go-ethereum/core/vm"
|
|
"github.com/ethereum/go-ethereum/event"
|
|
"github.com/ethereum/go-ethereum/params"
|
|
"github.com/ethereum/go-ethereum/rlp"
|
|
"github.com/ethereum/go-ethereum/triedb"
|
|
)
|
|
|
|
// CurrentHeader retrieves the current head header of the canonical chain. The
|
|
// header is retrieved from the HeaderChain's internal cache.
|
|
func (bc *BlockChain) CurrentHeader() *types.Header {
|
|
return bc.hc.CurrentHeader()
|
|
}
|
|
|
|
// CurrentBlock retrieves the current head block of the canonical chain. The
|
|
// block is retrieved from the blockchain's internal cache.
|
|
func (bc *BlockChain) CurrentBlock() *types.Header {
|
|
return bc.currentBlock.Load()
|
|
}
|
|
|
|
// CurrentSnapBlock retrieves the current snap-sync head block of the canonical
|
|
// chain. The block is retrieved from the blockchain's internal cache.
|
|
func (bc *BlockChain) CurrentSnapBlock() *types.Header {
|
|
return bc.currentSnapBlock.Load()
|
|
}
|
|
|
|
// CurrentFinalBlock retrieves the current finalized block of the canonical
|
|
// chain. The block is retrieved from the blockchain's internal cache.
|
|
func (bc *BlockChain) CurrentFinalBlock() *types.Header {
|
|
return bc.currentFinalBlock.Load()
|
|
}
|
|
|
|
// CurrentSafeBlock retrieves the current safe block of the canonical
|
|
// chain. The block is retrieved from the blockchain's internal cache.
|
|
func (bc *BlockChain) CurrentSafeBlock() *types.Header {
|
|
return bc.currentSafeBlock.Load()
|
|
}
|
|
|
|
// HasHeader checks if a block header is present in the database or not, caching
|
|
// it if present.
|
|
func (bc *BlockChain) HasHeader(hash common.Hash, number uint64) bool {
|
|
return bc.hc.HasHeader(hash, number)
|
|
}
|
|
|
|
// GetHeader retrieves a block header from the database by hash and number,
|
|
// caching it if found.
|
|
func (bc *BlockChain) GetHeader(hash common.Hash, number uint64) *types.Header {
|
|
return bc.hc.GetHeader(hash, number)
|
|
}
|
|
|
|
// GetHeaderByHash retrieves a block header from the database by hash, caching it if
|
|
// found.
|
|
func (bc *BlockChain) GetHeaderByHash(hash common.Hash) *types.Header {
|
|
return bc.hc.GetHeaderByHash(hash)
|
|
}
|
|
|
|
// GetHeaderByNumber retrieves a block header from the database by number,
|
|
// caching it (associated with its hash) if found.
|
|
func (bc *BlockChain) GetHeaderByNumber(number uint64) *types.Header {
|
|
return bc.hc.GetHeaderByNumber(number)
|
|
}
|
|
|
|
// GetBlockNumber retrieves the block number associated with a block hash.
|
|
func (bc *BlockChain) GetBlockNumber(hash common.Hash) *uint64 {
|
|
if num, ok := bc.hc.GetBlockNumber(hash); ok {
|
|
return &num
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// GetHeadersFrom returns a contiguous segment of headers, in rlp-form, going
|
|
// backwards from the given number.
|
|
func (bc *BlockChain) GetHeadersFrom(number, count uint64) []rlp.RawValue {
|
|
return bc.hc.GetHeadersFrom(number, count)
|
|
}
|
|
|
|
// GetBody retrieves a block body (transactions and uncles) from the database by
|
|
// hash, caching it if found.
|
|
func (bc *BlockChain) GetBody(hash common.Hash) *types.Body {
|
|
// Short circuit if the body's already in the cache, retrieve otherwise
|
|
if cached, ok := bc.bodyCache.Get(hash); ok {
|
|
return cached
|
|
}
|
|
number, ok := bc.hc.GetBlockNumber(hash)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
body := rawdb.ReadBody(bc.db, hash, number)
|
|
if body == nil {
|
|
return nil
|
|
}
|
|
// Cache the found body for next time and return
|
|
bc.bodyCache.Add(hash, body)
|
|
return body
|
|
}
|
|
|
|
// GetBodyRLP retrieves a block body in RLP encoding from the database by hash,
|
|
// caching it if found.
|
|
func (bc *BlockChain) GetBodyRLP(hash common.Hash) rlp.RawValue {
|
|
// Short circuit if the body's already in the cache, retrieve otherwise
|
|
if cached, ok := bc.bodyRLPCache.Get(hash); ok {
|
|
return cached
|
|
}
|
|
number, ok := bc.hc.GetBlockNumber(hash)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
body := rawdb.ReadBodyRLP(bc.db, hash, number)
|
|
if len(body) == 0 {
|
|
return nil
|
|
}
|
|
// Cache the found body for next time and return
|
|
bc.bodyRLPCache.Add(hash, body)
|
|
return body
|
|
}
|
|
|
|
// HasBlock checks if a block is fully present in the database or not.
|
|
func (bc *BlockChain) HasBlock(hash common.Hash, number uint64) bool {
|
|
if bc.blockCache.Contains(hash) {
|
|
return true
|
|
}
|
|
if !bc.HasHeader(hash, number) {
|
|
return false
|
|
}
|
|
return rawdb.HasBody(bc.db, hash, number)
|
|
}
|
|
|
|
// HasFastBlock checks if a fast block is fully present in the database or not.
|
|
func (bc *BlockChain) HasFastBlock(hash common.Hash, number uint64) bool {
|
|
if !bc.HasBlock(hash, number) {
|
|
return false
|
|
}
|
|
if bc.receiptsCache.Contains(hash) {
|
|
return true
|
|
}
|
|
return rawdb.HasReceipts(bc.db, hash, number)
|
|
}
|
|
|
|
// GetBlock retrieves a block from the database by hash and number,
|
|
// caching it if found.
|
|
func (bc *BlockChain) GetBlock(hash common.Hash, number uint64) *types.Block {
|
|
// Short circuit if the block's already in the cache, retrieve otherwise
|
|
if block, ok := bc.blockCache.Get(hash); ok {
|
|
return block
|
|
}
|
|
block := rawdb.ReadBlock(bc.db, hash, number)
|
|
if block == nil {
|
|
return nil
|
|
}
|
|
// Cache the found block for next time and return
|
|
bc.blockCache.Add(block.Hash(), block)
|
|
return block
|
|
}
|
|
|
|
// GetBlockByHash retrieves a block from the database by hash, caching it if found.
|
|
func (bc *BlockChain) GetBlockByHash(hash common.Hash) *types.Block {
|
|
number, ok := bc.hc.GetBlockNumber(hash)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
return bc.GetBlock(hash, number)
|
|
}
|
|
|
|
// GetBlockByNumber retrieves a block from the database by number, caching it
|
|
// (associated with its hash) if found.
|
|
func (bc *BlockChain) GetBlockByNumber(number uint64) *types.Block {
|
|
hash := rawdb.ReadCanonicalHash(bc.db, number)
|
|
if hash == (common.Hash{}) {
|
|
return nil
|
|
}
|
|
return bc.GetBlock(hash, number)
|
|
}
|
|
|
|
// GetBlocksFromHash returns the block corresponding to hash and up to n-1 ancestors.
|
|
// [deprecated by eth/62]
|
|
func (bc *BlockChain) GetBlocksFromHash(hash common.Hash, n int) (blocks []*types.Block) {
|
|
number, ok := bc.hc.GetBlockNumber(hash)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
for i := 0; i < n; i++ {
|
|
block := bc.GetBlock(hash, number)
|
|
if block == nil {
|
|
break
|
|
}
|
|
blocks = append(blocks, block)
|
|
hash = block.ParentHash()
|
|
number--
|
|
}
|
|
return
|
|
}
|
|
|
|
// GetCanonicalReceipt allows fetching a receipt for a transaction that was
|
|
// already looked up on the index. Notably, only receipt in canonical chain
|
|
// is visible.
|
|
func (bc *BlockChain) GetCanonicalReceipt(tx *types.Transaction, blockHash common.Hash, blockNumber, txIndex uint64) (*types.Receipt, error) {
|
|
// The receipt retrieved from the cache contains all previously derived fields
|
|
if receipts, ok := bc.receiptsCache.Get(blockHash); ok {
|
|
if int(txIndex) >= len(receipts) {
|
|
return nil, fmt.Errorf("receipt out of index, length: %d, index: %d", len(receipts), txIndex)
|
|
}
|
|
return receipts[int(txIndex)], nil
|
|
}
|
|
header := bc.GetHeader(blockHash, blockNumber)
|
|
if header == nil {
|
|
return nil, fmt.Errorf("block header is not found, %d, %x", blockNumber, blockHash)
|
|
}
|
|
var blobGasPrice *big.Int
|
|
if header.ExcessBlobGas != nil {
|
|
blobGasPrice = eip4844.CalcBlobFee(bc.chainConfig, header)
|
|
}
|
|
receipt, ctx, err := rawdb.ReadCanonicalRawReceipt(bc.db, blockHash, blockNumber, txIndex)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
signer := types.MakeSigner(bc.chainConfig, new(big.Int).SetUint64(blockNumber), header.Time)
|
|
receipt.DeriveFields(signer, types.DeriveReceiptContext{
|
|
BlockHash: blockHash,
|
|
BlockNumber: blockNumber,
|
|
BlockTime: header.Time,
|
|
BaseFee: header.BaseFee,
|
|
BlobGasPrice: blobGasPrice,
|
|
GasUsed: ctx.GasUsed,
|
|
LogIndex: ctx.LogIndex,
|
|
Tx: tx,
|
|
TxIndex: uint(txIndex),
|
|
})
|
|
return receipt, nil
|
|
}
|
|
|
|
// GetReceiptsByHash retrieves the receipts for all transactions in a given block.
|
|
func (bc *BlockChain) GetReceiptsByHash(hash common.Hash) types.Receipts {
|
|
if receipts, ok := bc.receiptsCache.Get(hash); ok {
|
|
return receipts
|
|
}
|
|
number, ok := rawdb.ReadHeaderNumber(bc.db, hash)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
header := bc.GetHeader(hash, number)
|
|
if header == nil {
|
|
return nil
|
|
}
|
|
receipts := rawdb.ReadReceipts(bc.db, hash, number, header.Time, bc.chainConfig)
|
|
if receipts == nil {
|
|
return nil
|
|
}
|
|
bc.receiptsCache.Add(hash, receipts)
|
|
return receipts
|
|
}
|
|
|
|
// GetRawReceipts retrieves the receipts for all transactions in a given block
|
|
// without deriving the internal fields and the Bloom.
|
|
func (bc *BlockChain) GetRawReceipts(hash common.Hash, number uint64) types.Receipts {
|
|
if receipts, ok := bc.receiptsCache.Get(hash); ok {
|
|
return receipts
|
|
}
|
|
return rawdb.ReadRawReceipts(bc.db, hash, number)
|
|
}
|
|
|
|
// GetReceiptsRLP retrieves the receipts of a block.
|
|
func (bc *BlockChain) GetReceiptsRLP(hash common.Hash) rlp.RawValue {
|
|
number, ok := rawdb.ReadHeaderNumber(bc.db, hash)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
return rawdb.ReadReceiptsRLP(bc.db, hash, number)
|
|
}
|
|
|
|
// GetAccessListRLP retrieves the block access list of a block in RLP encoding.
|
|
func (bc *BlockChain) GetAccessListRLP(hash common.Hash) rlp.RawValue {
|
|
number, ok := rawdb.ReadHeaderNumber(bc.db, hash)
|
|
if !ok {
|
|
return nil
|
|
}
|
|
return rawdb.ReadAccessListRLP(bc.db, hash, number)
|
|
}
|
|
|
|
// GetUnclesInChain retrieves all the uncles from a given block backwards until
|
|
// a specific distance is reached.
|
|
func (bc *BlockChain) GetUnclesInChain(block *types.Block, length int) []*types.Header {
|
|
uncles := []*types.Header{}
|
|
for i := 0; block != nil && i < length; i++ {
|
|
uncles = append(uncles, block.Uncles()...)
|
|
block = bc.GetBlock(block.ParentHash(), block.NumberU64()-1)
|
|
}
|
|
return uncles
|
|
}
|
|
|
|
// GetCanonicalHash returns the canonical hash for a given block number
|
|
func (bc *BlockChain) GetCanonicalHash(number uint64) common.Hash {
|
|
return bc.hc.GetCanonicalHash(number)
|
|
}
|
|
|
|
// GetAncestor retrieves the Nth ancestor of a given block. It assumes that either the given block or
|
|
// a close ancestor of it is canonical. maxNonCanonical points to a downwards counter limiting the
|
|
// number of blocks to be individually checked before we reach the canonical chain.
|
|
//
|
|
// Note: ancestor == 0 returns the same block, 1 returns its parent and so on.
|
|
func (bc *BlockChain) GetAncestor(hash common.Hash, number, ancestor uint64, maxNonCanonical *uint64) (common.Hash, uint64) {
|
|
return bc.hc.GetAncestor(hash, number, ancestor, maxNonCanonical)
|
|
}
|
|
|
|
// GetCanonicalTransaction retrieves the lookup along with the transaction
|
|
// itself associate with the given transaction hash.
|
|
//
|
|
// A null will be returned if the transaction is not found. This can be due to
|
|
// the transaction indexer not being finished. The caller must explicitly check
|
|
// the indexer progress.
|
|
//
|
|
// Notably, only the transaction in the canonical chain is visible.
|
|
func (bc *BlockChain) GetCanonicalTransaction(hash common.Hash) (*rawdb.LegacyTxLookupEntry, *types.Transaction) {
|
|
bc.txLookupLock.RLock()
|
|
defer bc.txLookupLock.RUnlock()
|
|
|
|
// Short circuit if the txlookup already in the cache, retrieve otherwise
|
|
if item, exist := bc.txLookupCache.Get(hash); exist {
|
|
return item.lookup, item.transaction
|
|
}
|
|
tx, blockHash, blockNumber, txIndex := rawdb.ReadCanonicalTransaction(bc.db, hash)
|
|
if tx == nil {
|
|
return nil, nil
|
|
}
|
|
lookup := &rawdb.LegacyTxLookupEntry{
|
|
BlockHash: blockHash,
|
|
BlockIndex: blockNumber,
|
|
Index: txIndex,
|
|
}
|
|
bc.txLookupCache.Add(hash, txLookup{
|
|
lookup: lookup,
|
|
transaction: tx,
|
|
})
|
|
return lookup, tx
|
|
}
|
|
|
|
// TxIndexDone returns true if the transaction indexer has finished indexing.
|
|
func (bc *BlockChain) TxIndexDone() bool {
|
|
progress, err := bc.TxIndexProgress()
|
|
if err != nil {
|
|
// No error is returned if the transaction indexing progress is unreachable
|
|
// due to unexpected internal errors. In such cases, it is impossible to
|
|
// determine whether the transaction does not exist or has simply not been
|
|
// indexed yet without a progress marker.
|
|
//
|
|
// In such scenarios, the transaction is treated as unreachable, though
|
|
// this is clearly an unintended and unexpected situation.
|
|
return true
|
|
}
|
|
return progress.Done()
|
|
}
|
|
|
|
// HasState checks if state trie is fully present in the database or not.
|
|
func (bc *BlockChain) HasState(hash common.Hash) bool {
|
|
_, err := bc.triedb.NodeReader(hash)
|
|
return err == nil
|
|
}
|
|
|
|
// HasBlockAndState checks if a block and associated state trie is fully present
|
|
// in the database or not, caching it if present.
|
|
func (bc *BlockChain) HasBlockAndState(hash common.Hash, number uint64) bool {
|
|
// Check first that the block itself is known
|
|
block := bc.GetBlock(hash, number)
|
|
if block == nil {
|
|
return false
|
|
}
|
|
return bc.HasState(block.Root())
|
|
}
|
|
|
|
// stateRecoverable checks if the specified state is recoverable.
|
|
// Note, this function assumes the state is not present, because
|
|
// state is not treated as recoverable if it's available, thus
|
|
// false will be returned in this case.
|
|
func (bc *BlockChain) stateRecoverable(root common.Hash) bool {
|
|
if bc.triedb.Scheme() == rawdb.HashScheme {
|
|
return false
|
|
}
|
|
result, _ := bc.triedb.Recoverable(root)
|
|
return result
|
|
}
|
|
|
|
// ContractCodeWithPrefix retrieves a blob of data associated with a contract
|
|
// hash either from ephemeral in-memory cache, or from persistent storage.
|
|
func (bc *BlockChain) ContractCodeWithPrefix(hash common.Hash) []byte {
|
|
// TODO(rjl493456442) The associated account address is also required
|
|
// in Verkle scheme. Fix it once snap-sync is supported for Verkle.
|
|
return bc.codedb.Reader().CodeWithPrefix(common.Address{}, hash)
|
|
}
|
|
|
|
// State returns a new mutable state based on the current HEAD block.
|
|
func (bc *BlockChain) State() (*state.StateDB, error) {
|
|
return bc.StateAt(bc.CurrentBlock())
|
|
}
|
|
|
|
// StateAt returns a new mutable state based on a particular point in time.
|
|
func (bc *BlockChain) StateAt(header *types.Header) (*state.StateDB, error) {
|
|
if bc.chainConfig.IsUBT(header.Number, header.Time) {
|
|
return state.New(header.Root, state.NewUBTDatabase(bc.triedb, bc.codedb))
|
|
}
|
|
return state.New(header.Root, state.NewMPTDatabase(bc.triedb, bc.codedb).WithSnapshot(bc.snaps))
|
|
}
|
|
|
|
// StateAtForkBoundary returns a new mutable state based on the parent state
|
|
// and the given header, handling the transition across the UBT fork.
|
|
func (bc *BlockChain) StateAtForkBoundary(parent *types.Header, header *types.Header) (*state.StateDB, error) {
|
|
// The parent is already in the UBT fork.
|
|
if bc.chainConfig.IsUBT(parent.Number, parent.Time) {
|
|
return state.New(parent.Root, state.NewUBTDatabase(bc.triedb, bc.codedb))
|
|
}
|
|
// The current block is the first block in the UBT fork
|
|
// (i.e., the parent is the last MPT block).
|
|
if bc.chainConfig.IsUBT(header.Number, header.Time) {
|
|
// TODO(gballet): register chain context if needed
|
|
return state.New(parent.Root, state.NewUBTDatabase(bc.triedb, bc.codedb))
|
|
}
|
|
// Both the parent and current block are in the MPT fork.
|
|
return state.New(parent.Root, state.NewMPTDatabase(bc.triedb, bc.codedb).WithSnapshot(bc.snaps))
|
|
}
|
|
|
|
// HistoricState returns a historic state specified by the given header.
|
|
// Live states are not available and won't be served, please use `State`
|
|
// or `StateAt` instead.
|
|
func (bc *BlockChain) HistoricState(header *types.Header) (*state.StateDB, error) {
|
|
if bc.chainConfig.IsUBT(header.Number, header.Time) {
|
|
return nil, errors.New("historical state over ubt is not yet supported")
|
|
}
|
|
return state.New(header.Root, state.NewHistoricDatabase(bc.triedb, bc.codedb))
|
|
}
|
|
|
|
// Config retrieves the chain's fork configuration.
|
|
func (bc *BlockChain) Config() *params.ChainConfig { return bc.chainConfig }
|
|
|
|
// Engine retrieves the blockchain's consensus engine.
|
|
func (bc *BlockChain) Engine() consensus.Engine { return bc.engine }
|
|
|
|
// Snapshots returns the blockchain snapshot tree.
|
|
func (bc *BlockChain) Snapshots() *snapshot.Tree {
|
|
return bc.snaps
|
|
}
|
|
|
|
// Validator returns the current validator.
|
|
func (bc *BlockChain) Validator() Validator {
|
|
return bc.validator
|
|
}
|
|
|
|
// Processor returns the current processor.
|
|
func (bc *BlockChain) Processor() Processor {
|
|
return bc.processor
|
|
}
|
|
|
|
// GasLimit returns the gas limit of the current HEAD block.
|
|
func (bc *BlockChain) GasLimit() uint64 {
|
|
return bc.CurrentBlock().GasLimit
|
|
}
|
|
|
|
// Genesis retrieves the chain's genesis block.
|
|
func (bc *BlockChain) Genesis() *types.Block {
|
|
return bc.genesisBlock
|
|
}
|
|
|
|
// GetVMConfig returns the block chain VM config.
|
|
func (bc *BlockChain) GetVMConfig() *vm.Config {
|
|
return &bc.cfg.VmConfig
|
|
}
|
|
|
|
// TxIndexProgress returns the transaction indexing progress.
|
|
func (bc *BlockChain) TxIndexProgress() (TxIndexProgress, error) {
|
|
if bc.txIndexer == nil {
|
|
return TxIndexProgress{}, errors.New("tx indexer is not enabled")
|
|
}
|
|
return bc.txIndexer.txIndexProgress(), nil
|
|
}
|
|
|
|
// StateIndexProgress returns the historical state indexing progress.
|
|
func (bc *BlockChain) StateIndexProgress() (uint64, uint64, error) {
|
|
return bc.triedb.IndexProgress()
|
|
}
|
|
|
|
// HistoryPruningCutoff returns the configured history pruning point.
|
|
// Blocks before this might not be available in the database.
|
|
func (bc *BlockChain) HistoryPruningCutoff() (uint64, common.Hash) {
|
|
pt := bc.historyPrunePoint.Load()
|
|
if pt == nil {
|
|
return 0, bc.genesisBlock.Hash()
|
|
}
|
|
return pt.BlockNumber, pt.BlockHash
|
|
}
|
|
|
|
// TrieDB retrieves the low level trie database used for data storage.
|
|
func (bc *BlockChain) TrieDB() *triedb.Database {
|
|
return bc.triedb
|
|
}
|
|
|
|
// CodeDB retrieves the low level contract code database used for data storage.
|
|
func (bc *BlockChain) CodeDB() *state.CodeDB {
|
|
return bc.codedb
|
|
}
|
|
|
|
// HeaderChain returns the underlying header chain.
|
|
func (bc *BlockChain) HeaderChain() *HeaderChain {
|
|
return bc.hc
|
|
}
|
|
|
|
// SubscribeRemovedLogsEvent registers a subscription of RemovedLogsEvent.
|
|
func (bc *BlockChain) SubscribeRemovedLogsEvent(ch chan<- RemovedLogsEvent) event.Subscription {
|
|
return bc.scope.Track(bc.rmLogsFeed.Subscribe(ch))
|
|
}
|
|
|
|
// SubscribeChainEvent registers a subscription of ChainEvent.
|
|
func (bc *BlockChain) SubscribeChainEvent(ch chan<- ChainEvent) event.Subscription {
|
|
return bc.scope.Track(bc.chainFeed.Subscribe(ch))
|
|
}
|
|
|
|
// SubscribeChainHeadEvent registers a subscription of ChainHeadEvent.
|
|
func (bc *BlockChain) SubscribeChainHeadEvent(ch chan<- ChainHeadEvent) event.Subscription {
|
|
return bc.scope.Track(bc.chainHeadFeed.Subscribe(ch))
|
|
}
|
|
|
|
// SubscribeLogsEvent registers a subscription of []*types.Log.
|
|
func (bc *BlockChain) SubscribeLogsEvent(ch chan<- []*types.Log) event.Subscription {
|
|
return bc.scope.Track(bc.logsFeed.Subscribe(ch))
|
|
}
|
|
|
|
// SubscribeBlockProcessingEvent registers a subscription of bool where true means
|
|
// block processing has started while false means it has stopped.
|
|
func (bc *BlockChain) SubscribeBlockProcessingEvent(ch chan<- bool) event.Subscription {
|
|
return bc.scope.Track(bc.blockProcFeed.Subscribe(ch))
|
|
}
|
|
|
|
// SubscribeNewPayloadEvent registers a subscription for NewPayloadEvent.
|
|
func (bc *BlockChain) SubscribeNewPayloadEvent(ch chan<- NewPayloadEvent) event.Subscription {
|
|
return bc.scope.Track(bc.newPayloadFeed.Subscribe(ch))
|
|
}
|
|
|
|
// SendNewPayloadEvent sends a NewPayloadEvent to subscribers.
|
|
func (bc *BlockChain) SendNewPayloadEvent(ev NewPayloadEvent) {
|
|
bc.newPayloadFeed.Send(ev)
|
|
}
|