mirror of https://github.com/ethereum/go-ethereum
cmd, triedb: implement history inspection (#29267)
This pull request introduces a database tool for inspecting the state history. It can be used for either account history or storage slot history, within a specific block range. The state output format can be chosen either with - the "rlp-encoded" values (those inserted into the merkle trie) - the "rlp-decoded" value (the raw state value) The latter one needs --raw flag.pull/29047/head
parent
f46fe62c5d
commit
6490d9897a
@ -0,0 +1,72 @@ |
||||
// Copyright 2023 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 triedb |
||||
|
||||
import ( |
||||
"errors" |
||||
|
||||
"github.com/ethereum/go-ethereum/common" |
||||
"github.com/ethereum/go-ethereum/triedb/pathdb" |
||||
) |
||||
|
||||
// AccountHistory inspects the account history within the specified range.
|
||||
//
|
||||
// Start: State ID of the first history object for the query. 0 implies the first
|
||||
// available object is selected as the starting point.
|
||||
//
|
||||
// End: State ID of the last history for the query. 0 implies the last available
|
||||
// object is selected as the starting point. Note end is included for query.
|
||||
//
|
||||
// This function is only supported by path mode database.
|
||||
func (db *Database) AccountHistory(address common.Address, start, end uint64) (*pathdb.HistoryStats, error) { |
||||
pdb, ok := db.backend.(*pathdb.Database) |
||||
if !ok { |
||||
return nil, errors.New("not supported") |
||||
} |
||||
return pdb.AccountHistory(address, start, end) |
||||
} |
||||
|
||||
// StorageHistory inspects the storage history within the specified range.
|
||||
//
|
||||
// Start: State ID of the first history object for the query. 0 implies the first
|
||||
// available object is selected as the starting point.
|
||||
//
|
||||
// End: State ID of the last history for the query. 0 implies the last available
|
||||
// object is selected as the starting point. Note end is included for query.
|
||||
//
|
||||
// Note, slot refers to the hash of the raw slot key.
|
||||
//
|
||||
// This function is only supported by path mode database.
|
||||
func (db *Database) StorageHistory(address common.Address, slot common.Hash, start uint64, end uint64) (*pathdb.HistoryStats, error) { |
||||
pdb, ok := db.backend.(*pathdb.Database) |
||||
if !ok { |
||||
return nil, errors.New("not supported") |
||||
} |
||||
return pdb.StorageHistory(address, slot, start, end) |
||||
} |
||||
|
||||
// HistoryRange returns the block numbers associated with earliest and latest
|
||||
// state history in the local store.
|
||||
//
|
||||
// This function is only supported by path mode database.
|
||||
func (db *Database) HistoryRange() (uint64, uint64, error) { |
||||
pdb, ok := db.backend.(*pathdb.Database) |
||||
if !ok { |
||||
return 0, 0, errors.New("not supported") |
||||
} |
||||
return pdb.HistoryRange() |
||||
} |
@ -0,0 +1,151 @@ |
||||
// Copyright 2024 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 pathdb |
||||
|
||||
import ( |
||||
"fmt" |
||||
"time" |
||||
|
||||
"github.com/ethereum/go-ethereum/common" |
||||
"github.com/ethereum/go-ethereum/core/rawdb" |
||||
"github.com/ethereum/go-ethereum/log" |
||||
) |
||||
|
||||
// HistoryStats wraps the history inspection statistics.
|
||||
type HistoryStats struct { |
||||
Start uint64 // Block number of the first queried history
|
||||
End uint64 // Block number of the last queried history
|
||||
Blocks []uint64 // Blocks refers to the list of block numbers in which the state is mutated
|
||||
Origins [][]byte // Origins refers to the original value of the state before its mutation
|
||||
} |
||||
|
||||
// sanitizeRange limits the given range to fit within the local history store.
|
||||
func sanitizeRange(start, end uint64, freezer *rawdb.ResettableFreezer) (uint64, uint64, error) { |
||||
// Load the id of the first history object in local store.
|
||||
tail, err := freezer.Tail() |
||||
if err != nil { |
||||
return 0, 0, err |
||||
} |
||||
first := tail + 1 |
||||
if start != 0 && start > first { |
||||
first = start |
||||
} |
||||
// Load the id of the last history object in local store.
|
||||
head, err := freezer.Ancients() |
||||
if err != nil { |
||||
return 0, 0, err |
||||
} |
||||
last := head - 1 |
||||
if end != 0 && end < last { |
||||
last = end |
||||
} |
||||
// Make sure the range is valid
|
||||
if first >= last { |
||||
return 0, 0, fmt.Errorf("range is invalid, first: %d, last: %d", first, last) |
||||
} |
||||
return first, last, nil |
||||
} |
||||
|
||||
func inspectHistory(freezer *rawdb.ResettableFreezer, start, end uint64, onHistory func(*history, *HistoryStats)) (*HistoryStats, error) { |
||||
var ( |
||||
stats = &HistoryStats{} |
||||
init = time.Now() |
||||
logged = time.Now() |
||||
) |
||||
start, end, err := sanitizeRange(start, end, freezer) |
||||
if err != nil { |
||||
return nil, err |
||||
} |
||||
for id := start; id <= end; id += 1 { |
||||
// The entire history object is decoded, although it's unnecessary for
|
||||
// account inspection. TODO(rjl493456442) optimization is worthwhile.
|
||||
h, err := readHistory(freezer, id) |
||||
if err != nil { |
||||
return nil, err |
||||
} |
||||
if id == start { |
||||
stats.Start = h.meta.block |
||||
} |
||||
if id == end { |
||||
stats.End = h.meta.block |
||||
} |
||||
onHistory(h, stats) |
||||
|
||||
if time.Since(logged) > time.Second*8 { |
||||
logged = time.Now() |
||||
eta := float64(time.Since(init)) / float64(id-start+1) * float64(end-id) |
||||
log.Info("Inspecting state history", "checked", id-start+1, "left", end-id, "elapsed", common.PrettyDuration(time.Since(init)), "eta", common.PrettyDuration(eta)) |
||||
} |
||||
} |
||||
log.Info("Inspected state history", "total", end-start+1, "elapsed", common.PrettyDuration(time.Since(init))) |
||||
return stats, nil |
||||
} |
||||
|
||||
// accountHistory inspects the account history within the range.
|
||||
func accountHistory(freezer *rawdb.ResettableFreezer, address common.Address, start, end uint64) (*HistoryStats, error) { |
||||
return inspectHistory(freezer, start, end, func(h *history, stats *HistoryStats) { |
||||
blob, exists := h.accounts[address] |
||||
if !exists { |
||||
return |
||||
} |
||||
stats.Blocks = append(stats.Blocks, h.meta.block) |
||||
stats.Origins = append(stats.Origins, blob) |
||||
}) |
||||
} |
||||
|
||||
// storageHistory inspects the storage history within the range.
|
||||
func storageHistory(freezer *rawdb.ResettableFreezer, address common.Address, slot common.Hash, start uint64, end uint64) (*HistoryStats, error) { |
||||
return inspectHistory(freezer, start, end, func(h *history, stats *HistoryStats) { |
||||
slots, exists := h.storages[address] |
||||
if !exists { |
||||
return |
||||
} |
||||
blob, exists := slots[slot] |
||||
if !exists { |
||||
return |
||||
} |
||||
stats.Blocks = append(stats.Blocks, h.meta.block) |
||||
stats.Origins = append(stats.Origins, blob) |
||||
}) |
||||
} |
||||
|
||||
// historyRange returns the block number range of local state histories.
|
||||
func historyRange(freezer *rawdb.ResettableFreezer) (uint64, uint64, error) { |
||||
// Load the id of the first history object in local store.
|
||||
tail, err := freezer.Tail() |
||||
if err != nil { |
||||
return 0, 0, err |
||||
} |
||||
first := tail + 1 |
||||
|
||||
// Load the id of the last history object in local store.
|
||||
head, err := freezer.Ancients() |
||||
if err != nil { |
||||
return 0, 0, err |
||||
} |
||||
last := head - 1 |
||||
|
||||
fh, err := readHistory(freezer, first) |
||||
if err != nil { |
||||
return 0, 0, err |
||||
} |
||||
lh, err := readHistory(freezer, last) |
||||
if err != nil { |
||||
return 0, 0, err |
||||
} |
||||
return fh.meta.block, lh.meta.block, nil |
||||
} |
Loading…
Reference in new issue