erigon-pulse/core/state/db_state_writer.go
Alex Sharov ce96cf75b2
Intermediate hash phase 3 (#377)
* #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>
2020-03-11 10:31:49 +00:00

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
}