2019-05-27 13:51:49 +00:00
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bufio"
|
|
|
|
"bytes"
|
|
|
|
"encoding/binary"
|
|
|
|
"flag"
|
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"log"
|
|
|
|
"math/big"
|
|
|
|
"os"
|
|
|
|
"runtime/pprof"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/ledgerwatch/turbo-geth/common/dbutils"
|
|
|
|
|
|
|
|
"github.com/ledgerwatch/bolt"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/common"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/consensus/ethash"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/core"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/core/rawdb"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/core/state"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/core/types"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/core/vm"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/crypto"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/ethdb"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/params"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/rlp"
|
|
|
|
"github.com/ledgerwatch/turbo-geth/trie"
|
|
|
|
"github.com/wcharczuk/go-chart"
|
|
|
|
"github.com/wcharczuk/go-chart/util"
|
|
|
|
)
|
|
|
|
|
|
|
|
var emptyCodeHash = crypto.Keccak256(nil)
|
|
|
|
|
|
|
|
var action = flag.String("action", "", "action to execute")
|
|
|
|
var cpuprofile = flag.String("cpuprofile", "", "write cpu profile `file`")
|
|
|
|
var reset = flag.Int("reset", -1, "reset to given block number")
|
|
|
|
var rewind = flag.Int("rewind", 1, "rewind to given number of blocks")
|
|
|
|
var block = flag.Int("block", 1, "specifies a block number for operation")
|
|
|
|
var account = flag.String("account", "0x", "specifies account to investigate")
|
|
|
|
var name = flag.String("name", "", "name to add to the file names")
|
2019-10-18 12:11:50 +00:00
|
|
|
var chaindata = flag.String("chaindata", "chaindata", "path to the chaindata database file")
|
2019-11-21 15:56:39 +00:00
|
|
|
var hash = flag.String("hash", "0x00", "image for preimage or state root for testBlockHashes action")
|
2019-05-27 13:51:49 +00:00
|
|
|
|
|
|
|
func bucketList(db *bolt.DB) [][]byte {
|
|
|
|
bucketList := [][]byte{}
|
|
|
|
err := db.View(func(tx *bolt.Tx) error {
|
|
|
|
err := tx.ForEach(func(name []byte, b *bolt.Bucket) error {
|
|
|
|
if len(name) == 20 || bytes.Equal(name, dbutils.AccountsBucket) {
|
|
|
|
n := make([]byte, len(name))
|
|
|
|
copy(n, name)
|
|
|
|
bucketList = append(bucketList, n)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return err
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("Could view db: %s", err))
|
|
|
|
}
|
|
|
|
return bucketList
|
|
|
|
}
|
|
|
|
|
|
|
|
// prefixLen returns the length of the common prefix of a and b.
|
|
|
|
func prefixLen(a, b []byte) int {
|
|
|
|
var i, length = 0, len(a)
|
|
|
|
if len(b) < length {
|
|
|
|
length = len(b)
|
|
|
|
}
|
|
|
|
for ; i < length; i++ {
|
|
|
|
if a[i] != b[i] {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return i
|
|
|
|
}
|
|
|
|
|
|
|
|
func check(e error) {
|
|
|
|
if e != nil {
|
|
|
|
panic(e)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func parseFloat64(str string) float64 {
|
|
|
|
v, _ := strconv.ParseFloat(str, 64)
|
|
|
|
return v
|
|
|
|
}
|
|
|
|
|
|
|
|
func readData(filename string) (blocks []float64, hours []float64, dbsize []float64, trienodes []float64, heap []float64) {
|
|
|
|
err := util.File.ReadByLines(filename, func(line string) error {
|
|
|
|
parts := strings.Split(line, ",")
|
|
|
|
blocks = append(blocks, parseFloat64(strings.Trim(parts[0], " ")))
|
|
|
|
hours = append(hours, parseFloat64(strings.Trim(parts[1], " ")))
|
|
|
|
dbsize = append(dbsize, parseFloat64(strings.Trim(parts[2], " ")))
|
|
|
|
trienodes = append(trienodes, parseFloat64(strings.Trim(parts[3], " ")))
|
|
|
|
heap = append(heap, parseFloat64(strings.Trim(parts[4], " ")))
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
fmt.Println(err.Error())
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
func notables() []chart.GridLine {
|
|
|
|
return []chart.GridLine{
|
|
|
|
{Value: 1.0},
|
|
|
|
{Value: 2.0},
|
|
|
|
{Value: 3.0},
|
|
|
|
{Value: 4.0},
|
|
|
|
{Value: 5.0},
|
|
|
|
{Value: 6.0},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func days() []chart.GridLine {
|
|
|
|
return []chart.GridLine{
|
|
|
|
{Value: 24.0},
|
|
|
|
{Value: 48.0},
|
|
|
|
{Value: 72.0},
|
|
|
|
{Value: 96.0},
|
|
|
|
{Value: 120.0},
|
|
|
|
{Value: 144.0},
|
|
|
|
{Value: 168.0},
|
|
|
|
{Value: 192.0},
|
|
|
|
{Value: 216.0},
|
|
|
|
{Value: 240.0},
|
|
|
|
{Value: 264.0},
|
|
|
|
{Value: 288.0},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func mychart() {
|
|
|
|
blocks, hours, dbsize, trienodes, heap := readData("geth.csv")
|
|
|
|
//blocks0, hours0, _, _, _ := readData("geth.csv")
|
|
|
|
mainSeries := &chart.ContinuousSeries{
|
|
|
|
Name: "Cumulative sync time (SSD)",
|
|
|
|
Style: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.ColorBlue,
|
|
|
|
FillColor: chart.ColorBlue.WithAlpha(100),
|
|
|
|
},
|
|
|
|
XValues: blocks,
|
|
|
|
YValues: hours,
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
hddSeries := &chart.ContinuousSeries{
|
|
|
|
Name: "Cumulative sync time (HDD)",
|
|
|
|
Style: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.ColorRed,
|
|
|
|
FillColor: chart.ColorRed.WithAlpha(100),
|
|
|
|
},
|
|
|
|
XValues: blocks0,
|
|
|
|
YValues: hours0,
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
dbsizeSeries := &chart.ContinuousSeries{
|
|
|
|
Name: "Database size",
|
|
|
|
Style: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.ColorBlack,
|
|
|
|
},
|
|
|
|
YAxis: chart.YAxisSecondary,
|
|
|
|
XValues: blocks,
|
|
|
|
YValues: dbsize,
|
|
|
|
}
|
|
|
|
|
|
|
|
graph1 := chart.Chart{
|
|
|
|
Width: 1280,
|
|
|
|
Height: 720,
|
|
|
|
Background: chart.Style{
|
|
|
|
Padding: chart.Box{
|
|
|
|
Top: 50,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
YAxis: chart.YAxis{
|
|
|
|
Name: "Elapsed time",
|
|
|
|
NameStyle: chart.StyleShow(),
|
|
|
|
Style: chart.StyleShow(),
|
|
|
|
TickStyle: chart.Style{
|
|
|
|
TextRotationDegrees: 45.0,
|
|
|
|
},
|
|
|
|
ValueFormatter: func(v interface{}) string {
|
|
|
|
return fmt.Sprintf("%d h", int(v.(float64)))
|
|
|
|
},
|
|
|
|
GridMajorStyle: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.ColorBlue,
|
|
|
|
StrokeWidth: 1.0,
|
|
|
|
},
|
|
|
|
GridLines: days(),
|
|
|
|
},
|
|
|
|
YAxisSecondary: chart.YAxis{
|
|
|
|
NameStyle: chart.StyleShow(),
|
|
|
|
Style: chart.StyleShow(),
|
|
|
|
TickStyle: chart.Style{
|
|
|
|
TextRotationDegrees: 45.0,
|
|
|
|
},
|
|
|
|
ValueFormatter: func(v interface{}) string {
|
|
|
|
return fmt.Sprintf("%d G", int(v.(float64)))
|
|
|
|
},
|
|
|
|
},
|
|
|
|
XAxis: chart.XAxis{
|
|
|
|
Name: "Blocks, million",
|
|
|
|
Style: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
},
|
|
|
|
ValueFormatter: func(v interface{}) string {
|
|
|
|
return fmt.Sprintf("%.3fm", v.(float64))
|
|
|
|
},
|
|
|
|
GridMajorStyle: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.ColorAlternateGray,
|
|
|
|
StrokeWidth: 1.0,
|
|
|
|
},
|
|
|
|
GridLines: notables(),
|
|
|
|
},
|
|
|
|
Series: []chart.Series{
|
|
|
|
mainSeries,
|
|
|
|
//hddSeries,
|
|
|
|
dbsizeSeries,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
graph1.Elements = []chart.Renderable{chart.LegendThin(&graph1)}
|
|
|
|
|
|
|
|
buffer := bytes.NewBuffer([]byte{})
|
|
|
|
err := graph1.Render(chart.PNG, buffer)
|
|
|
|
check(err)
|
|
|
|
err = ioutil.WriteFile("chart1.png", buffer.Bytes(), 0644)
|
|
|
|
check(err)
|
|
|
|
|
|
|
|
heapSeries := &chart.ContinuousSeries{
|
|
|
|
Name: "Allocated heap",
|
|
|
|
Style: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.ColorYellow,
|
|
|
|
FillColor: chart.ColorYellow.WithAlpha(100),
|
|
|
|
},
|
|
|
|
XValues: blocks,
|
|
|
|
YValues: heap,
|
|
|
|
}
|
|
|
|
trienodesSeries := &chart.ContinuousSeries{
|
|
|
|
Name: "Trie nodes",
|
|
|
|
Style: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.ColorGreen,
|
|
|
|
},
|
|
|
|
YAxis: chart.YAxisSecondary,
|
|
|
|
XValues: blocks,
|
|
|
|
YValues: trienodes,
|
|
|
|
}
|
|
|
|
graph2 := chart.Chart{
|
|
|
|
Width: 1280,
|
|
|
|
Height: 720,
|
|
|
|
Background: chart.Style{
|
|
|
|
Padding: chart.Box{
|
|
|
|
Top: 50,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
YAxis: chart.YAxis{
|
|
|
|
Name: "Allocated heap",
|
|
|
|
NameStyle: chart.StyleShow(),
|
|
|
|
Style: chart.StyleShow(),
|
|
|
|
TickStyle: chart.Style{
|
|
|
|
TextRotationDegrees: 45.0,
|
|
|
|
},
|
|
|
|
ValueFormatter: func(v interface{}) string {
|
|
|
|
return fmt.Sprintf("%.1f G", v.(float64))
|
|
|
|
},
|
|
|
|
GridMajorStyle: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.ColorYellow,
|
|
|
|
StrokeWidth: 1.0,
|
|
|
|
},
|
|
|
|
GridLines: days(),
|
|
|
|
},
|
|
|
|
YAxisSecondary: chart.YAxis{
|
|
|
|
NameStyle: chart.StyleShow(),
|
|
|
|
Style: chart.StyleShow(),
|
|
|
|
TickStyle: chart.Style{
|
|
|
|
TextRotationDegrees: 45.0,
|
|
|
|
},
|
|
|
|
ValueFormatter: func(v interface{}) string {
|
|
|
|
return fmt.Sprintf("%.1f m", v.(float64))
|
|
|
|
},
|
|
|
|
},
|
|
|
|
XAxis: chart.XAxis{
|
|
|
|
Name: "Blocks, million",
|
|
|
|
Style: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
},
|
|
|
|
ValueFormatter: func(v interface{}) string {
|
|
|
|
return fmt.Sprintf("%.3fm", v.(float64))
|
|
|
|
},
|
|
|
|
GridMajorStyle: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.ColorAlternateGray,
|
|
|
|
StrokeWidth: 1.0,
|
|
|
|
},
|
|
|
|
GridLines: notables(),
|
|
|
|
},
|
|
|
|
Series: []chart.Series{
|
|
|
|
heapSeries,
|
|
|
|
trienodesSeries,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
graph2.Elements = []chart.Renderable{chart.LegendThin(&graph2)}
|
|
|
|
buffer.Reset()
|
|
|
|
err = graph2.Render(chart.PNG, buffer)
|
|
|
|
check(err)
|
|
|
|
err = ioutil.WriteFile("chart2.png", buffer.Bytes(), 0644)
|
|
|
|
check(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func accountSavings(db *bolt.DB) (int, int) {
|
|
|
|
emptyRoots := 0
|
|
|
|
emptyCodes := 0
|
|
|
|
db.View(func(tx *bolt.Tx) error {
|
|
|
|
b := tx.Bucket(dbutils.AccountsBucket)
|
|
|
|
c := b.Cursor()
|
|
|
|
for k, v := c.First(); k != nil; k, v = c.Next() {
|
|
|
|
if bytes.Contains(v, trie.EmptyRoot.Bytes()) {
|
|
|
|
emptyRoots++
|
|
|
|
}
|
|
|
|
if bytes.Contains(v, emptyCodeHash) {
|
|
|
|
emptyCodes++
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return emptyRoots, emptyCodes
|
|
|
|
}
|
|
|
|
|
|
|
|
func allBuckets(db *bolt.DB) [][]byte {
|
|
|
|
bucketList := [][]byte{}
|
|
|
|
err := db.View(func(tx *bolt.Tx) error {
|
|
|
|
err := tx.ForEach(func(name []byte, b *bolt.Bucket) error {
|
|
|
|
n := make([]byte, len(name))
|
|
|
|
copy(n, name)
|
|
|
|
bucketList = append(bucketList, n)
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
return err
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("Could view db: %s", err))
|
|
|
|
}
|
|
|
|
return bucketList
|
|
|
|
}
|
|
|
|
|
|
|
|
func printBuckets(db *bolt.DB) {
|
|
|
|
bucketList := allBuckets(db)
|
|
|
|
for _, bucket := range bucketList {
|
|
|
|
fmt.Printf("%s\n", bucket)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func bucketStats(db *bolt.DB) {
|
|
|
|
bucketList := allBuckets(db)
|
|
|
|
storageStats := new(bolt.BucketStats)
|
|
|
|
hStorageStats := new(bolt.BucketStats)
|
|
|
|
fmt.Printf(",BranchPageN,BranchOverflowN,LeafPageN,LeafOverflowN,KeyN,Depth,BranchAlloc,BranchInuse,LeafAlloc,LeafInuse,BucketN,InlineBucketN,InlineBucketInuse\n")
|
|
|
|
db.View(func(tx *bolt.Tx) error {
|
|
|
|
for _, bucket := range bucketList {
|
|
|
|
b := tx.Bucket(bucket)
|
|
|
|
bs := b.Stats()
|
|
|
|
if len(bucket) == 20 {
|
|
|
|
storageStats.Add(bs)
|
|
|
|
} else if len(bucket) == 21 {
|
|
|
|
hStorageStats.Add(bs)
|
|
|
|
} else {
|
|
|
|
fmt.Printf("%s,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d\n", string(bucket),
|
|
|
|
bs.BranchPageN, bs.BranchOverflowN, bs.LeafPageN, bs.LeafOverflowN, bs.KeyN, bs.Depth, bs.BranchAlloc, bs.BranchInuse,
|
|
|
|
bs.LeafAlloc, bs.LeafInuse, bs.BucketN, bs.InlineBucketN, bs.InlineBucketInuse)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
bs := *storageStats
|
|
|
|
fmt.Printf("%s,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d\n", "Contract Storage",
|
|
|
|
bs.BranchPageN, bs.BranchOverflowN, bs.LeafPageN, bs.LeafOverflowN, bs.KeyN, bs.Depth, bs.BranchAlloc, bs.BranchInuse,
|
|
|
|
bs.LeafAlloc, bs.LeafInuse, bs.BucketN, bs.InlineBucketN, bs.InlineBucketInuse)
|
|
|
|
bs = *hStorageStats
|
|
|
|
fmt.Printf("%s,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d,%d\n", "Contract hStorage",
|
|
|
|
bs.BranchPageN, bs.BranchOverflowN, bs.LeafPageN, bs.LeafOverflowN, bs.KeyN, bs.Depth, bs.BranchAlloc, bs.BranchInuse,
|
|
|
|
bs.LeafAlloc, bs.LeafInuse, bs.BucketN, bs.InlineBucketN, bs.InlineBucketInuse)
|
|
|
|
}
|
|
|
|
|
|
|
|
func readTrieLog() ([]float64, map[int][]float64, []float64) {
|
|
|
|
data, err := ioutil.ReadFile("dust/hack.log")
|
|
|
|
check(err)
|
|
|
|
thresholds := []float64{}
|
|
|
|
counts := map[int][]float64{}
|
|
|
|
for i := 2; i <= 16; i++ {
|
|
|
|
counts[i] = []float64{}
|
|
|
|
}
|
|
|
|
shorts := []float64{}
|
|
|
|
lines := bytes.Split(data, []byte("\n"))
|
|
|
|
for _, line := range lines {
|
|
|
|
if bytes.HasPrefix(line, []byte("Threshold:")) {
|
|
|
|
tokens := bytes.Split(line, []byte(" "))
|
|
|
|
if len(tokens) == 23 {
|
|
|
|
wei := parseFloat64(string(tokens[1]))
|
|
|
|
thresholds = append(thresholds, wei)
|
|
|
|
for i := 2; i <= 16; i++ {
|
|
|
|
pair := bytes.Split(tokens[i+3], []byte(":"))
|
|
|
|
counts[i] = append(counts[i], parseFloat64(string(pair[1])))
|
|
|
|
}
|
|
|
|
pair := bytes.Split(tokens[21], []byte(":"))
|
|
|
|
shorts = append(shorts, parseFloat64(string(pair[1])))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return thresholds, counts, shorts
|
|
|
|
}
|
|
|
|
|
|
|
|
func ts() []chart.GridLine {
|
|
|
|
return []chart.GridLine{
|
|
|
|
{Value: 420.0},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func trieChart() {
|
|
|
|
thresholds, counts, shorts := readTrieLog()
|
|
|
|
fmt.Printf("%d %d %d\n", len(thresholds), len(counts), len(shorts))
|
|
|
|
shortsSeries := &chart.ContinuousSeries{
|
|
|
|
Name: "Short nodes",
|
|
|
|
Style: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.ColorBlue,
|
|
|
|
FillColor: chart.ColorBlue.WithAlpha(100),
|
|
|
|
},
|
|
|
|
XValues: thresholds,
|
|
|
|
YValues: shorts,
|
|
|
|
}
|
|
|
|
countSeries := make(map[int]*chart.ContinuousSeries)
|
|
|
|
for i := 2; i <= 16; i++ {
|
|
|
|
countSeries[i] = &chart.ContinuousSeries{
|
|
|
|
Name: fmt.Sprintf("%d-nodes", i),
|
|
|
|
Style: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.GetAlternateColor(i),
|
|
|
|
},
|
|
|
|
XValues: thresholds,
|
|
|
|
YValues: counts[i],
|
|
|
|
}
|
|
|
|
}
|
|
|
|
xaxis := &chart.XAxis{
|
|
|
|
Name: "Dust theshold",
|
|
|
|
Style: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
},
|
|
|
|
ValueFormatter: func(v interface{}) string {
|
|
|
|
return fmt.Sprintf("%d wei", int(v.(float64)))
|
|
|
|
},
|
|
|
|
GridMajorStyle: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.DefaultStrokeColor,
|
|
|
|
StrokeWidth: 1.0,
|
|
|
|
},
|
|
|
|
Range: &chart.ContinuousRange{
|
|
|
|
Min: thresholds[0],
|
|
|
|
Max: thresholds[len(thresholds)-1],
|
|
|
|
},
|
|
|
|
Ticks: []chart.Tick{
|
|
|
|
{Value: 0.0, Label: "0"},
|
|
|
|
{Value: 1.0, Label: "wei"},
|
|
|
|
{Value: 10.0, Label: "10"},
|
|
|
|
{Value: 100.0, Label: "100"},
|
|
|
|
{Value: 1e3, Label: "1e3"},
|
|
|
|
{Value: 1e4, Label: "1e4"},
|
|
|
|
{Value: 1e5, Label: "1e5"},
|
|
|
|
{Value: 1e6, Label: "1e6"},
|
|
|
|
{Value: 1e7, Label: "1e7"},
|
|
|
|
{Value: 1e8, Label: "1e8"},
|
|
|
|
{Value: 1e9, Label: "1e9"},
|
|
|
|
{Value: 1e10, Label: "1e10"},
|
|
|
|
//{1e15, "finney"},
|
|
|
|
//{1e18, "ether"},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
graph3 := chart.Chart{
|
|
|
|
Width: 1280,
|
|
|
|
Height: 720,
|
|
|
|
Background: chart.Style{
|
|
|
|
Padding: chart.Box{
|
|
|
|
Top: 50,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
XAxis: *xaxis,
|
|
|
|
YAxis: chart.YAxis{
|
|
|
|
Name: "Node count",
|
|
|
|
NameStyle: chart.StyleShow(),
|
|
|
|
Style: chart.StyleShow(),
|
|
|
|
TickStyle: chart.Style{
|
|
|
|
TextRotationDegrees: 45.0,
|
|
|
|
},
|
|
|
|
ValueFormatter: func(v interface{}) string {
|
|
|
|
return fmt.Sprintf("%dm", int(v.(float64)/1e6))
|
|
|
|
},
|
|
|
|
GridMajorStyle: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.DefaultStrokeColor,
|
|
|
|
StrokeWidth: 1.0,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Series: []chart.Series{
|
|
|
|
shortsSeries,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
graph3.Elements = []chart.Renderable{chart.LegendThin(&graph3)}
|
|
|
|
buffer := bytes.NewBuffer([]byte{})
|
|
|
|
err := graph3.Render(chart.PNG, buffer)
|
|
|
|
check(err)
|
|
|
|
err = ioutil.WriteFile("chart3.png", buffer.Bytes(), 0644)
|
|
|
|
check(err)
|
|
|
|
graph4 := chart.Chart{
|
|
|
|
Width: 1280,
|
|
|
|
Height: 720,
|
|
|
|
Background: chart.Style{
|
|
|
|
Padding: chart.Box{
|
|
|
|
Top: 50,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
XAxis: *xaxis,
|
|
|
|
YAxis: chart.YAxis{
|
|
|
|
Name: "Node count",
|
|
|
|
NameStyle: chart.StyleShow(),
|
|
|
|
Style: chart.StyleShow(),
|
|
|
|
TickStyle: chart.Style{
|
|
|
|
TextRotationDegrees: 45.0,
|
|
|
|
},
|
|
|
|
ValueFormatter: func(v interface{}) string {
|
|
|
|
return fmt.Sprintf("%.2fm", v.(float64)/1e6)
|
|
|
|
},
|
|
|
|
GridMajorStyle: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.DefaultStrokeColor,
|
|
|
|
StrokeWidth: 1.0,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Series: []chart.Series{
|
|
|
|
countSeries[2],
|
|
|
|
countSeries[3],
|
|
|
|
},
|
|
|
|
}
|
|
|
|
graph4.Elements = []chart.Renderable{chart.LegendThin(&graph4)}
|
|
|
|
buffer = bytes.NewBuffer([]byte{})
|
|
|
|
err = graph4.Render(chart.PNG, buffer)
|
|
|
|
check(err)
|
|
|
|
err = ioutil.WriteFile("chart4.png", buffer.Bytes(), 0644)
|
|
|
|
check(err)
|
|
|
|
graph5 := chart.Chart{
|
|
|
|
Width: 1280,
|
|
|
|
Height: 720,
|
|
|
|
Background: chart.Style{
|
|
|
|
Padding: chart.Box{
|
|
|
|
Top: 50,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
XAxis: *xaxis,
|
|
|
|
YAxis: chart.YAxis{
|
|
|
|
Name: "Node count",
|
|
|
|
NameStyle: chart.StyleShow(),
|
|
|
|
Style: chart.StyleShow(),
|
|
|
|
TickStyle: chart.Style{
|
|
|
|
TextRotationDegrees: 45.0,
|
|
|
|
},
|
|
|
|
ValueFormatter: func(v interface{}) string {
|
|
|
|
return fmt.Sprintf("%.2fk", v.(float64)/1e3)
|
|
|
|
},
|
|
|
|
GridMajorStyle: chart.Style{
|
|
|
|
Show: true,
|
|
|
|
StrokeColor: chart.DefaultStrokeColor,
|
|
|
|
StrokeWidth: 1.0,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Series: []chart.Series{
|
|
|
|
countSeries[4],
|
|
|
|
countSeries[5],
|
|
|
|
countSeries[6],
|
|
|
|
countSeries[7],
|
|
|
|
countSeries[8],
|
|
|
|
countSeries[9],
|
|
|
|
countSeries[10],
|
|
|
|
countSeries[11],
|
|
|
|
countSeries[12],
|
|
|
|
countSeries[13],
|
|
|
|
countSeries[14],
|
|
|
|
countSeries[15],
|
|
|
|
countSeries[16],
|
|
|
|
},
|
|
|
|
}
|
|
|
|
graph5.Elements = []chart.Renderable{chart.LegendThin(&graph5)}
|
|
|
|
buffer = bytes.NewBuffer([]byte{})
|
|
|
|
err = graph5.Render(chart.PNG, buffer)
|
|
|
|
check(err)
|
|
|
|
err = ioutil.WriteFile("chart5.png", buffer.Bytes(), 0644)
|
|
|
|
check(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func execToBlock(block int) {
|
|
|
|
blockDb, err := ethdb.NewBoltDatabase("/Users/alexeyakhunov/Library/Ethereum/testnet/geth/chaindata")
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("/home/akhounov/.ethereum/geth/chaindata")
|
|
|
|
check(err)
|
|
|
|
bcb, err := core.NewBlockChain(blockDb, nil, params.TestnetChainConfig, ethash.NewFaker(), vm.Config{}, nil)
|
|
|
|
check(err)
|
|
|
|
defer blockDb.Close()
|
|
|
|
os.Remove("statedb")
|
|
|
|
stateDb, err := ethdb.NewBoltDatabase("statedb")
|
|
|
|
check(err)
|
|
|
|
defer stateDb.Close()
|
|
|
|
_, _, _, err = core.SetupGenesisBlock(stateDb, core.DefaultTestnetGenesisBlock())
|
|
|
|
check(err)
|
|
|
|
bc, err := core.NewBlockChain(stateDb, nil, params.TestnetChainConfig, ethash.NewFaker(), vm.Config{}, nil)
|
|
|
|
check(err)
|
|
|
|
//bc.SetNoHistory(true)
|
|
|
|
blocks := types.Blocks{}
|
|
|
|
var lastBlock *types.Block
|
|
|
|
for i := 1; i <= block; i++ {
|
|
|
|
lastBlock = bcb.GetBlockByNumber(uint64(i))
|
|
|
|
blocks = append(blocks, lastBlock)
|
|
|
|
if len(blocks) >= 100 || i == block {
|
|
|
|
_, err = bc.InsertChain(blocks)
|
|
|
|
check(err)
|
|
|
|
fmt.Printf("Inserted %d blocks\n", i)
|
|
|
|
blocks = types.Blocks{}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
tds, err := bc.GetTrieDbState()
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
root := tds.LastRoot()
|
|
|
|
fmt.Printf("Root hash: %x\n", root)
|
|
|
|
fmt.Printf("Last block root hash: %x\n", lastBlock.Root())
|
|
|
|
filename := fmt.Sprintf("right_%d.txt", lastBlock.NumberU64())
|
|
|
|
fmt.Printf("Generating deep snapshot of the right tries... %s\n", filename)
|
|
|
|
f, err := os.Create(filename)
|
|
|
|
if err == nil {
|
|
|
|
defer f.Close()
|
|
|
|
tds.PrintTrie(f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func extractTrie(block int) {
|
|
|
|
stateDb, err := ethdb.NewBoltDatabase("statedb")
|
|
|
|
check(err)
|
|
|
|
defer stateDb.Close()
|
|
|
|
bc, err := core.NewBlockChain(stateDb, nil, params.TestnetChainConfig, ethash.NewFaker(), vm.Config{}, nil)
|
|
|
|
check(err)
|
|
|
|
baseBlock := bc.GetBlockByNumber(uint64(block))
|
|
|
|
tds, err := state.NewTrieDbState(baseBlock.Root(), stateDb, baseBlock.NumberU64())
|
|
|
|
check(err)
|
|
|
|
startTime := time.Now()
|
|
|
|
tds.Rebuild()
|
|
|
|
fmt.Printf("Rebuld done in %v\n", time.Since(startTime))
|
|
|
|
rebuiltRoot := tds.LastRoot()
|
|
|
|
fmt.Printf("Rebuit root hash: %x\n", rebuiltRoot)
|
|
|
|
filename := fmt.Sprintf("right_%d.txt", baseBlock.NumberU64())
|
|
|
|
fmt.Printf("Generating deep snapshot of the right tries... %s\n", filename)
|
|
|
|
f, err := os.Create(filename)
|
|
|
|
if err == nil {
|
|
|
|
defer f.Close()
|
|
|
|
tds.PrintTrie(f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func testRewind(block, rewind int) {
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("/Users/alexeyakhunov/Library/Ethereum/testnet/geth/chaindata")
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("/home/akhounov/.ethereum/geth/chaindata")
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("statedb")
|
|
|
|
ethDb, err := ethdb.NewBoltDatabase("/Volumes/tb4/turbo-geth/geth//chaindata")
|
|
|
|
check(err)
|
|
|
|
defer ethDb.Close()
|
|
|
|
bc, err := core.NewBlockChain(ethDb, nil, params.MainnetChainConfig, ethash.NewFaker(), vm.Config{}, nil)
|
|
|
|
check(err)
|
|
|
|
currentBlock := bc.CurrentBlock()
|
|
|
|
currentBlockNr := currentBlock.NumberU64()
|
|
|
|
if block == 1 {
|
|
|
|
block = int(currentBlockNr)
|
|
|
|
}
|
|
|
|
baseBlock := bc.GetBlockByNumber(uint64(block))
|
|
|
|
baseBlockNr := baseBlock.NumberU64()
|
|
|
|
fmt.Printf("Base block number: %d\n", baseBlockNr)
|
|
|
|
fmt.Printf("Base block root hash: %x\n", baseBlock.Root())
|
|
|
|
db := ethDb.NewBatch()
|
|
|
|
defer db.Rollback()
|
|
|
|
tds, err := state.NewTrieDbState(baseBlock.Root(), db, baseBlockNr)
|
|
|
|
tds.SetHistorical(baseBlockNr != currentBlockNr)
|
|
|
|
check(err)
|
|
|
|
startTime := time.Now()
|
|
|
|
tds.Rebuild()
|
|
|
|
fmt.Printf("Rebuld done in %v\n", time.Since(startTime))
|
|
|
|
rebuiltRoot := tds.LastRoot()
|
|
|
|
fmt.Printf("Rebuit root hash: %x\n", rebuiltRoot)
|
|
|
|
startTime = time.Now()
|
|
|
|
rewindLen := uint64(rewind)
|
|
|
|
|
|
|
|
err = tds.UnwindTo(baseBlockNr - rewindLen)
|
|
|
|
fmt.Printf("Unwind done in %v\n", time.Since(startTime))
|
|
|
|
check(err)
|
|
|
|
rewoundBlock1 := bc.GetBlockByNumber(baseBlockNr - rewindLen + 1)
|
|
|
|
fmt.Printf("Rewound+1 block number: %d\n", rewoundBlock1.NumberU64())
|
|
|
|
fmt.Printf("Rewound+1 block hash: %x\n", rewoundBlock1.Hash())
|
|
|
|
fmt.Printf("Rewound+1 block root hash: %x\n", rewoundBlock1.Root())
|
|
|
|
fmt.Printf("Rewound+1 block parent hash: %x\n", rewoundBlock1.ParentHash())
|
|
|
|
|
|
|
|
rewoundBlock := bc.GetBlockByNumber(baseBlockNr - rewindLen)
|
|
|
|
fmt.Printf("Rewound block number: %d\n", rewoundBlock.NumberU64())
|
|
|
|
fmt.Printf("Rewound block hash: %x\n", rewoundBlock.Hash())
|
|
|
|
fmt.Printf("Rewound block root hash: %x\n", rewoundBlock.Root())
|
|
|
|
fmt.Printf("Rewound block parent hash: %x\n", rewoundBlock.ParentHash())
|
|
|
|
rewoundRoot := tds.LastRoot()
|
|
|
|
fmt.Printf("Calculated rewound root hash: %x\n", rewoundRoot)
|
|
|
|
/*
|
|
|
|
filename := fmt.Sprintf("root_%d.txt", rewoundBlock.NumberU64())
|
|
|
|
fmt.Printf("Generating deep snapshot of the wront tries... %s\n", filename)
|
|
|
|
f, err := os.Create(filename)
|
|
|
|
if err == nil {
|
|
|
|
defer f.Close()
|
|
|
|
tds.PrintTrie(f)
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
tds, err = state.NewTrieDbState(rewoundBlock.Root(), db, rewoundBlock.NumberU64())
|
|
|
|
tds.SetHistorical(true)
|
|
|
|
check(err)
|
|
|
|
startTime := time.Now()
|
|
|
|
tds.Rebuild()
|
|
|
|
fmt.Printf("Rebuld done in %v\n", time.Since(startTime))
|
|
|
|
rebuiltRoot, err := tds.TrieRoot()
|
|
|
|
fmt.Printf("Rebuilt root: %x\n", rebuiltRoot)
|
|
|
|
check(err)
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
}
|
|
|
|
|
|
|
|
func testStartup() {
|
|
|
|
startTime := time.Now()
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("/Users/alexeyakhunov/Library/Ethereum/geth/chaindata")
|
|
|
|
ethDb, err := ethdb.NewBoltDatabase("/home/akhounov/.ethereum/geth/chaindata")
|
|
|
|
check(err)
|
|
|
|
defer ethDb.Close()
|
|
|
|
bc, err := core.NewBlockChain(ethDb, nil, params.MainnetChainConfig, ethash.NewFaker(), vm.Config{}, nil)
|
|
|
|
check(err)
|
|
|
|
currentBlock := bc.CurrentBlock()
|
|
|
|
currentBlockNr := currentBlock.NumberU64()
|
|
|
|
fmt.Printf("Current block number: %d\n", currentBlockNr)
|
|
|
|
fmt.Printf("Current block root hash: %x\n", currentBlock.Root())
|
|
|
|
t := trie.New(common.Hash{})
|
|
|
|
r := trie.NewResolver(0, true, currentBlockNr)
|
|
|
|
key := []byte{}
|
|
|
|
rootHash := currentBlock.Root()
|
|
|
|
req := t.NewResolveRequest(nil, key, 0, rootHash[:])
|
|
|
|
r.AddRequest(req)
|
|
|
|
err = r.ResolveWithDb(ethDb, currentBlockNr)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("%v\n", err)
|
|
|
|
}
|
|
|
|
fmt.Printf("Took %v\n", time.Since(startTime))
|
|
|
|
}
|
|
|
|
|
|
|
|
func testResolve() {
|
|
|
|
startTime := time.Now()
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("/home/akhounov/.ethereum/geth/chaindata")
|
|
|
|
ethDb, err := ethdb.NewBoltDatabase("/Users/alexeyakhunov/Library/Ethereum/geth/chaindata")
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("statedb")
|
|
|
|
check(err)
|
|
|
|
defer ethDb.Close()
|
|
|
|
bc, err := core.NewBlockChain(ethDb, nil, params.MainnetChainConfig, ethash.NewFaker(), vm.Config{}, nil)
|
|
|
|
check(err)
|
|
|
|
currentBlock := bc.CurrentBlock()
|
|
|
|
currentBlockNr := currentBlock.NumberU64()
|
|
|
|
fmt.Printf("Current block number: %d\n", currentBlockNr)
|
|
|
|
fmt.Printf("Current block root hash: %x\n", currentBlock.Root())
|
|
|
|
prevBlock := bc.GetBlockByNumber(currentBlockNr - 2)
|
|
|
|
fmt.Printf("Prev block root hash: %x\n", prevBlock.Root())
|
|
|
|
contract := common.FromHex("0x578e1f34346cb1067347b2ad256ada250b7853de763bd54110271a39e0cd52750000000000000000")
|
|
|
|
r := trie.NewResolver(2, false, 225281)
|
|
|
|
r.SetHistorical(true)
|
|
|
|
key := []byte{}
|
|
|
|
resolveHash := common.FromHex("a3a02e29c6dc8fbb769555a80e3f2b0789a0376296be013a956676d58deaf791")
|
|
|
|
t := trie.New(common.Hash{})
|
|
|
|
req := t.NewResolveRequest(contract, key, 0, resolveHash)
|
|
|
|
r.AddRequest(req)
|
|
|
|
//err = r.ResolveWithDb(ethDb, 225281)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("%v\n", err)
|
|
|
|
}
|
|
|
|
fmt.Printf("Took %v\n", time.Since(startTime))
|
|
|
|
t.Update(common.FromHex("0x578e1f34346cb1067347b2ad256ada250b7853de763bd54110271a39e0cd52757c17435002e70fd7d982a101b0549945244f496bf4bb5503d5de3aa770842bce"),
|
|
|
|
common.FromHex("0xa06fe9c682fbb890c09eec59047f97d165875d4f3c86bc65c2870d577b8245f111"), 0)
|
|
|
|
_, h := t.DeepHash(common.FromHex("0x578e1f34346cb1067347b2ad256ada250b7853de763bd54110271a39e0cd5275"))
|
|
|
|
fmt.Printf("deep hash: %x\n", h)
|
|
|
|
//t.Print(os.Stdout)
|
|
|
|
/*
|
|
|
|
filename := fmt.Sprintf("root_%d.txt", currentBlockNr)
|
|
|
|
f, err := os.Create(filename)
|
|
|
|
if err == nil {
|
|
|
|
t.Print(f)
|
|
|
|
f.Close()
|
|
|
|
} else {
|
|
|
|
check(err)
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
//enc, _ := ethDb.GetAsOf(state.AccountsBucket, state.AccountsHistoryBucket, crypto.Keccak256(contract), 2701646)
|
|
|
|
//fmt.Printf("Account: %x\n", enc)
|
|
|
|
}
|
|
|
|
|
|
|
|
func hashFile() {
|
|
|
|
f, err := os.Open("/Users/alexeyakhunov/mygit/go-ethereum/geth.log")
|
|
|
|
check(err)
|
|
|
|
defer f.Close()
|
|
|
|
w, err := os.Create("/Users/alexeyakhunov/mygit/go-ethereum/geth_read.log")
|
|
|
|
check(err)
|
|
|
|
defer w.Close()
|
|
|
|
scanner := bufio.NewScanner(f)
|
|
|
|
count := 0
|
|
|
|
for scanner.Scan() {
|
|
|
|
line := scanner.Text()
|
|
|
|
if strings.HasPrefix(line, "ResolveWithDb") || strings.HasPrefix(line, "Error") ||
|
|
|
|
strings.HasPrefix(line, "0000000000000000000000000000000000000000000000000000000000000000") ||
|
|
|
|
strings.HasPrefix(line, "ERROR") || strings.HasPrefix(line, "tc{") {
|
|
|
|
fmt.Printf("%d %s\n", count, line)
|
|
|
|
count++
|
|
|
|
} else if count == 66 {
|
|
|
|
w.WriteString(line)
|
|
|
|
w.WriteString("\n")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fmt.Printf("%d lines scanned\n", count)
|
|
|
|
}
|
|
|
|
|
|
|
|
func rlpIndices() {
|
|
|
|
keybuf := new(bytes.Buffer)
|
|
|
|
for i := 0; i < 512; i++ {
|
|
|
|
keybuf.Reset()
|
|
|
|
rlp.Encode(keybuf, uint(i))
|
|
|
|
fmt.Printf("Encoding of %d is %x\n", i, keybuf.Bytes())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func printFullNodeRLPs() {
|
|
|
|
trie.FullNode1()
|
|
|
|
trie.FullNode2()
|
|
|
|
trie.FullNode3()
|
|
|
|
trie.FullNode4()
|
|
|
|
trie.ShortNode1()
|
|
|
|
trie.ShortNode2()
|
|
|
|
trie.Hash1()
|
|
|
|
trie.Hash2()
|
|
|
|
trie.Hash3()
|
|
|
|
trie.Hash4()
|
|
|
|
trie.Hash5()
|
|
|
|
trie.Hash6()
|
|
|
|
trie.Hash7()
|
|
|
|
}
|
|
|
|
|
|
|
|
func testDifficulty() {
|
|
|
|
genesisBlock, _, _, err := core.DefaultGenesisBlock().ToBlock(nil)
|
|
|
|
check(err)
|
|
|
|
d1 := ethash.CalcDifficulty(params.MainnetChainConfig, 100000, genesisBlock.Header())
|
|
|
|
fmt.Printf("Block 1 difficulty: %d\n", d1)
|
|
|
|
}
|
|
|
|
|
2019-11-21 15:56:39 +00:00
|
|
|
// 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, err := ethdb.NewBoltDatabase(chaindata)
|
2019-05-27 13:51:49 +00:00
|
|
|
check(err)
|
2019-11-21 15:56:39 +00:00
|
|
|
blocksToSearch := 1000
|
|
|
|
for i := uint64(block); i < uint64(block+blocksToSearch); i++ {
|
2019-05-27 13:51:49 +00:00
|
|
|
hash := rawdb.ReadCanonicalHash(ethDb, i)
|
|
|
|
header := rawdb.ReadHeader(ethDb, hash, i)
|
2019-11-21 15:56:39 +00:00
|
|
|
if header.Root == stateRoot || stateRoot == (common.Hash{}) {
|
2019-05-27 13:51:49 +00:00
|
|
|
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)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func printTxHashes() {
|
|
|
|
ethDb, err := ethdb.NewBoltDatabase("/Users/alexeyakhunov/Library/Ethereum/geth/chaindata")
|
|
|
|
check(err)
|
|
|
|
defer ethDb.Close()
|
|
|
|
for b := uint64(0); b < uint64(100000); b++ {
|
|
|
|
hash := rawdb.ReadCanonicalHash(ethDb, b)
|
|
|
|
block := rawdb.ReadBlock(ethDb, hash, b)
|
|
|
|
if block == nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
for _, tx := range block.Transactions() {
|
|
|
|
fmt.Printf("%x\n", tx.Hash())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func relayoutKeys() {
|
|
|
|
//db, err := bolt.Open("/home/akhounov/.ethereum/geth/chaindata", 0600, &bolt.Options{ReadOnly: true})
|
|
|
|
db, err := bolt.Open("/Users/alexeyakhunov/Library/Ethereum/geth/chaindata", 0600, &bolt.Options{ReadOnly: true})
|
|
|
|
check(err)
|
|
|
|
defer db.Close()
|
|
|
|
var count int
|
|
|
|
err = db.View(func(tx *bolt.Tx) error {
|
2019-11-07 15:51:25 +00:00
|
|
|
b := tx.Bucket(dbutils.ChangeSetBucket)
|
2019-05-27 13:51:49 +00:00
|
|
|
if b == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
c := b.Cursor()
|
|
|
|
for k, _ := c.First(); k != nil; k, _ = c.Next() {
|
|
|
|
count++
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
check(err)
|
|
|
|
fmt.Printf("Records: %d\n", count)
|
|
|
|
}
|
|
|
|
|
|
|
|
func upgradeBlocks() {
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("/Users/alexeyakhunov/Library/Ethereum/geth/chaindata")
|
|
|
|
ethDb, err := ethdb.NewBoltDatabase("/home/akhounov/.ethereum/geth/chaindata")
|
|
|
|
check(err)
|
|
|
|
defer ethDb.Close()
|
|
|
|
start := []byte{}
|
|
|
|
var keys [][]byte
|
|
|
|
if err := ethDb.Walk([]byte("b"), start, 0, func(k, v []byte) (bool, error) {
|
|
|
|
if len(keys)%1000 == 0 {
|
|
|
|
fmt.Printf("Collected keys: %d\n", len(keys))
|
|
|
|
}
|
|
|
|
keys = append(keys, common.CopyBytes(k))
|
|
|
|
return true, nil
|
|
|
|
}); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
for i, key := range keys {
|
|
|
|
v, err := ethDb.Get([]byte("b"), key)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
smallBody := new(types.SmallBody) // To be changed to SmallBody
|
|
|
|
if err := rlp.Decode(bytes.NewReader(v), smallBody); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
body := new(types.Body)
|
|
|
|
blockNum := binary.BigEndian.Uint64(key[:8])
|
|
|
|
signer := types.MakeSigner(params.MainnetChainConfig, big.NewInt(int64(blockNum)))
|
|
|
|
body.Senders = make([]common.Address, len(smallBody.Transactions))
|
|
|
|
for j, tx := range smallBody.Transactions {
|
|
|
|
addr, err := signer.Sender(tx)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
body.Senders[j] = addr
|
|
|
|
}
|
|
|
|
body.Transactions = smallBody.Transactions
|
|
|
|
body.Uncles = smallBody.Uncles
|
|
|
|
newV, err := rlp.EncodeToBytes(body)
|
|
|
|
if err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
ethDb.Put([]byte("b"), key, newV)
|
|
|
|
if i%1000 == 0 {
|
|
|
|
fmt.Printf("Upgraded keys: %d\n", i)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
check(ethDb.DeleteBucket([]byte("r")))
|
|
|
|
}
|
|
|
|
|
|
|
|
func readTrie(filename string) *trie.Trie {
|
|
|
|
f, err := os.Open(filename)
|
|
|
|
check(err)
|
|
|
|
defer f.Close()
|
|
|
|
t, err := trie.Load(f)
|
|
|
|
check(err)
|
|
|
|
return t
|
|
|
|
}
|
|
|
|
|
|
|
|
func invTree(wrong, right, diff string, name string) {
|
|
|
|
fmt.Printf("Reading trie...\n")
|
|
|
|
t1 := readTrie(fmt.Sprintf("%s_%s.txt", wrong, name))
|
|
|
|
fmt.Printf("Root hash: %x\n", t1.Hash())
|
|
|
|
fmt.Printf("Reading trie 2...\n")
|
|
|
|
t2 := readTrie(fmt.Sprintf("%s_%s.txt", right, name))
|
|
|
|
fmt.Printf("Root hash: %x\n", t2.Hash())
|
|
|
|
c, err := os.Create(fmt.Sprintf("%s_%s.txt", diff, name))
|
|
|
|
check(err)
|
|
|
|
defer c.Close()
|
|
|
|
t1.PrintDiff(t2, c)
|
|
|
|
}
|
|
|
|
|
2019-10-18 12:11:50 +00:00
|
|
|
func preimage(chaindata string, image common.Hash) {
|
|
|
|
ethDb, err := ethdb.NewBoltDatabase(chaindata)
|
2019-05-27 13:51:49 +00:00
|
|
|
check(err)
|
|
|
|
defer ethDb.Close()
|
2019-10-18 12:11:50 +00:00
|
|
|
p, err := ethDb.Get(dbutils.PreimagePrefix, image[:])
|
2019-05-27 13:51:49 +00:00
|
|
|
check(err)
|
|
|
|
fmt.Printf("%x\n", p)
|
|
|
|
}
|
|
|
|
|
|
|
|
func loadAccount() {
|
|
|
|
ethDb, err := ethdb.NewBoltDatabase("/home/akhounov/.ethereum/geth/chaindata")
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("/Users/alexeyakhunov/Library/Ethereum/geth/chaindata")
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("/Volumes/tb4/turbo-geth/geth/chaindata")
|
|
|
|
check(err)
|
|
|
|
defer ethDb.Close()
|
|
|
|
blockNr := uint64(*block)
|
2019-11-07 15:51:25 +00:00
|
|
|
blockSuffix := dbutils.EncodeTimestamp(blockNr)
|
2019-05-27 13:51:49 +00:00
|
|
|
accountBytes := common.FromHex(*account)
|
|
|
|
secKey := crypto.Keccak256(accountBytes)
|
|
|
|
accountData, err := ethDb.GetAsOf(dbutils.AccountsBucket, dbutils.AccountsHistoryBucket, secKey, blockNr+1)
|
|
|
|
check(err)
|
|
|
|
fmt.Printf("Account data: %x\n", accountData)
|
|
|
|
startkey := make([]byte, len(accountBytes)+32)
|
|
|
|
copy(startkey, accountBytes)
|
|
|
|
t := trie.New(common.Hash{})
|
|
|
|
count := 0
|
|
|
|
if err := ethDb.WalkAsOf(dbutils.StorageBucket, dbutils.StorageHistoryBucket, startkey, uint(len(accountBytes)*8), blockNr, func(k, v []byte) (bool, error) {
|
|
|
|
key := k[len(accountBytes):]
|
|
|
|
//fmt.Printf("%x: %x\n", key, v)
|
|
|
|
t.Update(key, v, blockNr)
|
|
|
|
count++
|
|
|
|
return true, nil
|
|
|
|
}); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
fmt.Printf("After %d updates, reconstructed storage root: %x\n", count, t.Hash())
|
|
|
|
var keys [][]byte
|
|
|
|
if err := ethDb.Walk(dbutils.StorageHistoryBucket, accountBytes, uint(len(accountBytes)*8), func(k, v []byte) (bool, error) {
|
|
|
|
if !bytes.HasSuffix(k, blockSuffix) {
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
key := k[:len(k)-len(blockSuffix)]
|
|
|
|
keys = append(keys, common.CopyBytes(key))
|
|
|
|
return true, nil
|
|
|
|
}); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
fmt.Printf("%d keys updated\n", len(keys))
|
|
|
|
for _, k := range keys {
|
|
|
|
v, err := ethDb.GetAsOf(dbutils.StorageBucket, dbutils.StorageHistoryBucket, k, blockNr+1)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("for key %x err %v\n", k, err)
|
|
|
|
}
|
|
|
|
vOrig, err := ethDb.GetAsOf(dbutils.StorageBucket, dbutils.StorageHistoryBucket, k, blockNr)
|
|
|
|
if err != nil {
|
|
|
|
fmt.Printf("for key %x err %v\n", k, err)
|
|
|
|
}
|
|
|
|
key := ([]byte(k))[len(accountBytes):]
|
|
|
|
if len(v) > 0 {
|
|
|
|
fmt.Printf("Updated %x: %x from %x\n", key, v, vOrig)
|
|
|
|
t.Update(key, v, blockNr)
|
|
|
|
check(err)
|
|
|
|
} else {
|
|
|
|
fmt.Printf("Deleted %x from %x\n", key, vOrig)
|
|
|
|
t.Delete(key, blockNr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fmt.Printf("Updated storage root: %x\n", t.Hash())
|
|
|
|
}
|
|
|
|
|
|
|
|
func printBranches(block uint64) {
|
|
|
|
ethDb, err := ethdb.NewBoltDatabase("/Users/alexeyakhunov/Library/Ethereum/testnet/geth/chaindata")
|
|
|
|
//ethDb, err := ethdb.NewBoltDatabase("/home/akhounov/.ethereum/geth/chaindata")
|
|
|
|
check(err)
|
|
|
|
defer ethDb.Close()
|
|
|
|
fmt.Printf("All headers at the same height %d\n", block)
|
|
|
|
{
|
|
|
|
var hashes []common.Hash
|
|
|
|
numberEnc := make([]byte, 8)
|
|
|
|
binary.BigEndian.PutUint64(numberEnc, block)
|
|
|
|
if err := ethDb.Walk([]byte("h"), numberEnc, 8*8, func(k, v []byte) (bool, error) {
|
|
|
|
if len(k) == 8+32 {
|
|
|
|
hashes = append(hashes, common.BytesToHash(k[8:]))
|
|
|
|
}
|
|
|
|
return true, nil
|
|
|
|
}); err != nil {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
for _, hash := range hashes {
|
|
|
|
h := rawdb.ReadHeader(ethDb, hash, block)
|
|
|
|
fmt.Printf("block hash: %x, root hash: %x\n", h.Hash(), h.Root)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func readAccount() {
|
|
|
|
ethDb, err := ethdb.NewBoltDatabase("statedb")
|
|
|
|
check(err)
|
|
|
|
accountBytes := common.FromHex(*account)
|
|
|
|
secKey := crypto.Keccak256(accountBytes)
|
|
|
|
v, _ := ethDb.Get(dbutils.AccountsBucket, secKey)
|
|
|
|
fmt.Printf("%x:%x\n", secKey, v)
|
|
|
|
}
|
|
|
|
|
|
|
|
func repairCurrent() {
|
|
|
|
historyDb, err := bolt.Open("/Volumes/tb4/turbo-geth/ropsten/geth/chaindata", 0600, &bolt.Options{})
|
|
|
|
check(err)
|
|
|
|
defer historyDb.Close()
|
|
|
|
currentDb, err := bolt.Open("statedb", 0600, &bolt.Options{})
|
|
|
|
check(err)
|
|
|
|
defer currentDb.Close()
|
|
|
|
check(historyDb.Update(func(tx *bolt.Tx) error {
|
|
|
|
if err := tx.DeleteBucket(dbutils.StorageBucket); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
newB, err := tx.CreateBucket(dbutils.StorageBucket, true)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
count := 0
|
|
|
|
if err := currentDb.View(func(ctx *bolt.Tx) error {
|
|
|
|
b := ctx.Bucket(dbutils.StorageBucket)
|
|
|
|
c := b.Cursor()
|
|
|
|
for k, v := c.First(); k != nil; k, v = c.Next() {
|
|
|
|
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, err := bolt.Open("/Users/alexeyakhunov/Library/Ethereum/geth/chaindata", 0600, &bolt.Options{ReadOnly: true})
|
|
|
|
check(err)
|
|
|
|
err = db.View(func(tx *bolt.Tx) error {
|
|
|
|
sb := tx.Bucket(dbutils.StorageHistoryBucket)
|
|
|
|
if sb == nil {
|
|
|
|
fmt.Printf("Storage bucket not found\n")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return sb.ForEach(func(k, v []byte) error {
|
|
|
|
fmt.Printf("%x %x\n", k, v)
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
})
|
|
|
|
check(err)
|
|
|
|
db.Close()
|
|
|
|
}
|
|
|
|
|
|
|
|
func testMemBolt() {
|
|
|
|
db, err := bolt.Open("membolt", 0600, &bolt.Options{MemOnly: true})
|
|
|
|
check(err)
|
|
|
|
defer db.Close()
|
|
|
|
err = db.Update(func(tx *bolt.Tx) error {
|
|
|
|
bucket, err := tx.CreateBucketIfNotExists([]byte("B"), false)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Bucket creation: %v", err)
|
|
|
|
}
|
|
|
|
for i := 0; i < 1000; i++ {
|
|
|
|
err = bucket.Put(append([]byte("gjdfigjkdfljgdlfkjg"), []byte(fmt.Sprintf("%d", i))...), []byte("kljklgjfdkljkdjd"))
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("Put: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
check(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
func main() {
|
|
|
|
flag.Parse()
|
|
|
|
if *cpuprofile != "" {
|
|
|
|
f, err := os.Create(*cpuprofile)
|
|
|
|
if err != nil {
|
|
|
|
log.Fatal("could not create CPU profile: ", err)
|
|
|
|
}
|
|
|
|
if err := pprof.StartCPUProfile(f); err != nil {
|
|
|
|
log.Fatal("could not start CPU profile: ", err)
|
|
|
|
}
|
|
|
|
defer pprof.StopCPUProfile()
|
|
|
|
}
|
|
|
|
//db, err := bolt.Open("/home/akhounov/.ethereum/geth/chaindata", 0600, &bolt.Options{ReadOnly: true})
|
|
|
|
//db, err := bolt.Open("/Users/alexeyakhunov/Library/Ethereum/geth/chaindata", 0600, &bolt.Options{ReadOnly: true})
|
|
|
|
//check(err)
|
|
|
|
//defer db.Close()
|
|
|
|
//bucketStats(db)
|
|
|
|
//mychart()
|
|
|
|
//testRebuild()
|
|
|
|
if *action == "testRewind" {
|
|
|
|
testRewind(*block, *rewind)
|
|
|
|
}
|
|
|
|
//hashFile()
|
|
|
|
//buildHashFromFile()
|
|
|
|
if *action == "testResolve" {
|
|
|
|
testResolve()
|
|
|
|
}
|
|
|
|
//rlpIndices()
|
|
|
|
//printFullNodeRLPs()
|
|
|
|
//testStartup()
|
|
|
|
//testDifficulty()
|
|
|
|
//testRewindTests()
|
|
|
|
//if *reset != -1 {
|
|
|
|
// testReset(uint64(*reset))
|
|
|
|
//}
|
|
|
|
if *action == "testBlockHashes" {
|
2019-11-21 15:56:39 +00:00
|
|
|
testBlockHashes(*chaindata, *block, common.HexToHash(*hash))
|
2019-05-27 13:51:49 +00:00
|
|
|
}
|
|
|
|
//printBuckets(db)
|
|
|
|
//printTxHashes()
|
|
|
|
//relayoutKeys()
|
|
|
|
//testRedis()
|
|
|
|
//upgradeBlocks()
|
|
|
|
//compareTries()
|
|
|
|
if *action == "invTree" {
|
|
|
|
invTree("root", "right", "diff", *name)
|
|
|
|
}
|
|
|
|
//invTree("iw", "ir", "id", *block, true)
|
|
|
|
//loadAccount()
|
|
|
|
if *action == "preimage" {
|
2019-11-21 15:56:39 +00:00
|
|
|
preimage(*chaindata, common.HexToHash(*hash))
|
2019-05-27 13:51:49 +00:00
|
|
|
}
|
|
|
|
//printBranches(uint64(*block))
|
|
|
|
//execToBlock(*block)
|
|
|
|
//extractTrie(*block)
|
|
|
|
//repair()
|
|
|
|
//readAccount()
|
|
|
|
//repairCurrent()
|
|
|
|
//testMemBolt()
|
|
|
|
//fmt.Printf("\u00b3\n")
|
|
|
|
if *action == "dumpStorage" {
|
|
|
|
dumpStorage()
|
|
|
|
}
|
|
|
|
}
|