mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-01 16:47:39 +00:00
ce96cf75b2
* #remove debug prints * remove storage-mode="i" * minnet re-execute hack with checkpoints * minnet re-execute hack with checkpoints * rollback to master setup * mainnet re-exec hack * rollback some changes * v0 of "push down" functionality * move all logic to own functions * handle case when re-created account already has some storage * clear path for storage * try to rely on tree structure (but maybe need to rely on DB because can be intra-block re-creations of account) * fix some bugs with indexes, moving to tests * tests added * make linter happy * make linter happy * simplify logic * adjust comparison of keys with and without incarnation * test for keyIsBefore * test for keyIsBefore * better nibbles alignment * better nibbles alignment * cleanup * continue work on tests * simplify test * check tombstone existence before pushing it down. * put tombstone only when account deleted, not created * put tombstone only when account has storage * make linter happy * test for storage resolver * make fixedbytes work without incarnation * fix panic on short keys * use special comparison only when working with keys from cache * add blockNr for better tracing * fix: incorrect tombstone check * fix: incorrect tombstone check * trigger ci * hack for problem block * more test-cases * add test case for too long keys * speedup cached resolver by removing bucket creation transaction * remove parent type check in pruning, remove unused copy from mutation.put * dump resolving info on fail * dump resolving info on fail * set tombstone everytime for now to check if it will help * on unload: check parent type, not type of node * fix wrong order of checking node type * fix wrong order of checking node type * rebase to new master * make linter happy * rebase to new master * place tombstone only if acc has storage * rebase master * rebase master * rebase master * rebase master Co-authored-by: alex.sharov <alex.sharov@lazada.com>
131 lines
4.0 KiB
Go
131 lines
4.0 KiB
Go
package state
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/common"
|
|
"github.com/ledgerwatch/turbo-geth/common/dbutils"
|
|
"github.com/ledgerwatch/turbo-geth/common/debug"
|
|
"github.com/ledgerwatch/turbo-geth/core/types/accounts"
|
|
"github.com/ledgerwatch/turbo-geth/trie"
|
|
)
|
|
|
|
type DbStateWriter struct {
|
|
tds *TrieDbState
|
|
}
|
|
|
|
func (dsw *DbStateWriter) UpdateAccountData(ctx context.Context, address common.Address, original, account *accounts.Account) error {
|
|
dataLen := account.EncodingLengthForStorage()
|
|
data := make([]byte, dataLen)
|
|
account.EncodeForStorage(data)
|
|
|
|
addrHash, err := dsw.tds.HashAddress(address, true /*save*/)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err = dsw.tds.db.Put(dbutils.AccountsBucket, addrHash[:], data); err != nil {
|
|
return err
|
|
}
|
|
|
|
noHistory := dsw.tds.noHistory
|
|
// Don't write historical record if the account did not change
|
|
if accountsEqual(original, account) {
|
|
return nil
|
|
}
|
|
var originalData []byte
|
|
if !original.Initialised {
|
|
originalData = []byte{}
|
|
} else {
|
|
// we can reduce storage size for history there
|
|
// because we have accountHash+incarnation -> codehash of contract in separate bucket
|
|
// and we don't need root in history requests
|
|
testAcc := original.SelfCopy()
|
|
if debug.IsThinHistory() {
|
|
copy(testAcc.CodeHash[:], emptyCodeHash)
|
|
testAcc.Root = trie.EmptyRoot
|
|
}
|
|
|
|
originalDataLen := testAcc.EncodingLengthForStorage()
|
|
originalData = make([]byte, originalDataLen)
|
|
testAcc.EncodeForStorage(originalData)
|
|
}
|
|
return dsw.tds.db.PutS(dbutils.AccountsHistoryBucket, addrHash[:], originalData, dsw.tds.blockNr, noHistory)
|
|
}
|
|
|
|
func (dsw *DbStateWriter) DeleteAccount(ctx context.Context, address common.Address, original *accounts.Account) error {
|
|
addrHash, err := dsw.tds.HashAddress(address, true /*save*/)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err := dsw.tds.db.Delete(dbutils.AccountsBucket, addrHash[:]); err != nil {
|
|
return err
|
|
}
|
|
|
|
var originalData []byte
|
|
if !original.Initialised {
|
|
// Account has been created and deleted in the same block
|
|
originalData = []byte{}
|
|
} else {
|
|
originalDataLen := original.EncodingLengthForStorage()
|
|
originalData = make([]byte, originalDataLen)
|
|
original.EncodeForStorage(originalData)
|
|
// We must keep root using thin history on deleting account as is
|
|
}
|
|
|
|
noHistory := dsw.tds.noHistory
|
|
return dsw.tds.db.PutS(dbutils.AccountsHistoryBucket, addrHash[:], originalData, dsw.tds.blockNr, noHistory)
|
|
}
|
|
|
|
func (dsw *DbStateWriter) UpdateAccountCode(addrHash common.Hash, incarnation uint64, codeHash common.Hash, code []byte) error {
|
|
//save contract code mapping
|
|
if err := dsw.tds.db.Put(dbutils.CodeBucket, codeHash[:], code); err != nil {
|
|
return err
|
|
}
|
|
if debug.IsThinHistory() {
|
|
//save contract to codeHash mapping
|
|
return dsw.tds.db.Put(dbutils.ContractCodeBucket, dbutils.GenerateStoragePrefix(addrHash, incarnation), codeHash.Bytes())
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (dsw *DbStateWriter) WriteAccountStorage(ctx context.Context, address common.Address, incarnation uint64, key, original, value *common.Hash) error {
|
|
if *original == *value {
|
|
return nil
|
|
}
|
|
seckey, err := dsw.tds.HashKey(key, true /*save*/)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
v := bytes.TrimLeft(value[:], "\x00")
|
|
vv := make([]byte, len(v))
|
|
copy(vv, v)
|
|
|
|
addrHash, err := dsw.tds.HashAddress(address, false /*save*/)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
compositeKey := dbutils.GenerateCompositeStorageKey(addrHash, incarnation, seckey)
|
|
if len(v) == 0 {
|
|
err = dsw.tds.db.Delete(dbutils.StorageBucket, compositeKey)
|
|
} else {
|
|
|
|
err = dsw.tds.db.Put(dbutils.StorageBucket, compositeKey, vv)
|
|
}
|
|
//fmt.Printf("WriteAccountStorage (db) %x %d %x: %x\n", address, incarnation, key, value)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
noHistory := dsw.tds.noHistory
|
|
o := bytes.TrimLeft(original[:], "\x00")
|
|
originalValue := make([]byte, len(o))
|
|
copy(originalValue, o)
|
|
return dsw.tds.db.PutS(dbutils.StorageHistoryBucket, compositeKey, originalValue, dsw.tds.blockNr, noHistory)
|
|
}
|
|
|
|
func (dsw *DbStateWriter) CreateContract(address common.Address) error {
|
|
return nil
|
|
}
|