mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-09 20:41:20 +00:00
e48609ee8d
This branch implements support of btree indexes instead of .kvi files. Kvi also created, but not used during search/lookup, only for size comparation. Btree allows to make seek calls, while recsplit indexes does not support that query. During initial tests, btree indexes are ~1-3% smaller than recsplit but avg lookup time bigger up to 1.5us (which compensates with ability to fetch next key). BtAlloc ``` 337000 searches, last took 5.417µs avg=4.385µs next_access_last[of 10000 keys] 105ns 338000 searches, last took 3.792µs avg=4.385µs next_access_last[of 10000 keys] 103ns 339000 searches, last took 5.125µs avg=4.386µs next_access_last[of 10000 keys] 103ns 340000 searches, last took 4.75µs avg=4.386µs next_access_last[of 10000 keys] 103ns 341000 searches, last took 3.875µs avg=4.386µs next_access_last[of 10000 keys] 104ns 342000 searches, last took 6.042µs avg=4.387µs next_access_last[of 10000 keys] 107ns 343000 searches, last took 3.209µs avg=4.388µs next_access_last[of 10000 keys] 108ns 344000 searches, last took 2.292µs avg=4.393µs next_access_last[of 10000 keys] 113ns 345000 searches, last took 2.417µs avg=4.394µs next_access_last[of 10000 keys] 106ns 346000 searches, last took 2.625µs avg=4.394µs next_access_last[of 10000 keys] 103ns 347000 searches, last took 4.417µs avg=4.397µs next_access_last[of 10000 keys] 104ns 348000 searches, last took 5.458µs avg=4.398µs next_access_last[of 10000 keys] 103ns 349000 searches, last took 4.875µs avg=4.4µs next_access_last[of 10000 keys] 102ns 350000 searches, last took 4.25µs avg=4.4µs next_access_last[of 10000 keys] 104ns 351000 searches, last took 6.25µs avg=4.403µs next_access_last[of 10000 keys] 106ns 352000 searches, last took 3.125µs avg=4.404µs next_access_last[of 10000 keys] 106ns ``` Recsplit ``` 354000 searches, last took 1.209µs avg=3.081µs 355000 searches, last took 1.333µs avg=3.076µs 356000 searches, last took 1.541µs avg=3.071µs 357000 searches, last took 1.333µs avg=3.066µs 358000 searches, last took 1.333µs avg=3.062µs 359000 searches, last took 1.167µs avg=3.057µs 360000 searches, last took 1.791µs avg=3.052µs 361000 searches, last took 1.209µs avg=3.048µs 362000 searches, last took 1.375µs avg=3.043µs 363000 searches, last took 1.166µs avg=3.038µs 364000 searches, last took 1.625µs avg=3.033µs 365000 searches, last took 1.125µs avg=3.029µs 366000 searches, last took 1.208µs avg=3.024µs 367000 searches, last took 1.333µs avg=3.019µs 368000 searches, last took 1.791µs avg=3.015µs 369000 searches, last took 1.333µs avg=3.01µs 370000 searches, last took 1.166µs avg=3.006µs 371000 searches, last took 1.125µs avg=3.001µs 372000 searches, last took 1.333µs avg=2.997µs ``` https://github.com/ledgerwatch/erigon-lib/pull/891/files
276 lines
8.2 KiB
Go
276 lines
8.2 KiB
Go
package commands
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"os"
|
|
"os/signal"
|
|
"path"
|
|
"path/filepath"
|
|
"syscall"
|
|
"time"
|
|
|
|
chain2 "github.com/ledgerwatch/erigon-lib/chain"
|
|
"github.com/ledgerwatch/erigon-lib/commitment"
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
"github.com/ledgerwatch/erigon-lib/common/datadir"
|
|
"github.com/ledgerwatch/erigon-lib/kv"
|
|
"github.com/ledgerwatch/erigon-lib/kv/mdbx"
|
|
kv2 "github.com/ledgerwatch/erigon-lib/kv/mdbx"
|
|
libstate "github.com/ledgerwatch/erigon-lib/state"
|
|
"github.com/ledgerwatch/log/v3"
|
|
"github.com/spf13/cobra"
|
|
|
|
"github.com/ledgerwatch/erigon/consensus/ethash"
|
|
"github.com/ledgerwatch/erigon/consensus/misc"
|
|
"github.com/ledgerwatch/erigon/core"
|
|
"github.com/ledgerwatch/erigon/core/rawdb"
|
|
"github.com/ledgerwatch/erigon/core/state"
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
"github.com/ledgerwatch/erigon/core/vm"
|
|
"github.com/ledgerwatch/erigon/eth/ethconfig"
|
|
"github.com/ledgerwatch/erigon/turbo/services"
|
|
"github.com/ledgerwatch/erigon/turbo/snapshotsync"
|
|
)
|
|
|
|
var (
|
|
blockTo int
|
|
traceBlock int
|
|
)
|
|
|
|
func init() {
|
|
withBlock(history22Cmd)
|
|
withDataDir(history22Cmd)
|
|
history22Cmd.Flags().IntVar(&traceBlock, "traceblock", 0, "block number at which to turn on tracing")
|
|
history22Cmd.Flags().IntVar(&blockTo, "blockto", 0, "block number to stop replay of history at")
|
|
rootCmd.AddCommand(history22Cmd)
|
|
}
|
|
|
|
var history22Cmd = &cobra.Command{
|
|
Use: "history22",
|
|
Short: "Exerimental command to re-execute historical transactions in erigon2 format",
|
|
RunE: func(cmd *cobra.Command, args []string) error {
|
|
logger := log.New()
|
|
return History22(genesis, logger)
|
|
},
|
|
}
|
|
|
|
func History22(genesis *core.Genesis, logger log.Logger) error {
|
|
sigs := make(chan os.Signal, 1)
|
|
interruptCh := make(chan bool, 1)
|
|
signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)
|
|
|
|
go func() {
|
|
<-sigs
|
|
interruptCh <- true
|
|
}()
|
|
dirs := datadir.New(datadirCli)
|
|
historyDb, err := kv2.NewMDBX(logger).Path(dirs.Chaindata).Open()
|
|
if err != nil {
|
|
return fmt.Errorf("opening chaindata as read only: %v", err)
|
|
}
|
|
defer historyDb.Close()
|
|
ctx := context.Background()
|
|
historyTx, err1 := historyDb.BeginRo(ctx)
|
|
if err1 != nil {
|
|
return err1
|
|
}
|
|
defer historyTx.Rollback()
|
|
aggPath := filepath.Join(datadirCli, "erigon23")
|
|
h, err := libstate.NewAggregator(aggPath, dirs.Tmp, ethconfig.HistoryV3AggregationStep, libstate.CommitmentModeDirect, commitment.VariantHexPatriciaTrie)
|
|
if err != nil {
|
|
return fmt.Errorf("create history: %w", err)
|
|
}
|
|
if err := h.ReopenFolder(); err != nil {
|
|
return err
|
|
}
|
|
|
|
defer h.Close()
|
|
readDbPath := path.Join(datadirCli, "readdb")
|
|
if block == 0 {
|
|
if _, err = os.Stat(readDbPath); err != nil {
|
|
if !errors.Is(err, os.ErrNotExist) {
|
|
return err
|
|
}
|
|
} else if err = os.RemoveAll(readDbPath); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
db, err := kv2.NewMDBX(logger).Path(readDbPath).WriteMap().Open()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer db.Close()
|
|
readPath := filepath.Join(datadirCli, "reads")
|
|
if block == 0 {
|
|
if _, err = os.Stat(readPath); err != nil {
|
|
if !errors.Is(err, os.ErrNotExist) {
|
|
return err
|
|
}
|
|
} else if err = os.RemoveAll(readPath); err != nil {
|
|
return err
|
|
}
|
|
if err = os.Mkdir(readPath, os.ModePerm); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
ri, err := libstate.NewReadIndices(readPath, dirs.Tmp, ethconfig.HistoryV3AggregationStep)
|
|
if err != nil {
|
|
return fmt.Errorf("create read indices: %w", err)
|
|
}
|
|
var rwTx kv.RwTx
|
|
defer func() {
|
|
if rwTx != nil {
|
|
rwTx.Rollback()
|
|
}
|
|
}()
|
|
if rwTx, err = db.BeginRw(ctx); err != nil {
|
|
return err
|
|
}
|
|
ri.SetTx(rwTx)
|
|
chainConfig := genesis.Config
|
|
vmConfig := vm.Config{}
|
|
|
|
interrupt := false
|
|
blockNum := uint64(0)
|
|
var txNum uint64 = 2
|
|
trace := false
|
|
logEvery := time.NewTicker(logInterval)
|
|
defer logEvery.Stop()
|
|
prevBlock := blockNum
|
|
prevTime := time.Now()
|
|
|
|
var blockReader services.FullBlockReader
|
|
allSnapshots := snapshotsync.NewRoSnapshots(ethconfig.NewSnapCfg(true, false, true), path.Join(datadirCli, "snapshots"))
|
|
defer allSnapshots.Close()
|
|
if err := allSnapshots.ReopenWithDB(db); err != nil {
|
|
return fmt.Errorf("reopen snapshot segments: %w", err)
|
|
}
|
|
blockReader = snapshotsync.NewBlockReaderWithSnapshots(allSnapshots, ethconfig.Defaults.TransactionsV3)
|
|
readWrapper := state.NewHistoryReader23(h.MakeContext(), ri)
|
|
|
|
for !interrupt {
|
|
select {
|
|
default:
|
|
case <-logEvery.C:
|
|
currentTime := time.Now()
|
|
interval := currentTime.Sub(prevTime)
|
|
speed := float64(blockNum-prevBlock) / (float64(interval) / float64(time.Second))
|
|
prevBlock = blockNum
|
|
prevTime = currentTime
|
|
log.Info("Progress", "block", blockNum, "blk/s", speed)
|
|
}
|
|
blockNum++
|
|
if blockNum > uint64(blockTo) {
|
|
break
|
|
}
|
|
blockHash, err := rawdb.ReadCanonicalHash(historyTx, blockNum)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b, _, err := blockReader.BlockWithSenders(ctx, historyTx, blockHash, blockNum)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if b == nil {
|
|
log.Info("history: block is nil", "block", blockNum)
|
|
break
|
|
}
|
|
if blockNum <= block {
|
|
// Skip that block, but increase txNum
|
|
txNum += uint64(len(b.Transactions())) + 2 // Pre and Post block transaction
|
|
continue
|
|
}
|
|
if traceBlock != 0 {
|
|
readWrapper.SetTrace(blockNum == uint64(traceBlock))
|
|
}
|
|
writeWrapper := state.NewNoopWriter()
|
|
getHeader := func(hash libcommon.Hash, number uint64) *types.Header {
|
|
h, err := blockReader.Header(ctx, historyTx, hash, number)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
return h
|
|
}
|
|
if txNum, _, err = runHistory22(trace, blockNum, txNum, readWrapper, writeWrapper, chainConfig, getHeader, b, vmConfig); err != nil {
|
|
return fmt.Errorf("block %d: %w", blockNum, err)
|
|
}
|
|
txNum++ // Post block transaction
|
|
// Check for interrupts
|
|
select {
|
|
case interrupt = <-interruptCh:
|
|
log.Info(fmt.Sprintf("interrupted, please wait for cleanup, next time start with --block %d", blockNum))
|
|
default:
|
|
}
|
|
// Commit transaction only when interrupted or just before computing commitment (so it can be re-done)
|
|
commit := interrupt
|
|
if !commit && (blockNum+1)%uint64(commitmentFrequency) == 0 {
|
|
var spaceDirty uint64
|
|
if spaceDirty, _, err = rwTx.(*mdbx.MdbxTx).SpaceDirty(); err != nil {
|
|
return fmt.Errorf("retrieving spaceDirty: %w", err)
|
|
}
|
|
if spaceDirty >= dirtySpaceThreshold {
|
|
log.Info("Initiated tx commit", "block", blockNum, "space dirty", libcommon.ByteCount(spaceDirty))
|
|
commit = true
|
|
}
|
|
}
|
|
if commit {
|
|
if err = rwTx.Commit(); err != nil {
|
|
return err
|
|
}
|
|
if !interrupt {
|
|
if rwTx, err = db.BeginRw(ctx); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
ri.SetTx(rwTx)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func runHistory22(trace bool, blockNum, txNumStart uint64, hw *state.HistoryReaderV4, ww state.StateWriter, chainConfig *chain2.Config, getHeader func(hash libcommon.Hash, number uint64) *types.Header, block *types.Block, vmConfig vm.Config) (uint64, types.Receipts, error) {
|
|
header := block.Header()
|
|
vmConfig.TraceJumpDest = true
|
|
engine := ethash.NewFullFaker()
|
|
gp := new(core.GasPool).AddGas(block.GasLimit())
|
|
usedGas := new(uint64)
|
|
var receipts types.Receipts
|
|
rules := chainConfig.Rules(block.NumberU64(), block.Time())
|
|
txNum := txNumStart
|
|
hw.SetTxNum(txNum)
|
|
daoFork := chainConfig.DAOForkSupport && chainConfig.DAOForkBlock != nil && chainConfig.DAOForkBlock.Cmp(block.Number()) == 0
|
|
if daoFork {
|
|
ibs := state.New(hw)
|
|
misc.ApplyDAOHardFork(ibs)
|
|
if err := ibs.FinalizeTx(rules, ww); err != nil {
|
|
return 0, nil, err
|
|
}
|
|
if err := hw.FinishTx(); err != nil {
|
|
return 0, nil, fmt.Errorf("finish dao fork failed: %w", err)
|
|
}
|
|
}
|
|
txNum++ // Pre block transaction
|
|
for i, tx := range block.Transactions() {
|
|
hw.SetTxNum(txNum)
|
|
ibs := state.New(hw)
|
|
ibs.Prepare(tx.Hash(), block.Hash(), i)
|
|
receipt, _, err := core.ApplyTransaction(chainConfig, core.GetHashFn(header, getHeader), engine, nil, gp, ibs, ww, header, tx, usedGas, vmConfig)
|
|
if err != nil {
|
|
return 0, nil, fmt.Errorf("could not apply tx %d [%x] failed: %w", i, tx.Hash(), err)
|
|
}
|
|
if traceBlock != 0 && blockNum == uint64(traceBlock) {
|
|
fmt.Printf("tx idx %d, num %d, gas used %d\n", i, txNum, receipt.GasUsed)
|
|
}
|
|
receipts = append(receipts, receipt)
|
|
if err = hw.FinishTx(); err != nil {
|
|
return 0, nil, fmt.Errorf("finish tx %d [%x] failed: %w", i, tx.Hash(), err)
|
|
}
|
|
txNum++
|
|
hw.SetTxNum(txNum)
|
|
}
|
|
|
|
return txNum, receipts, nil
|
|
}
|