erigon-pulse/cmd/restapi/apis/remote_reader.go
Andrew Ashikhmin 7b40cbb6fa
Incarnation should be read by StateReader, not StateWriter (#506)
* GetCurrentAccountIncarnation

* Incarnation should be read by StateReader, not StateWriter

* Use GetHistoricalAccountIncarnation in DbState

* RemoteReader ReadAccountIncarnation

* Handle the case where a contract has self-destructed, then Eth sent to it, then it got recreated again
2020-05-02 19:00:42 +01:00

264 lines
7.0 KiB
Go

package apis
import (
"bytes"
"encoding/binary"
"math/big"
"github.com/ledgerwatch/turbo-geth/common"
"github.com/ledgerwatch/turbo-geth/common/dbutils"
"github.com/ledgerwatch/turbo-geth/consensus"
"github.com/ledgerwatch/turbo-geth/core/state"
"github.com/ledgerwatch/turbo-geth/core/types"
"github.com/ledgerwatch/turbo-geth/core/types/accounts"
"github.com/ledgerwatch/turbo-geth/crypto"
"github.com/ledgerwatch/turbo-geth/ethdb"
"github.com/ledgerwatch/turbo-geth/ethdb/remote/remotechain"
"github.com/ledgerwatch/turbo-geth/params"
"github.com/ledgerwatch/turbo-geth/rpc"
"golang.org/x/net/context"
)
// ChangeSetReader is a mock StateWriter that accumulates changes in-memory into ChangeSets.
type RemoteReader struct {
accountReads map[common.Address]bool
storageReads map[common.Hash]bool
blockNr uint64
db ethdb.KV
}
type RemoteContext struct {
db ethdb.KV
}
type powEngine struct {
}
func (c *powEngine) VerifyHeader(chain consensus.ChainReader, header *types.Header, seal bool) error {
panic("must not be called")
}
func (c *powEngine) VerifyHeaders(chain consensus.ChainReader, headers []*types.Header, seals []bool) (chan<- struct{}, <-chan error) {
panic("must not be called")
}
func (c *powEngine) VerifyUncles(chain consensus.ChainReader, block *types.Block) error {
panic("must not be called")
}
func (c *powEngine) VerifySeal(chain consensus.ChainReader, header *types.Header) error {
panic("must not be called")
}
func (c *powEngine) Prepare(chain consensus.ChainReader, header *types.Header) error {
panic("must not be called")
}
func (c *powEngine) Finalize(chainConfig *params.ChainConfig, header *types.Header, state *state.IntraBlockState, txs []*types.Transaction, uncles []*types.Header) {
panic("must not be called")
}
func (c *powEngine) FinalizeAndAssemble(chainConfig *params.ChainConfig, header *types.Header, state *state.IntraBlockState, txs []*types.Transaction,
uncles []*types.Header, receipts []*types.Receipt) (*types.Block, error) {
panic("must not be called")
}
func (c *powEngine) Seal(_ consensus.Cancel, chain consensus.ChainReader, block *types.Block, results chan<- consensus.ResultWithContext, stop <-chan struct{}) error {
panic("must not be called")
}
func (c *powEngine) SealHash(header *types.Header) common.Hash {
panic("must not be called")
}
func (c *powEngine) CalcDifficulty(chain consensus.ChainReader, time uint64, parent *types.Header) *big.Int {
panic("must not be called")
}
func (c *powEngine) APIs(chain consensus.ChainReader) []rpc.API {
panic("must not be called")
}
func (c *powEngine) Close() error {
panic("must not be called")
}
func (c *powEngine) Author(header *types.Header) (common.Address, error) {
return header.Coinbase, nil
}
func NewRemoteReader(db ethdb.KV, blockNr uint64) *RemoteReader {
return &RemoteReader{
accountReads: make(map[common.Address]bool),
storageReads: make(map[common.Hash]bool),
db: db,
blockNr: blockNr,
}
}
func (r *RemoteReader) GetAccountReads() [][]byte {
output := make([][]byte, 0)
for key := range r.accountReads {
output = append(output, key.Bytes())
}
return output
}
func (r *RemoteReader) GetStorageReads() [][]byte {
output := make([][]byte, 0)
for key := range r.storageReads {
output = append(output, key.Bytes())
}
return output
}
func (r *RemoteReader) ReadAccountData(address common.Address) (*accounts.Account, error) {
r.accountReads[address] = true
addrHash, _ := common.HashData(address[:])
key := addrHash[:]
r.accountReads[address] = true
composite, _ := dbutils.CompositeKeySuffix(key, r.blockNr)
acc := accounts.NewAccount()
err := r.db.View(context.Background(), func(tx ethdb.Tx) error {
{
hB := tx.Bucket(dbutils.AccountsHistoryBucket)
hC := hB.Cursor()
hK, hV, err := hC.Seek(composite)
if err != nil {
return err
}
if hK != nil && bytes.HasPrefix(hK, key) {
err = acc.DecodeForStorage(hV)
if err != nil {
return err
}
return nil
}
}
{
v, err := tx.Bucket(dbutils.CurrentStateBucket).Get(key)
if err != nil {
return err
}
if v == nil {
return nil
}
root, err := tx.Bucket(dbutils.IntermediateTrieHashBucket).Get(key)
if err != nil {
return err
}
if root == nil {
return nil
}
err = acc.DecodeForStorage(v)
if err != nil {
return err
}
acc.Root = common.BytesToHash(root)
return nil
}
return ethdb.ErrKeyNotFound
})
if err != nil {
return nil, nil
}
return &acc, nil
}
func (r *RemoteReader) ReadAccountStorage(address common.Address, incarnation uint64, key *common.Hash) ([]byte, error) {
keyHash, _ := common.HashData(key[:])
addrHash, _ := common.HashData(address[:])
compositeKey := dbutils.GenerateCompositeStorageKey(addrHash, incarnation, keyHash)
var val []byte
err := r.db.View(context.Background(), func(tx ethdb.Tx) error {
b := tx.Bucket(dbutils.CurrentStateBucket)
v, err := b.Get(compositeKey)
val = v
return err
})
if err != nil {
return nil, err
}
r.storageReads[*key] = true
return val, nil
}
func (r *RemoteReader) ReadAccountCode(address common.Address, codeHash common.Hash) ([]byte, error) {
if bytes.Equal(codeHash[:], crypto.Keccak256(nil)) {
return nil, nil
}
var val []byte
err := r.db.View(context.Background(), func(tx ethdb.Tx) error {
b := tx.Bucket(dbutils.CurrentStateBucket)
v, err := b.Get(codeHash[:])
val = v
return err
})
if err != nil {
return nil, err
}
return val, nil
}
func (r *RemoteReader) ReadAccountCodeSize(address common.Address, codeHash common.Hash) (int, error) {
if bytes.Equal(codeHash[:], crypto.Keccak256(nil)) {
return 0, nil
}
var val []byte
err := r.db.View(context.Background(), func(tx ethdb.Tx) error {
b := tx.Bucket(dbutils.CurrentStateBucket)
v, err := b.Get(codeHash[:])
val = v
return err
})
if err != nil {
return 0, err
}
return len(val), nil
}
func (r *RemoteReader) ReadAccountIncarnation(address common.Address) (uint64, error) {
addrHash, _ := common.HashData(address[:])
var incarnationBytes [common.IncarnationLength]byte
startkey := make([]byte, common.HashLength+common.IncarnationLength+common.HashLength)
copy(startkey, addrHash[:])
found := false
err := r.db.View(context.Background(), func(tx ethdb.Tx) error {
b := tx.Bucket(dbutils.CurrentStateBucket)
c := b.Cursor()
k, _, err := c.Seek(startkey)
if err != nil {
return err
}
if k != nil && bytes.HasPrefix(k, addrHash[:]) {
copy(incarnationBytes[:], k[common.HashLength:])
found = true
}
return nil
})
if err != nil {
return 0, err
}
if found {
return (^binary.BigEndian.Uint64(incarnationBytes[:])), nil
}
return 0, nil
}
func NewRemoteContext(db ethdb.KV) *RemoteContext {
return &RemoteContext{
db: db,
}
}
func (e *RemoteContext) Engine() consensus.Engine {
return &powEngine{}
}
func (e *RemoteContext) GetHeader(hash common.Hash, number uint64) *types.Header {
var header *types.Header
_ = e.db.View(context.Background(), func(tx ethdb.Tx) error {
h, _ := remotechain.ReadHeader(tx, hash, number)
header = h
return nil
})
return header
}