erigon-pulse/cmd/state/stats/index_stats.go

137 lines
3.2 KiB
Go
Raw Normal View History

package stats
import (
"bytes"
2021-07-07 16:15:49 +00:00
"context"
"encoding/csv"
"fmt"
"log"
"os"
"sort"
"strconv"
2020-08-10 23:55:32 +00:00
"strings"
"time"
2021-07-29 11:53:13 +00:00
"github.com/ledgerwatch/erigon-lib/kv"
"github.com/ledgerwatch/erigon-lib/kv/mdbx"
"github.com/ledgerwatch/erigon/common"
)
2020-08-10 23:55:32 +00:00
func IndexStats(chaindata string, indexBucket string, statsFile string) error {
db := mdbx.MustOpen(chaindata)
startTime := time.Now()
2021-06-20 06:00:22 +00:00
lenOfKey := common.AddressLength
if strings.HasPrefix(indexBucket, kv.StorageHistory) {
2021-06-20 06:00:22 +00:00
lenOfKey = common.AddressLength + common.HashLength + common.IncarnationLength
}
more1index := 0
more10index := make(map[string]uint64)
more50index := make(map[string]uint64)
more100index := make(map[string]uint64)
more200index := make(map[string]uint64)
more500index := make(map[string]uint64)
more1000index := make(map[string]uint64)
prevKey := []byte{}
count := uint64(1)
added := false
i := uint64(0)
2021-07-07 16:15:49 +00:00
tx, err := db.BeginRo(context.Background())
if err != nil {
return err
}
defer tx.Rollback()
if err = tx.ForEach(indexBucket, []byte{}, func(k, v []byte) error {
if i%100_000 == 0 {
fmt.Printf("Processed %dK, %s\n", i/1000, time.Since(startTime))
}
if bytes.Equal(k[:lenOfKey], prevKey) {
count++
if count > 1 && !added {
more1index++
added = true
}
if count > 10 {
more10index[string(common.CopyBytes(k[:lenOfKey]))] = count
}
if count > 50 {
more50index[string(common.CopyBytes(k[:lenOfKey]))] = count
}
if count > 100 {
more100index[string(common.CopyBytes(k[:lenOfKey]))] = count
}
if count > 200 {
more200index[string(common.CopyBytes(k[:lenOfKey]))] = count
}
if count > 500 {
more500index[string(common.CopyBytes(k[:lenOfKey]))] = count
}
if count > 1000 {
more1000index[string(common.CopyBytes(k[:lenOfKey]))] = count
}
} else {
added = false
count = 1
2021-06-20 06:00:22 +00:00
prevKey = common.CopyBytes(k[:common.AddressLength])
}
2021-07-07 16:15:49 +00:00
return nil
}); err != nil {
return err
}
fmt.Println("more1", more1index)
fmt.Println("more10", len(more10index))
fmt.Println("more50", len(more50index))
fmt.Println("more100", len(more100index))
fmt.Println("more200", len(more200index))
fmt.Println("more500", len(more500index))
fmt.Println("more1000", len(more1000index))
if statsFile != "" {
f, err := os.Create(statsFile)
if err != nil {
log.Fatal(err)
}
defer f.Close() //nolint
save10 := make([]struct {
Address string
Hash string
NumOfIndexes uint64
}, 0, len(more10index))
for hash, v := range more10index {
2021-06-20 06:00:22 +00:00
p := []byte(hash)[:common.AddressLength]
if len(p) == 0 {
p = make([]byte, 20)
}
save10 = append(save10, struct {
Address string
Hash string
NumOfIndexes uint64
}{
Address: common.BytesToAddress(p).String(),
NumOfIndexes: v,
Hash: common.Bytes2Hex([]byte(hash)),
})
}
sort.Slice(save10, func(i, j int) bool {
return save10[i].NumOfIndexes > save10[j].NumOfIndexes
})
csvWriter := csv.NewWriter(f)
err = csvWriter.Write([]string{"hash", "address", "num"})
if err != nil {
return err
}
for _, v := range save10 {
err = csvWriter.Write([]string{v.Hash, v.Address, strconv.FormatUint(v.NumOfIndexes, 10)})
if err != nil {
return err
}
}
}
return nil
}