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 }