mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-07 11:32:20 +00:00
de26634d8d
* use trace for TEVM status * restore testdata * fix getting contract from plain state * unwind * linters * reproducible build debug * debug * debug * remove debug
443 lines
13 KiB
Go
443 lines
13 KiB
Go
package main
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"encoding/binary"
|
|
"fmt"
|
|
"math/big"
|
|
"os"
|
|
"os/exec"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/holiman/uint256"
|
|
"github.com/ledgerwatch/erigon/accounts/abi/bind"
|
|
"github.com/ledgerwatch/erigon/accounts/abi/bind/backends"
|
|
"github.com/ledgerwatch/erigon/cmd/pics/contracts"
|
|
"github.com/ledgerwatch/erigon/common"
|
|
"github.com/ledgerwatch/erigon/common/dbutils"
|
|
"github.com/ledgerwatch/erigon/core"
|
|
"github.com/ledgerwatch/erigon/core/types"
|
|
"github.com/ledgerwatch/erigon/crypto"
|
|
"github.com/ledgerwatch/erigon/ethdb"
|
|
"github.com/ledgerwatch/erigon/log"
|
|
"github.com/ledgerwatch/erigon/params"
|
|
"github.com/ledgerwatch/erigon/turbo/stages"
|
|
"github.com/ledgerwatch/erigon/turbo/trie"
|
|
"github.com/ledgerwatch/erigon/visual"
|
|
)
|
|
|
|
/*func statePicture(t *trie.Trie, number int, keyCompression int, codeCompressed bool, valCompressed bool,
|
|
quadTrie bool, quadColors bool, highlights [][]byte) (*trie.Trie, error) {
|
|
filename := fmt.Sprintf("state_%d.dot", number)
|
|
f, err := os.Create(filename)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
indexColors := visual.HexIndexColors
|
|
fontColors := visual.HexFontColors
|
|
if quadTrie {
|
|
t = trie.HexToQuad(t)
|
|
}
|
|
if quadColors {
|
|
indexColors = visual.QuadIndexColors
|
|
fontColors = visual.QuadFontColors
|
|
}
|
|
visual.StartGraph(f, false)
|
|
trie.Visual(t, f, &trie.VisualOpts{
|
|
Highlights: highlights,
|
|
IndexColors: indexColors,
|
|
FontColors: fontColors,
|
|
Values: true,
|
|
CutTerminals: keyCompression,
|
|
CodeCompressed: codeCompressed,
|
|
ValCompressed: valCompressed,
|
|
ValHex: true,
|
|
})
|
|
visual.EndGraph(f)
|
|
if err := f.Close(); err != nil {
|
|
return nil, err
|
|
}
|
|
//nolint:gosec
|
|
cmd := exec.Command("dot", "-Tpng:gd", "-o"+dot2png(filename), filename)
|
|
if output, err := cmd.CombinedOutput(); err != nil {
|
|
fmt.Printf("error: %v, output: %s\n", err, output)
|
|
}
|
|
return t, nil
|
|
}*/
|
|
|
|
var bucketLabels = map[string]string{
|
|
dbutils.BlockReceiptsPrefix: "Receipts",
|
|
dbutils.Log: "Event Logs",
|
|
dbutils.AccountsHistoryBucket: "History Of Accounts",
|
|
dbutils.StorageHistoryBucket: "History Of Storage",
|
|
dbutils.HeadersBucket: "Headers",
|
|
dbutils.HeaderCanonicalBucket: "Canonical headers",
|
|
dbutils.HeaderTDBucket: "Headers TD",
|
|
dbutils.BlockBodyPrefix: "Block Bodies",
|
|
dbutils.HeaderNumberBucket: "Header Numbers",
|
|
dbutils.TxLookupPrefix: "Transaction Index",
|
|
dbutils.CodeBucket: "Code Of Contracts",
|
|
dbutils.SyncStageProgress: "Sync Progress",
|
|
dbutils.PlainStateBucket: "Plain State",
|
|
dbutils.HashedAccountsBucket: "Hashed Accounts",
|
|
dbutils.HashedStorageBucket: "Hashed Storage",
|
|
dbutils.TrieOfAccountsBucket: "Intermediate Hashes Of Accounts",
|
|
dbutils.TrieOfStorageBucket: "Intermediate Hashes Of Storage",
|
|
dbutils.SyncStageUnwind: "Unwind",
|
|
dbutils.AccountChangeSetBucket: "Account Changes",
|
|
dbutils.StorageChangeSetBucket: "Storage Changes",
|
|
dbutils.IncarnationMapBucket: "Incarnations",
|
|
dbutils.Senders: "Transaction Senders",
|
|
dbutils.ContractTEVMCodeBucket: "Contract TEVM code",
|
|
}
|
|
|
|
/*dbutils.PlainContractCodeBucket,
|
|
dbutils.CodeBucket,
|
|
dbutils.AccountsHistoryBucket,
|
|
dbutils.StorageHistoryBucket,
|
|
dbutils.TxLookupPrefix,*/
|
|
|
|
func hexPalette() error {
|
|
filename := "hex_palette.dot"
|
|
f, err := os.Create(filename)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
visual.StartGraph(f, true)
|
|
p := common.FromHex("0x000102030405060708090a0b0c0d0e0f")
|
|
visual.Horizontal(f, p, len(p), "p", visual.HexIndexColors, visual.HexFontColors, 0)
|
|
visual.EndGraph(f)
|
|
if err := f.Close(); err != nil {
|
|
return err
|
|
}
|
|
//nolint:gosec
|
|
cmd := exec.Command("dot", "-Tpng:gd", "-o"+dot2png(filename), filename)
|
|
if output, err := cmd.CombinedOutput(); err != nil {
|
|
fmt.Printf("error: %v, output: %s\n", err, output)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func stateDatabaseComparison(first ethdb.RwKV, second ethdb.RwKV, number int) error {
|
|
filename := fmt.Sprintf("changes_%d.dot", number)
|
|
f, err := os.Create(filename)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
i := 0
|
|
visual.StartGraph(f, true)
|
|
m := make(map[string][]int)
|
|
noValues := make(map[int]struct{})
|
|
perBucketFiles := make(map[string]*os.File)
|
|
|
|
if err = second.View(context.Background(), func(readTx ethdb.Tx) error {
|
|
return first.View(context.Background(), func(firstTx ethdb.Tx) error {
|
|
for bucketName := range bucketLabels {
|
|
bucketName := bucketName
|
|
c, err := readTx.Cursor(bucketName)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if err2 := ethdb.ForEach(c, func(k, v []byte) (bool, error) {
|
|
if firstV, _ := firstTx.GetOne(bucketName, k); firstV != nil && bytes.Equal(v, firstV) {
|
|
// Skip the record that is the same as in the first Db
|
|
return true, nil
|
|
}
|
|
// Produce pair of nodes
|
|
keyKeyBytes := &trie.Keybytes{
|
|
Data: k,
|
|
Odd: false,
|
|
Terminating: false,
|
|
}
|
|
valKeyBytes := &trie.Keybytes{
|
|
Data: v,
|
|
Odd: false,
|
|
Terminating: false,
|
|
}
|
|
val := valKeyBytes.ToHex()
|
|
key := keyKeyBytes.ToHex()
|
|
var f1 *os.File
|
|
var ok bool
|
|
if f1, ok = perBucketFiles[bucketName]; !ok {
|
|
f1, err = os.Create(fmt.Sprintf("changes_%d_%s.dot", number, strings.ReplaceAll(bucketLabels[bucketName], " ", "")))
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
visual.StartGraph(f1, true)
|
|
var clusterLabel string
|
|
var ok bool
|
|
if clusterLabel, ok = bucketLabels[bucketName]; !ok {
|
|
clusterLabel = bucketName
|
|
}
|
|
visual.StartCluster(f1, 0, clusterLabel)
|
|
perBucketFiles[bucketName] = f1
|
|
}
|
|
visual.Horizontal(f1, key, len(key), fmt.Sprintf("k_%d", i), visual.HexIndexColors, visual.HexFontColors, 0)
|
|
if len(val) > 0 {
|
|
if len(val) > 64 {
|
|
visual.HexBox(f1, fmt.Sprintf("v_%d", i), val, 64, false /*compresses*/, true /*highlighted*/)
|
|
} else {
|
|
visual.Horizontal(f1, val, len(val), fmt.Sprintf("v_%d", i), visual.HexIndexColors, visual.HexFontColors, 0)
|
|
}
|
|
// Produce edge
|
|
fmt.Fprintf(f1, "k_%d -> v_%d;\n", i, i)
|
|
} else {
|
|
noValues[i] = struct{}{}
|
|
}
|
|
visual.Horizontal(f, key, 0, fmt.Sprintf("k_%d", i), visual.HexIndexColors, visual.HexFontColors, 0)
|
|
if len(val) > 0 {
|
|
if len(val) > 64 {
|
|
visual.HexBox(f, fmt.Sprintf("v_%d", i), val, 64, false /*compressed*/, false /*highlighted*/)
|
|
} else {
|
|
visual.Horizontal(f, val, 0, fmt.Sprintf("v_%d", i), visual.HexIndexColors, visual.HexFontColors, 0)
|
|
}
|
|
// Produce edge
|
|
fmt.Fprintf(f, "k_%d -> v_%d;\n", i, i)
|
|
} else {
|
|
noValues[i] = struct{}{}
|
|
}
|
|
lst := m[bucketName]
|
|
lst = append(lst, i)
|
|
m[bucketName] = lst
|
|
i++
|
|
return true, nil
|
|
}); err2 != nil {
|
|
return err2
|
|
}
|
|
}
|
|
return nil
|
|
})
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
n := 0
|
|
for prefix, lst := range m {
|
|
var clusterLabel string
|
|
var ok bool
|
|
if clusterLabel, ok = bucketLabels[prefix]; !ok {
|
|
clusterLabel = prefix
|
|
}
|
|
if len(lst) == 0 {
|
|
continue
|
|
}
|
|
visual.StartCluster(f, n, clusterLabel)
|
|
for _, item := range lst {
|
|
if _, ok1 := noValues[item]; ok1 {
|
|
fmt.Fprintf(f, "k_%d;", item)
|
|
} else {
|
|
fmt.Fprintf(f, "k_%d;v_%d;", item, item)
|
|
}
|
|
}
|
|
fmt.Fprintf(f, "\n")
|
|
visual.EndCluster(f)
|
|
n++
|
|
}
|
|
visual.EndGraph(f)
|
|
if err := f.Close(); err != nil {
|
|
return err
|
|
}
|
|
//nolint:gosec
|
|
cmd := exec.Command("dot", "-Tpng:gd", "-o"+dot2png(filename), filename)
|
|
if output, err := cmd.CombinedOutput(); err != nil {
|
|
fmt.Printf("error: %v, output: %s\n", err, output)
|
|
}
|
|
for _, f1 := range perBucketFiles {
|
|
fmt.Fprintf(f1, "\n")
|
|
visual.EndCluster(f1)
|
|
visual.EndGraph(f1)
|
|
if err := f1.Close(); err != nil {
|
|
return err
|
|
}
|
|
//nolint:gosec
|
|
cmd := exec.Command("dot", "-Tpng:gd", "-o"+dot2png(f1.Name()), f1.Name())
|
|
if output, err := cmd.CombinedOutput(); err != nil {
|
|
fmt.Printf("error: %v, output: %s\n", err, output)
|
|
}
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func dot2png(dotFileName string) string {
|
|
return strings.TrimSuffix(dotFileName, filepath.Ext(dotFileName)) + ".png"
|
|
}
|
|
|
|
func initialState1() error {
|
|
defer log.Root().SetHandler(log.Root().GetHandler())
|
|
log.Root().SetHandler(log.LvlFilterHandler(log.LvlInfo, log.StreamHandler(os.Stderr, log.TerminalFormat(true))))
|
|
fmt.Printf("Initial state 1\n")
|
|
// Configure and generate a sample block chain
|
|
var (
|
|
key, _ = crypto.HexToECDSA("b71c71a67e1177ad4e901695e1b4b9ee17ae16c6668d313eac2f96dbcda3f291")
|
|
key1, _ = crypto.HexToECDSA("49a7b37aa6f6645917e7b807e9d1c00d4fa71f18343b0d4122a4d2df64dd6fee")
|
|
key2, _ = crypto.HexToECDSA("8a1f9a8f95be41cd7ccb6168179afb4504aefe388d1e14474d32c45c72ce7b7a")
|
|
address = crypto.PubkeyToAddress(key.PublicKey)
|
|
address1 = crypto.PubkeyToAddress(key1.PublicKey)
|
|
address2 = crypto.PubkeyToAddress(key2.PublicKey)
|
|
theAddr = common.Address{1}
|
|
gspec = &core.Genesis{
|
|
Config: params.AllEthashProtocolChanges,
|
|
Alloc: core.GenesisAlloc{
|
|
address: {Balance: big.NewInt(9000000000000000000)},
|
|
address1: {Balance: big.NewInt(200000000000000000)},
|
|
address2: {Balance: big.NewInt(300000000000000000)},
|
|
},
|
|
GasLimit: 10000000,
|
|
}
|
|
// this code generates a log
|
|
signer = types.MakeSigner(params.AllEthashProtocolChanges, 1)
|
|
)
|
|
m := stages.MockWithGenesis(nil, gspec, key)
|
|
defer m.DB.Close()
|
|
|
|
contractBackend := backends.NewSimulatedBackendWithConfig(gspec.Alloc, gspec.Config, gspec.GasLimit)
|
|
defer contractBackend.Close()
|
|
transactOpts := bind.NewKeyedTransactor(key)
|
|
transactOpts1 := bind.NewKeyedTransactor(key1)
|
|
transactOpts2 := bind.NewKeyedTransactor(key2)
|
|
|
|
var tokenContract *contracts.Token
|
|
// We generate the blocks without plainstant because it's not supported in core.GenerateChain
|
|
chain, err := core.GenerateChain(m.ChainConfig, m.Genesis, m.Engine, m.DB, 8, func(i int, block *core.BlockGen) {
|
|
var (
|
|
tx types.Transaction
|
|
txs []types.Transaction
|
|
err error
|
|
)
|
|
|
|
ctx := gspec.Config.WithEIPsFlags(context.Background(), block.Number().Uint64())
|
|
switch i {
|
|
case 0:
|
|
tx, err = types.SignTx(types.NewTransaction(0, theAddr, uint256.NewInt(1000000000000000), 21000, new(uint256.Int), nil), *signer, key)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
err = contractBackend.SendTransaction(ctx, tx)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
case 1:
|
|
tx, err = types.SignTx(types.NewTransaction(1, theAddr, uint256.NewInt(1000000000000000), 21000, new(uint256.Int), nil), *signer, key)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
err = contractBackend.SendTransaction(ctx, tx)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
case 2:
|
|
_, tx, tokenContract, err = contracts.DeployToken(transactOpts, contractBackend, address1)
|
|
case 3:
|
|
tx, err = tokenContract.Mint(transactOpts1, address2, big.NewInt(10))
|
|
case 4:
|
|
tx, err = tokenContract.Transfer(transactOpts2, address, big.NewInt(3))
|
|
case 5:
|
|
// Multiple transactions sending small amounts of ether to various accounts
|
|
var j uint64
|
|
var toAddr common.Address
|
|
nonce := block.TxNonce(address)
|
|
for j = 1; j <= 32; j++ {
|
|
binary.BigEndian.PutUint64(toAddr[:], j)
|
|
tx, err = types.SignTx(types.NewTransaction(nonce, toAddr, uint256.NewInt(1000000000000000), 21000, new(uint256.Int), nil), *signer, key)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
err = contractBackend.SendTransaction(ctx, tx)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
txs = append(txs, tx)
|
|
nonce++
|
|
}
|
|
case 6:
|
|
_, tx, tokenContract, err = contracts.DeployToken(transactOpts, contractBackend, address1)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
txs = append(txs, tx)
|
|
tx, err = tokenContract.Mint(transactOpts1, address2, big.NewInt(100))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
txs = append(txs, tx)
|
|
// Multiple transactions sending small amounts of ether to various accounts
|
|
var j uint64
|
|
var toAddr common.Address
|
|
for j = 1; j <= 32; j++ {
|
|
binary.BigEndian.PutUint64(toAddr[:], j)
|
|
tx, err = tokenContract.Transfer(transactOpts2, toAddr, big.NewInt(1))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
txs = append(txs, tx)
|
|
}
|
|
case 7:
|
|
var toAddr common.Address
|
|
nonce := block.TxNonce(address)
|
|
binary.BigEndian.PutUint64(toAddr[:], 4)
|
|
tx, err = types.SignTx(types.NewTransaction(nonce, toAddr, uint256.NewInt(1000000000000000), 21000, new(uint256.Int), nil), *signer, key)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
err = contractBackend.SendTransaction(ctx, tx)
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
txs = append(txs, tx)
|
|
binary.BigEndian.PutUint64(toAddr[:], 12)
|
|
tx, err = tokenContract.Transfer(transactOpts2, toAddr, big.NewInt(1))
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
txs = append(txs, tx)
|
|
}
|
|
|
|
if err != nil {
|
|
panic(err)
|
|
}
|
|
if txs == nil && tx != nil {
|
|
txs = append(txs, tx)
|
|
}
|
|
|
|
for _, tx := range txs {
|
|
block.AddTx(tx)
|
|
}
|
|
contractBackend.Commit()
|
|
}, true)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
m2 := stages.MockWithGenesis(nil, gspec, key)
|
|
defer m2.DB.Close()
|
|
|
|
if err = hexPalette(); err != nil {
|
|
return err
|
|
}
|
|
|
|
emptyKv := ethdb.NewMemKV()
|
|
if err = stateDatabaseComparison(emptyKv, m.DB, 0); err != nil {
|
|
return err
|
|
}
|
|
defer emptyKv.Close()
|
|
|
|
// BLOCKS
|
|
|
|
for i := 0; i < chain.Length; i++ {
|
|
if err = m2.InsertChain(chain.Slice(i, i+1)); err != nil {
|
|
return err
|
|
}
|
|
if err = stateDatabaseComparison(m.DB, m2.DB, i+1); err != nil {
|
|
return err
|
|
}
|
|
if err = m.InsertChain(chain.Slice(i, i+1)); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if err = stateDatabaseComparison(emptyKv, m.DB, 9); err != nil {
|
|
return err
|
|
}
|
|
return nil
|
|
}
|