package main import ( "bufio" "bytes" "context" "encoding/binary" "encoding/json" "flag" "fmt" "math/big" "net/http" _ "net/http/pprof" //nolint:gosec "os" "path/filepath" "runtime/pprof" "sort" "strings" "time" "github.com/RoaringBitmap/roaring/roaring64" "github.com/holiman/uint256" "github.com/ledgerwatch/erigon-lib/compress" "github.com/ledgerwatch/erigon-lib/kv" "github.com/ledgerwatch/erigon-lib/kv/mdbx" "github.com/ledgerwatch/erigon-lib/recsplit" "github.com/ledgerwatch/erigon-lib/recsplit/eliasfano32" "golang.org/x/exp/slices" hackdb "github.com/ledgerwatch/erigon/cmd/hack/db" "github.com/ledgerwatch/erigon/cmd/hack/flow" "github.com/ledgerwatch/erigon/cmd/hack/tool" "github.com/ledgerwatch/erigon/common" "github.com/ledgerwatch/erigon/common/changeset" "github.com/ledgerwatch/erigon/common/dbutils" "github.com/ledgerwatch/erigon/common/paths" "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/crypto" "github.com/ledgerwatch/erigon/eth/ethconfig" "github.com/ledgerwatch/erigon/eth/stagedsync/stages" "github.com/ledgerwatch/erigon/ethdb" "github.com/ledgerwatch/erigon/ethdb/cbor" "github.com/ledgerwatch/erigon/internal/debug" "github.com/ledgerwatch/erigon/params" "github.com/ledgerwatch/erigon/rlp" "github.com/ledgerwatch/erigon/turbo/snapshotsync" "github.com/ledgerwatch/log/v3" ) const ASSERT = false var ( verbosity = flag.Uint("verbosity", 3, "Logging verbosity: 0=silent, 1=error, 2=warn, 3=info, 4=debug, 5=detail (default 3)") action = flag.String("action", "", "action to execute") cpuprofile = flag.String("cpuprofile", "", "write cpu profile `file`") block = flag.Int("block", 1, "specifies a block number for operation") blockTotal = flag.Int("blocktotal", 1, "specifies a total amount of blocks to process (will offset from head block if <= 0)") account = flag.String("account", "0x", "specifies account to investigate") name = flag.String("name", "", "name to add to the file names") chaindata = flag.String("chaindata", "chaindata", "path to the chaindata database file") bucket = flag.String("bucket", "", "bucket in the database") hash = flag.String("hash", "0x00", "image for preimage or state root for testBlockHashes action") ) func dbSlice(chaindata string, bucket string, prefix []byte) { db := mdbx.MustOpen(chaindata) defer db.Close() if err := db.View(context.Background(), func(tx kv.Tx) error { c, err := tx.Cursor(bucket) if err != nil { return err } for k, v, err := c.Seek(prefix); k != nil && bytes.HasPrefix(k, prefix); k, v, err = c.Next() { if err != nil { return err } fmt.Printf("db.Put([]byte(\"%s\"), common.FromHex(\"%x\"), common.FromHex(\"%x\"))\n", bucket, k, v) } return nil }); err != nil { panic(err) } } // Searches 1000 blocks from the given one to try to find the one with the given state root hash func testBlockHashes(chaindata string, block int, stateRoot common.Hash) { ethDb := mdbx.MustOpen(chaindata) defer ethDb.Close() tool.Check(ethDb.View(context.Background(), func(tx kv.Tx) error { blocksToSearch := 10000000 for i := uint64(block); i < uint64(block+blocksToSearch); i++ { hash, err := rawdb.ReadCanonicalHash(tx, i) if err != nil { panic(err) } header := rawdb.ReadHeader(tx, hash, i) if header.Root == stateRoot || stateRoot == (common.Hash{}) { fmt.Printf("\n===============\nCanonical hash for %d: %x\n", i, hash) fmt.Printf("Header.Root: %x\n", header.Root) fmt.Printf("Header.TxHash: %x\n", header.TxHash) fmt.Printf("Header.UncleHash: %x\n", header.UncleHash) } } return nil })) } func getCurrentBlockNumber(tx kv.Tx) *uint64 { hash := rawdb.ReadHeadBlockHash(tx) if hash == (common.Hash{}) { return nil } return rawdb.ReadHeaderNumber(tx, hash) } func printCurrentBlockNumber(chaindata string) { ethDb := mdbx.MustOpen(chaindata) defer ethDb.Close() ethDb.View(context.Background(), func(tx kv.Tx) error { if number := getCurrentBlockNumber(tx); number != nil { fmt.Printf("Block number: %d\n", *number) } else { fmt.Println("Block number: ") } return nil }) } func printTxHashes(chaindata string, block uint64) error { db := mdbx.MustOpen(chaindata) defer db.Close() if err := db.View(context.Background(), func(tx kv.Tx) error { for b := block; b < block+1; b++ { hash, e := rawdb.ReadCanonicalHash(tx, b) if e != nil { return e } block := rawdb.ReadBlock(tx, hash, b) if block == nil { break } for i, tx := range block.Transactions() { fmt.Printf("%d: %x\n", i, tx.Hash()) } } return nil }); err != nil { return err } return nil } func readAccount(chaindata string, account common.Address) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, txErr := db.BeginRo(context.Background()) if txErr != nil { return txErr } defer tx.Rollback() a, err := state.NewPlainStateReader(tx).ReadAccountData(account) if err != nil { return err } else if a == nil { return fmt.Errorf("acc not found") } fmt.Printf("CodeHash:%x\nIncarnation:%d\n", a.CodeHash, a.Incarnation) c, err := tx.Cursor(kv.PlainState) if err != nil { return err } defer c.Close() for k, v, e := c.Seek(account.Bytes()); k != nil; k, v, e = c.Next() { if e != nil { return e } if !bytes.HasPrefix(k, account.Bytes()) { break } fmt.Printf("%x => %x\n", k, v) } cc, err := tx.Cursor(kv.PlainContractCode) if err != nil { return err } defer cc.Close() fmt.Printf("code hashes\n") for k, v, e := cc.Seek(account.Bytes()); k != nil; k, v, e = c.Next() { if e != nil { return e } if !bytes.HasPrefix(k, account.Bytes()) { break } fmt.Printf("%x => %x\n", k, v) } return nil } func nextIncarnation(chaindata string, addrHash common.Hash) { ethDb := mdbx.MustOpen(chaindata) defer ethDb.Close() var found bool var incarnationBytes [common.IncarnationLength]byte startkey := make([]byte, common.HashLength+common.IncarnationLength+common.HashLength) var fixedbits = 8 * common.HashLength copy(startkey, addrHash[:]) tool.Check(ethDb.View(context.Background(), func(tx kv.Tx) error { c, err := tx.Cursor(kv.HashedStorage) if err != nil { return err } defer c.Close() return ethdb.Walk(c, startkey, fixedbits, func(k, v []byte) (bool, error) { fmt.Printf("Incarnation(z): %d\n", 0) copy(incarnationBytes[:], k[common.HashLength:]) found = true return false, nil }) })) if found { fmt.Printf("Incarnation: %d\n", (binary.BigEndian.Uint64(incarnationBytes[:]))+1) return } fmt.Printf("Incarnation(f): %d\n", state.FirstContractIncarnation) } func repairCurrent() { historyDb := mdbx.MustOpen("/Volumes/tb4/erigon/ropsten/geth/chaindata") defer historyDb.Close() currentDb := mdbx.MustOpen("statedb") defer currentDb.Close() tool.Check(historyDb.Update(context.Background(), func(tx kv.RwTx) error { return tx.ClearBucket(kv.HashedStorage) })) tool.Check(historyDb.Update(context.Background(), func(tx kv.RwTx) error { newB, err := tx.RwCursor(kv.HashedStorage) if err != nil { return err } count := 0 if err := currentDb.View(context.Background(), func(ctx kv.Tx) error { c, err := ctx.Cursor(kv.HashedStorage) if err != nil { return err } for k, v, err := c.First(); k != nil; k, v, err = c.Next() { if err != nil { return err } tool.Check(newB.Put(k, v)) count++ if count == 10000 { fmt.Printf("Copied %d storage items\n", count) } } return nil }); err != nil { return err } return nil })) } func dumpStorage() { db := mdbx.MustOpen(paths.DefaultDataDir() + "/geth/chaindata") defer db.Close() if err := db.View(context.Background(), func(tx kv.Tx) error { return tx.ForEach(kv.StorageHistory, nil, func(k, v []byte) error { fmt.Printf("%x %x\n", k, v) return nil }) }); err != nil { panic(err) } } func printBucket(chaindata string) { db := mdbx.MustOpen(chaindata) defer db.Close() f, err := os.Create("bucket.txt") tool.Check(err) defer f.Close() fb := bufio.NewWriter(f) defer fb.Flush() if err := db.View(context.Background(), func(tx kv.Tx) error { c, err := tx.Cursor(kv.StorageHistory) if err != nil { return err } for k, v, err := c.First(); k != nil; k, v, err = c.Next() { if err != nil { return err } fmt.Fprintf(fb, "%x %x\n", k, v) } return nil }); err != nil { panic(err) } } func searchChangeSet(chaindata string, key []byte, block uint64) error { fmt.Printf("Searching changesets\n") db := mdbx.MustOpen(chaindata) defer db.Close() tx, err1 := db.BeginRw(context.Background()) if err1 != nil { return err1 } defer tx.Rollback() if err := changeset.ForEach(tx, kv.AccountChangeSet, dbutils.EncodeBlockNumber(block), func(blockN uint64, k, v []byte) error { if bytes.Equal(k, key) { fmt.Printf("Found in block %d with value %x\n", blockN, v) } return nil }); err != nil { return err } return nil } func searchStorageChangeSet(chaindata string, key []byte, block uint64) error { fmt.Printf("Searching storage changesets\n") db := mdbx.MustOpen(chaindata) defer db.Close() tx, err1 := db.BeginRw(context.Background()) if err1 != nil { return err1 } defer tx.Rollback() if err := changeset.ForEach(tx, kv.StorageChangeSet, dbutils.EncodeBlockNumber(block), func(blockN uint64, k, v []byte) error { if bytes.Equal(k, key) { fmt.Printf("Found in block %d with value %x\n", blockN, v) } return nil }); err != nil { return err } return nil } func extractCode(chaindata string) error { db := mdbx.MustOpen(chaindata) defer db.Close() var contractCount int if err1 := db.View(context.Background(), func(tx kv.Tx) error { c, err := tx.Cursor(kv.Code) if err != nil { return err } // This is a mapping of CodeHash => Byte code for k, v, err := c.First(); k != nil; k, v, err = c.Next() { if err != nil { return err } fmt.Printf("%x,%x", k, v) contractCount++ } return nil }); err1 != nil { return err1 } fmt.Fprintf(os.Stderr, "contractCount: %d\n", contractCount) return nil } func iterateOverCode(chaindata string) error { db := mdbx.MustOpen(chaindata) defer db.Close() hashes := make(map[common.Hash][]byte) if err1 := db.View(context.Background(), func(tx kv.Tx) error { // This is a mapping of CodeHash => Byte code if err := tx.ForEach(kv.Code, nil, func(k, v []byte) error { if len(v) > 0 && v[0] == 0xef { fmt.Printf("Found code with hash %x: %x\n", k, v) hashes[common.BytesToHash(k)] = common.CopyBytes(v) } return nil }); err != nil { return err } // This is a mapping of contractAddress + incarnation => CodeHash if err := tx.ForEach(kv.PlainContractCode, nil, func(k, v []byte) error { hash := common.BytesToHash(v) if code, ok := hashes[hash]; ok { fmt.Printf("address: %x: %x\n", k[:20], code) } return nil }); err != nil { return err } return nil }); err1 != nil { return err1 } return nil } func getBlockTotal(tx kv.Tx, blockFrom uint64, blockTotalOrOffset int64) uint64 { if blockTotalOrOffset > 0 { return uint64(blockTotalOrOffset) } if head := getCurrentBlockNumber(tx); head != nil { if blockSub := uint64(-blockTotalOrOffset); blockSub <= *head { if blockEnd := *head - blockSub; blockEnd > blockFrom { return blockEnd - blockFrom + 1 } } } return 1 } func extractHashes(chaindata string, blockStep uint64, blockTotalOrOffset int64, name string) error { db := mdbx.MustOpen(chaindata) defer db.Close() f, err := os.Create(fmt.Sprintf("preverified_hashes_%s.go", name)) if err != nil { return err } defer f.Close() w := bufio.NewWriter(f) defer w.Flush() fmt.Fprintf(w, "package headerdownload\n\n") fmt.Fprintf(w, "var %sPreverifiedHashes = []string{\n", name) b := uint64(0) tool.Check(db.View(context.Background(), func(tx kv.Tx) error { blockTotal := getBlockTotal(tx, b, blockTotalOrOffset) // Note: blockTotal used here as block number rather than block count for b <= blockTotal { hash, err := rawdb.ReadCanonicalHash(tx, b) if err != nil { return err } if hash == (common.Hash{}) { break } fmt.Fprintf(w, " \"%x\",\n", hash) b += blockStep } return nil })) b -= blockStep fmt.Fprintf(w, "}\n\n") fmt.Fprintf(w, "const %sPreverifiedHeight uint64 = %d\n", name, b) fmt.Printf("Last block is %d\n", b) return nil } func extractHeaders(chaindata string, block uint64, blockTotalOrOffset int64) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, err := db.BeginRo(context.Background()) if err != nil { return err } defer tx.Rollback() c, err := tx.Cursor(kv.Headers) if err != nil { return err } defer c.Close() blockEncoded := dbutils.EncodeBlockNumber(block) blockTotal := getBlockTotal(tx, block, blockTotalOrOffset) for k, v, err := c.Seek(blockEncoded); k != nil && blockTotal > 0; k, v, err = c.Next() { if err != nil { return err } blockNumber := binary.BigEndian.Uint64(k[:8]) blockHash := common.BytesToHash(k[8:]) var header types.Header if err = rlp.DecodeBytes(v, &header); err != nil { return fmt.Errorf("decoding header from %x: %w", v, err) } fmt.Printf("Header %d %x: stateRoot %x, parentHash %x, diff %d\n", blockNumber, blockHash, header.Root, header.ParentHash, header.Difficulty) blockTotal-- } return nil } func extractBodies(datadir string) error { snaps := snapshotsync.NewRoSnapshots(ethconfig.Snapshot{ Enabled: true, KeepBlocks: true, Produce: false, }, filepath.Join(datadir, "snapshots")) snaps.ReopenFolder() snaps.Bodies.View(func(sns []*snapshotsync.BodySegment) error { for _, sn := range sns { var firstBlockNum, firstBaseTxNum, firstAmount uint64 var lastBlockNum, lastBaseTxNum, lastAmount uint64 var prevBlockNum, prevBaseTxNum, prevAmount uint64 first := true sn.Iterate(func(blockNum uint64, baseTxNum uint64, txAmount uint64) error { if first { firstBlockNum = blockNum firstBaseTxNum = baseTxNum firstAmount = txAmount first = false } else { if blockNum != prevBlockNum+1 { fmt.Printf("Discount block Num: %d => %d\n", prevBlockNum, blockNum) } if baseTxNum != prevBaseTxNum+prevAmount { fmt.Printf("Wrong baseTxNum: %d+%d => %d\n", prevBaseTxNum, prevAmount, baseTxNum) } } prevBlockNum = blockNum lastBlockNum = blockNum prevBaseTxNum = baseTxNum lastBaseTxNum = baseTxNum prevAmount = txAmount lastAmount = txAmount return nil }) fmt.Printf("Seg: [%d, %d, %d] => [%d, %d, %d]\n", firstBlockNum, firstBaseTxNum, firstAmount, lastBlockNum, lastBaseTxNum, lastAmount) } return nil }) if _, err := snaps.ViewTxs(snaps.BlocksAvailable(), func(sn *snapshotsync.TxnSegment) error { lastTxnID := sn.IdxTxnHash.BaseDataID() + uint64(sn.Seg.Count()) fmt.Printf("txTxnID = %d\n", lastTxnID) return nil }); err != nil { return err } db := mdbx.MustOpen(filepath.Join(datadir, "chaindata")) defer db.Close() tx, err := db.BeginRo(context.Background()) if err != nil { return err } defer tx.Rollback() c, err := tx.Cursor(kv.BlockBody) if err != nil { return err } defer c.Close() i := 0 var txId uint64 for k, _, err := c.First(); k != nil; k, _, err = c.Next() { if err != nil { return err } blockNumber := binary.BigEndian.Uint64(k[:8]) blockHash := common.BytesToHash(k[8:]) var hash common.Hash if hash, err = rawdb.ReadCanonicalHash(tx, blockNumber); err != nil { return err } _, baseTxId, txAmount := rawdb.ReadBody(tx, blockHash, blockNumber) fmt.Printf("Body %d %x: baseTxId %d, txAmount %d\n", blockNumber, blockHash, baseTxId, txAmount) if hash != blockHash { fmt.Printf("Non-canonical\n") continue } i++ if txId > 0 { if txId != baseTxId { fmt.Printf("Mismatch txId for block %d, txId = %d, baseTxId = %d\n", blockNumber, txId, baseTxId) } } txId = baseTxId + uint64(txAmount) + 2 if i == 50 { break } } return nil } func snapSizes(chaindata string) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, err := db.BeginRo(context.Background()) if err != nil { return err } defer tx.Rollback() c, _ := tx.Cursor(kv.CliqueSeparate) defer c.Close() sizes := make(map[int]int) differentValues := make(map[string]struct{}) var ( total uint64 k, v []byte ) for k, v, err = c.First(); k != nil; k, v, err = c.Next() { if err != nil { return err } sizes[len(v)]++ differentValues[string(v)] = struct{}{} total += uint64(len(v) + len(k)) } var lens = make([]int, len(sizes)) i := 0 for l := range sizes { lens[i] = l i++ } slices.Sort(lens) for _, l := range lens { fmt.Printf("%6d - %d\n", l, sizes[l]) } fmt.Printf("Different keys %d\n", len(differentValues)) fmt.Printf("Total size: %d bytes\n", total) return nil } func readCallTraces(chaindata string, block uint64) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, err := db.BeginRw(context.Background()) if err != nil { return err } defer tx.Rollback() traceCursor, err1 := tx.RwCursorDupSort(kv.CallTraceSet) if err1 != nil { return err1 } defer traceCursor.Close() var k []byte var v []byte count := 0 for k, v, err = traceCursor.First(); k != nil; k, v, err = traceCursor.Next() { if err != nil { return err } blockNum := binary.BigEndian.Uint64(k) if blockNum == block { fmt.Printf("%x\n", v) } count++ } fmt.Printf("Found %d records\n", count) idxCursor, err2 := tx.Cursor(kv.CallToIndex) if err2 != nil { return err2 } var acc = common.HexToAddress("0x511bc4556d823ae99630ae8de28b9b80df90ea2e") for k, v, err = idxCursor.Seek(acc[:]); k != nil && err == nil && bytes.HasPrefix(k, acc[:]); k, v, err = idxCursor.Next() { bm := roaring64.New() _, err = bm.ReadFrom(bytes.NewReader(v)) if err != nil { return err } //fmt.Printf("%x: %d\n", k, bm.ToArray()) } if err != nil { return err } return nil } func fixTd(chaindata string) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, err := db.BeginRw(context.Background()) if err != nil { return err } defer tx.Rollback() c, err1 := tx.RwCursor(kv.Headers) if err1 != nil { return err1 } defer c.Close() var k, v []byte for k, v, err = c.First(); err == nil && k != nil; k, v, err = c.Next() { hv, herr := tx.GetOne(kv.HeaderTD, k) if herr != nil { return herr } if hv == nil { fmt.Printf("Missing TD record for %x, fixing\n", k) var header types.Header if err = rlp.DecodeBytes(v, &header); err != nil { return fmt.Errorf("decoding header from %x: %w", v, err) } if header.Number.Uint64() == 0 { continue } var parentK [40]byte binary.BigEndian.PutUint64(parentK[:], header.Number.Uint64()-1) copy(parentK[8:], header.ParentHash[:]) var parentTdRec []byte if parentTdRec, err = tx.GetOne(kv.HeaderTD, parentK[:]); err != nil { return fmt.Errorf("reading parentTd Rec for %d: %w", header.Number.Uint64(), err) } var parentTd big.Int if err = rlp.DecodeBytes(parentTdRec, &parentTd); err != nil { return fmt.Errorf("decoding parent Td record for block %d, from %x: %w", header.Number.Uint64(), parentTdRec, err) } var td big.Int td.Add(&parentTd, header.Difficulty) var newHv []byte if newHv, err = rlp.EncodeToBytes(&td); err != nil { return fmt.Errorf("encoding td record for block %d: %w", header.Number.Uint64(), err) } if err = tx.Put(kv.HeaderTD, k, newHv); err != nil { return err } } } if err != nil { return err } return tx.Commit() } func advanceExec(chaindata string) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, err := db.BeginRw(context.Background()) if err != nil { return err } defer tx.Rollback() stageExec, err := stages.GetStageProgress(tx, stages.Execution) if err != nil { return err } log.Info("ID exec", "progress", stageExec) if err = stages.SaveStageProgress(tx, stages.Execution, stageExec+1); err != nil { return err } stageExec, err = stages.GetStageProgress(tx, stages.Execution) if err != nil { return err } log.Info("ID exec", "changed to", stageExec) if err = tx.Commit(); err != nil { return err } return nil } func backExec(chaindata string) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, err := db.BeginRw(context.Background()) if err != nil { return err } defer tx.Rollback() stageExec, err := stages.GetStageProgress(tx, stages.Execution) if err != nil { return err } log.Info("ID exec", "progress", stageExec) if err = stages.SaveStageProgress(tx, stages.Execution, stageExec-1); err != nil { return err } stageExec, err = stages.GetStageProgress(tx, stages.Execution) if err != nil { return err } log.Info("ID exec", "changed to", stageExec) if err = tx.Commit(); err != nil { return err } return nil } func fixState(chaindata string) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, err := db.BeginRw(context.Background()) if err != nil { return err } defer tx.Rollback() c, err1 := tx.RwCursor(kv.HeaderCanonical) if err1 != nil { return err1 } defer c.Close() var prevHeaderKey [40]byte var k, v []byte for k, v, err = c.First(); err == nil && k != nil; k, v, err = c.Next() { var headerKey [40]byte copy(headerKey[:], k) copy(headerKey[8:], v) hv, herr := tx.GetOne(kv.Headers, headerKey[:]) if herr != nil { return herr } if hv == nil { return fmt.Errorf("missing header record for %x", headerKey) } var header types.Header if err = rlp.DecodeBytes(hv, &header); err != nil { return fmt.Errorf("decoding header from %x: %w", v, err) } if header.Number.Uint64() > 1 { var parentK [40]byte binary.BigEndian.PutUint64(parentK[:], header.Number.Uint64()-1) copy(parentK[8:], header.ParentHash[:]) if !bytes.Equal(parentK[:], prevHeaderKey[:]) { fmt.Printf("broken ancestry from %d %x (parent hash %x): prevKey %x\n", header.Number.Uint64(), v, header.ParentHash, prevHeaderKey) } } copy(prevHeaderKey[:], headerKey[:]) } if err != nil { return err } return tx.Commit() } func trimTxs(chaindata string) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, err := db.BeginRw(context.Background()) if err != nil { return err } defer tx.Rollback() lastTxId, err := tx.ReadSequence(kv.EthTx) if err != nil { return err } txs, err1 := tx.RwCursor(kv.EthTx) if err1 != nil { return err1 } defer txs.Close() bodies, err2 := tx.Cursor(kv.BlockBody) if err2 != nil { return err } defer bodies.Close() toDelete := roaring64.New() toDelete.AddRange(0, lastTxId) // Exclude transaction that are used, from the range for k, v, err := bodies.First(); k != nil; k, v, err = bodies.Next() { if err != nil { return err } var body types.BodyForStorage if err = rlp.DecodeBytes(v, &body); err != nil { return err } // Remove from the map toDelete.RemoveRange(body.BaseTxId, body.BaseTxId+uint64(body.TxAmount)) } fmt.Printf("Number of tx records to delete: %d\n", toDelete.GetCardinality()) // Takes 20min to iterate 1.4b toDelete2 := roaring64.New() var iterated int for k, _, err := txs.First(); k != nil; k, _, err = txs.Next() { if err != nil { return err } toDelete2.Add(binary.BigEndian.Uint64(k)) iterated++ if iterated%100_000_000 == 0 { fmt.Printf("Iterated %d\n", iterated) } } fmt.Printf("Number of tx records: %d\n", toDelete2.GetCardinality()) toDelete.And(toDelete2) fmt.Printf("Number of tx records to delete: %d\n", toDelete.GetCardinality()) fmt.Printf("Roaring size: %d\n", toDelete.GetSizeInBytes()) iter := toDelete.Iterator() for { var deleted int for iter.HasNext() { txId := iter.Next() var key [8]byte binary.BigEndian.PutUint64(key[:], txId) if err = txs.Delete(key[:]); err != nil { return err } deleted++ if deleted >= 10_000_000 { break } } if deleted == 0 { fmt.Printf("Nothing more to delete\n") break } fmt.Printf("Committing after deleting %d records\n", deleted) if err = tx.Commit(); err != nil { return err } txs.Close() tx, err = db.BeginRw(context.Background()) if err != nil { return err } defer tx.Rollback() txs, err = tx.RwCursor(kv.EthTx) if err != nil { return err } defer txs.Close() } return nil } func scanTxs(chaindata string) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, err := db.BeginRo(context.Background()) if err != nil { return err } defer tx.Rollback() c, err := tx.Cursor(kv.EthTx) if err != nil { return err } defer c.Close() trTypes := make(map[byte]int) trTypesAl := make(map[byte]int) for k, v, err := c.First(); k != nil; k, v, err = c.Next() { if err != nil { return err } var tr types.Transaction if tr, err = types.DecodeTransaction(rlp.NewStream(bytes.NewReader(v), 0)); err != nil { return err } if _, ok := trTypes[tr.Type()]; !ok { fmt.Printf("Example for type %d:\n%x\n", tr.Type(), v) } trTypes[tr.Type()]++ if tr.GetAccessList().StorageKeys() > 0 { if _, ok := trTypesAl[tr.Type()]; !ok { fmt.Printf("Example for type %d with AL:\n%x\n", tr.Type(), v) } trTypesAl[tr.Type()]++ } } fmt.Printf("Transaction types: %v\n", trTypes) return nil } func scanReceipts3(chaindata string, block uint64) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, err := db.BeginRw(context.Background()) if err != nil { return err } defer tx.Rollback() var key [8]byte var v []byte binary.BigEndian.PutUint64(key[:], block) if v, err = tx.GetOne(kv.Receipts, key[:]); err != nil { return err } fmt.Printf("%x\n", v) return nil } func scanReceipts2(chaindata string) error { f, err := os.Create("receipts.txt") if err != nil { return err } defer f.Close() w := bufio.NewWriter(f) defer w.Flush() dbdb := mdbx.MustOpen(chaindata) defer dbdb.Close() tx, err := dbdb.BeginRw(context.Background()) if err != nil { return err } defer tx.Rollback() blockNum, err := changeset.AvailableFrom(tx) if err != nil { return err } fixedCount := 0 logEvery := time.NewTicker(20 * time.Second) defer logEvery.Stop() var key [8]byte var v []byte for ; true; blockNum++ { select { default: case <-logEvery.C: log.Info("Scanned", "block", blockNum, "fixed", fixedCount) } var hash common.Hash if hash, err = rawdb.ReadCanonicalHash(tx, blockNum); err != nil { return err } if hash == (common.Hash{}) { break } binary.BigEndian.PutUint64(key[:], blockNum) if v, err = tx.GetOne(kv.Receipts, key[:]); err != nil { return err } var receipts types.Receipts if err = cbor.Unmarshal(&receipts, bytes.NewReader(v)); err == nil { broken := false for _, receipt := range receipts { if receipt.CumulativeGasUsed < 10000 { broken = true break } } if !broken { continue } } fmt.Fprintf(w, "%d %x\n", blockNum, v) fixedCount++ if fixedCount > 100 { break } } tx.Rollback() return nil } func devTx(chaindata string) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, err := db.BeginRo(context.Background()) if err != nil { return err } defer tx.Rollback() cc := tool.ChainConfig(tx) txn := types.NewTransaction(2, common.Address{}, uint256.NewInt(100), 100_000, uint256.NewInt(1), []byte{1}) signedTx, err := types.SignTx(txn, *types.LatestSigner(cc), core.DevnetSignPrivateKey) tool.Check(err) buf := bytes.NewBuffer(nil) err = signedTx.MarshalBinary(buf) tool.Check(err) fmt.Printf("%x\n", buf.Bytes()) return nil } func chainConfig(name string) error { var chainConfig *params.ChainConfig switch name { case "mainnet": chainConfig = params.MainnetChainConfig case "ropsten": chainConfig = params.RopstenChainConfig case "sepolia": chainConfig = params.SepoliaChainConfig case "rinkeby": chainConfig = params.RinkebyChainConfig case "goerli": chainConfig = params.GoerliChainConfig case "bsc": chainConfig = params.BSCChainConfig case "sokol": chainConfig = params.SokolChainConfig case "chapel": chainConfig = params.ChapelChainConfig case "rialto": chainConfig = params.RialtoChainConfig case "fermion": chainConfig = params.FermionChainConfig case "mumbai": chainConfig = params.MumbaiChainConfig case "bor-mainnet": chainConfig = params.BorMainnetChainConfig case "gnosis": chainConfig = params.GnosisChainConfig default: return fmt.Errorf("unknown name: %s", name) } f, err := os.Create(filepath.Join("params", "chainspecs", fmt.Sprintf("%s.json", name))) if err != nil { return err } w := bufio.NewWriter(f) encoder := json.NewEncoder(w) encoder.SetIndent("", " ") if err = encoder.Encode(chainConfig); err != nil { return err } if err = w.Flush(); err != nil { return err } if err = f.Close(); err != nil { return err } return nil } func keybytesToHex(str []byte) []byte { l := len(str)*2 + 1 var nibbles = make([]byte, l) for i, b := range str { nibbles[i*2] = b / 16 nibbles[i*2+1] = b % 16 } nibbles[l-1] = 16 return nibbles } func findPrefix(chaindata string) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, txErr := db.BeginRo(context.Background()) if txErr != nil { return txErr } defer tx.Rollback() c, err := tx.Cursor(kv.PlainState) if err != nil { return err } defer c.Close() var k []byte var e error prefix := common.FromHex("0x0901050b0c03") count := 0 for k, _, e = c.First(); k != nil && e == nil; k, _, e = c.Next() { if len(k) != 20 { continue } hash := crypto.Keccak256(k) nibbles := keybytesToHex(hash) if bytes.HasPrefix(nibbles, prefix) { fmt.Printf("addr = [%x], hash = [%x]\n", k, hash) break } count++ if count%1_000_000 == 0 { fmt.Printf("Searched %d records\n", count) } } if e != nil { return e } return nil } func findLogs(chaindata string, block uint64, blockTotal uint64) error { db := mdbx.MustOpen(chaindata) defer db.Close() tx, txErr := db.BeginRo(context.Background()) if txErr != nil { return txErr } defer tx.Rollback() logs, err := tx.Cursor(kv.Log) if err != nil { return err } defer logs.Close() reader := bytes.NewReader(nil) addrs := map[common.Address]int{} topics := map[string]int{} for k, v, err := logs.Seek(dbutils.LogKey(block, 0)); k != nil; k, v, err = logs.Next() { if err != nil { return err } blockNum := binary.BigEndian.Uint64(k[:8]) if blockNum >= block+blockTotal { break } var ll types.Logs reader.Reset(v) if err := cbor.Unmarshal(&ll, reader); err != nil { return fmt.Errorf("receipt unmarshal failed: %w, blocl=%d", err, blockNum) } for _, l := range ll { addrs[l.Address]++ for _, topic := range l.Topics { topics[fmt.Sprintf("%x | %x", l.Address, topic)]++ } } } addrsInv := map[int][]common.Address{} topicsInv := map[int][]string{} for a, c := range addrs { addrsInv[c] = append(addrsInv[c], a) } counts := make([]int, 0, len(addrsInv)) for c := range addrsInv { counts = append(counts, -c) } sort.Ints(counts) for i := 0; i < 10 && i < len(counts); i++ { as := addrsInv[-counts[i]] fmt.Printf("%d=%x\n", -counts[i], as) } for t, c := range topics { topicsInv[c] = append(topicsInv[c], t) } counts = make([]int, 0, len(topicsInv)) for c := range topicsInv { counts = append(counts, -c) } sort.Ints(counts) for i := 0; i < 10 && i < len(counts); i++ { as := topicsInv[-counts[i]] fmt.Printf("%d=%s\n", -counts[i], as) } return nil } func iterate(filename string, prefix string) error { pBytes := common.FromHex(prefix) efFilename := filename + ".ef" viFilename := filename + ".vi" vFilename := filename + ".v" efDecomp, err := compress.NewDecompressor(efFilename) if err != nil { return err } defer efDecomp.Close() viIndex, err := recsplit.OpenIndex(viFilename) if err != nil { return err } defer viIndex.Close() r := recsplit.NewIndexReader(viIndex) vDecomp, err := compress.NewDecompressor(vFilename) if err != nil { return err } defer vDecomp.Close() gv := vDecomp.MakeGetter() g := efDecomp.MakeGetter() for g.HasNext() { key, _ := g.NextUncompressed() if bytes.HasPrefix(key, pBytes) { val, _ := g.NextUncompressed() ef, _ := eliasfano32.ReadEliasFano(val) efIt := ef.Iterator() fmt.Printf("[%x] =>", key) for efIt.HasNext() { txNum := efIt.Next() var txKey [8]byte binary.BigEndian.PutUint64(txKey[:], txNum) offset := r.Lookup2(txKey[:], key) gv.Reset(offset) v, _ := gv.Next(nil) fmt.Printf(" %d", txNum) if len(v) == 0 { fmt.Printf("*") } } fmt.Printf("\n") } else { g.SkipUncompressed() } } return nil } func main() { debug.RaiseFdLimit() flag.Parse() log.Root().SetHandler(log.LvlFilterHandler(log.Lvl(*verbosity), log.StderrHandler)) if *cpuprofile != "" { f, err := os.Create(*cpuprofile) if err != nil { log.Error("could not create CPU profile", "err", err) return } if err := pprof.StartCPUProfile(f); err != nil { log.Error("could not start CPU profile", "err", err) return } defer pprof.StopCPUProfile() } go func() { if err := http.ListenAndServe("localhost:6060", nil); err != nil { log.Error("Failure in running pprof server", "err", err) } }() var err error switch *action { case "cfg": flow.TestGenCfg() case "testBlockHashes": testBlockHashes(*chaindata, *block, common.HexToHash(*hash)) case "readAccount": if err := readAccount(*chaindata, common.HexToAddress(*account)); err != nil { fmt.Printf("Error: %v\n", err) } case "nextIncarnation": nextIncarnation(*chaindata, common.HexToHash(*account)) case "dumpStorage": dumpStorage() case "current": printCurrentBlockNumber(*chaindata) case "bucket": printBucket(*chaindata) case "slice": dbSlice(*chaindata, *bucket, common.FromHex(*hash)) case "searchChangeSet": err = searchChangeSet(*chaindata, common.FromHex(*hash), uint64(*block)) case "searchStorageChangeSet": err = searchStorageChangeSet(*chaindata, common.FromHex(*hash), uint64(*block)) case "extractCode": err = extractCode(*chaindata) case "iterateOverCode": err = iterateOverCode(*chaindata) case "extractHeaders": err = extractHeaders(*chaindata, uint64(*block), int64(*blockTotal)) case "extractHashes": err = extractHashes(*chaindata, uint64(*block), int64(*blockTotal), *name) case "defrag": err = hackdb.Defrag() case "textInfo": err = hackdb.TextInfo(*chaindata, &strings.Builder{}) case "extractBodies": err = extractBodies(*chaindata) case "repairCurrent": repairCurrent() case "printTxHashes": printTxHashes(*chaindata, uint64(*block)) case "snapSizes": err = snapSizes(*chaindata) case "readCallTraces": err = readCallTraces(*chaindata, uint64(*block)) case "fixTd": err = fixTd(*chaindata) case "advanceExec": err = advanceExec(*chaindata) case "backExec": err = backExec(*chaindata) case "fixState": err = fixState(*chaindata) case "trimTxs": err = trimTxs(*chaindata) case "scanTxs": err = scanTxs(*chaindata) case "scanReceipts2": err = scanReceipts2(*chaindata) case "scanReceipts3": err = scanReceipts3(*chaindata, uint64(*block)) case "devTx": err = devTx(*chaindata) case "chainConfig": err = chainConfig(*name) case "findPrefix": err = findPrefix(*chaindata) case "findLogs": err = findLogs(*chaindata, uint64(*block), uint64(*blockTotal)) case "iterate": err = iterate(*chaindata, *account) } if err != nil { fmt.Printf("Error: %v\n", err) } }