mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-19 00:04:12 +00:00
945c76132c
* add more data * terence's review
305 lines
8.5 KiB
Go
305 lines
8.5 KiB
Go
package main
|
|
|
|
import (
|
|
"bufio"
|
|
"context"
|
|
"fmt"
|
|
"os"
|
|
"regexp"
|
|
"runtime"
|
|
"strings"
|
|
"time"
|
|
|
|
"github.com/kr/pretty"
|
|
fssz "github.com/prysmaticlabs/fastssz"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/transition"
|
|
state_native "github.com/prysmaticlabs/prysm/v4/beacon-chain/state/state-native"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/blocks"
|
|
"github.com/prysmaticlabs/prysm/v4/encoding/ssz/equality"
|
|
ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
prefixed "github.com/prysmaticlabs/prysm/v4/runtime/logging/logrus-prefixed-formatter"
|
|
"github.com/prysmaticlabs/prysm/v4/runtime/version"
|
|
log "github.com/sirupsen/logrus"
|
|
"github.com/urfave/cli/v2"
|
|
"gopkg.in/d4l3k/messagediff.v1"
|
|
)
|
|
|
|
func main() {
|
|
var blockPath string
|
|
var preStatePath string
|
|
var expectedPostStatePath string
|
|
var sszPath string
|
|
var sszType string
|
|
|
|
customFormatter := new(prefixed.TextFormatter)
|
|
customFormatter.TimestampFormat = "2006-01-02 15:04:05"
|
|
customFormatter.FullTimestamp = true
|
|
log.SetFormatter(customFormatter)
|
|
app := cli.App{}
|
|
app.Name = "pcli"
|
|
app.Usage = "A command line utility to run Ethereum consensus specific commands"
|
|
app.Version = version.Version()
|
|
app.Commands = []*cli.Command{
|
|
{
|
|
Name: "pretty",
|
|
Aliases: []string{"p"},
|
|
Usage: "pretty-print SSZ data",
|
|
Flags: []cli.Flag{
|
|
&cli.StringFlag{
|
|
Name: "ssz-path",
|
|
Usage: "Path to file(ssz)",
|
|
Required: true,
|
|
Destination: &sszPath,
|
|
},
|
|
&cli.StringFlag{
|
|
Name: "data-type",
|
|
Usage: "ssz file data type: " +
|
|
"block|" +
|
|
"blinded_block|" +
|
|
"signed_block|" +
|
|
"attestation|" +
|
|
"block_header|" +
|
|
"deposit|" +
|
|
"proposer_slashing|" +
|
|
"signed_block_header|" +
|
|
"signed_voluntary_exit|" +
|
|
"voluntary_exit|" +
|
|
"state_capella",
|
|
Required: true,
|
|
Destination: &sszType,
|
|
},
|
|
},
|
|
Action: func(c *cli.Context) error {
|
|
var data fssz.Unmarshaler
|
|
switch sszType {
|
|
case "block":
|
|
data = ðpb.BeaconBlock{}
|
|
case "signed_block":
|
|
data = ðpb.SignedBeaconBlock{}
|
|
case "blinded_block":
|
|
data = ðpb.BlindedBeaconBlockBellatrix{}
|
|
case "attestation":
|
|
data = ðpb.Attestation{}
|
|
case "block_header":
|
|
data = ðpb.BeaconBlockHeader{}
|
|
case "deposit":
|
|
data = ðpb.Deposit{}
|
|
case "deposit_message":
|
|
data = ðpb.DepositMessage{}
|
|
case "proposer_slashing":
|
|
data = ðpb.ProposerSlashing{}
|
|
case "signed_block_header":
|
|
data = ðpb.SignedBeaconBlockHeader{}
|
|
case "signed_voluntary_exit":
|
|
data = ðpb.SignedVoluntaryExit{}
|
|
case "voluntary_exit":
|
|
data = ðpb.VoluntaryExit{}
|
|
case "state_capella":
|
|
data = ðpb.BeaconStateCapella{}
|
|
default:
|
|
log.Fatal("Invalid type")
|
|
}
|
|
prettyPrint(sszPath, data)
|
|
return nil
|
|
},
|
|
},
|
|
{
|
|
Name: "benchmark-hash",
|
|
Aliases: []string{"b"},
|
|
Usage: "benchmark-hash SSZ data",
|
|
Flags: []cli.Flag{
|
|
&cli.StringFlag{
|
|
Name: "ssz-path",
|
|
Usage: "Path to file(ssz)",
|
|
Required: true,
|
|
Destination: &sszPath,
|
|
},
|
|
&cli.StringFlag{
|
|
Name: "data-type",
|
|
Usage: "ssz file data type: " +
|
|
"block_capella|" +
|
|
"blinded_block_capella|" +
|
|
"signed_block_capella|" +
|
|
"attestation|" +
|
|
"block_header|" +
|
|
"deposit|" +
|
|
"proposer_slashing|" +
|
|
"signed_block_header|" +
|
|
"signed_voluntary_exit|" +
|
|
"voluntary_exit|" +
|
|
"state_capella",
|
|
Required: true,
|
|
Destination: &sszType,
|
|
},
|
|
},
|
|
Action: func(c *cli.Context) error {
|
|
benchmarkHash(sszPath, sszType)
|
|
return nil
|
|
},
|
|
},
|
|
{
|
|
Name: "state-transition",
|
|
Category: "state-transition",
|
|
Usage: "Subcommand to run manual state transitions",
|
|
Flags: []cli.Flag{
|
|
&cli.StringFlag{
|
|
Name: "block-path",
|
|
Usage: "Path to block file(ssz)",
|
|
Destination: &blockPath,
|
|
},
|
|
&cli.StringFlag{
|
|
Name: "pre-state-path",
|
|
Usage: "Path to pre state file(ssz)",
|
|
Destination: &preStatePath,
|
|
},
|
|
&cli.StringFlag{
|
|
Name: "expected-post-state-path",
|
|
Usage: "Path to expected post state file(ssz)",
|
|
Destination: &expectedPostStatePath,
|
|
},
|
|
},
|
|
Action: func(c *cli.Context) error {
|
|
if blockPath == "" {
|
|
log.Info("Block path not provided for state transition. " +
|
|
"Please provide path")
|
|
reader := bufio.NewReader(os.Stdin)
|
|
text, err := reader.ReadString('\n')
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
if text = strings.ReplaceAll(text, "\n", ""); text == "" {
|
|
log.Fatal("Empty block path given")
|
|
}
|
|
blockPath = text
|
|
}
|
|
block := ðpb.SignedBeaconBlock{}
|
|
if err := dataFetcher(blockPath, block); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
blkRoot, err := block.Block.HashTreeRoot()
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
if preStatePath == "" {
|
|
log.Info("Pre State path not provided for state transition. " +
|
|
"Please provide path")
|
|
reader := bufio.NewReader(os.Stdin)
|
|
text, err := reader.ReadString('\n')
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
if text = strings.ReplaceAll(text, "\n", ""); text == "" {
|
|
log.Fatal("Empty state path given")
|
|
}
|
|
preStatePath = text
|
|
}
|
|
preState := ðpb.BeaconState{}
|
|
if err := dataFetcher(preStatePath, preState); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
stateObj, err := state_native.InitializeFromProtoPhase0(preState)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
preStateRoot, err := stateObj.HashTreeRoot(context.Background())
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
log.WithFields(log.Fields{
|
|
"blockSlot": fmt.Sprintf("%d", block.Block.Slot),
|
|
"preStateSlot": fmt.Sprintf("%d", stateObj.Slot()),
|
|
}).Infof(
|
|
"Performing state transition with a block root of %#x and pre state root of %#x",
|
|
blkRoot,
|
|
preStateRoot,
|
|
)
|
|
wsb, err := blocks.NewSignedBeaconBlock(block)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
postState, err := transition.ExecuteStateTransition(context.Background(), stateObj, wsb)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
postRoot, err := postState.HashTreeRoot(context.Background())
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
log.Infof("Finished state transition with post state root of %#x", postRoot)
|
|
|
|
// Diff the state if a post state is provided.
|
|
if expectedPostStatePath != "" {
|
|
expectedState := ðpb.BeaconState{}
|
|
if err := dataFetcher(expectedPostStatePath, expectedState); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
if !equality.DeepEqual(expectedState, postState.ToProtoUnsafe()) {
|
|
diff, _ := messagediff.PrettyDiff(expectedState, postState.ToProtoUnsafe())
|
|
log.Errorf("Derived state differs from provided post state: %s", diff)
|
|
}
|
|
}
|
|
return nil
|
|
},
|
|
},
|
|
}
|
|
if err := app.Run(os.Args); err != nil {
|
|
log.Error(err.Error())
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
// dataFetcher fetches and unmarshals data from file to provided data structure.
|
|
func dataFetcher(fPath string, data fssz.Unmarshaler) error {
|
|
rawFile, err := os.ReadFile(fPath) // #nosec G304
|
|
if err != nil {
|
|
return err
|
|
}
|
|
return data.UnmarshalSSZ(rawFile)
|
|
}
|
|
|
|
func prettyPrint(sszPath string, data fssz.Unmarshaler) {
|
|
if err := dataFetcher(sszPath, data); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
str := pretty.Sprint(data)
|
|
re := regexp.MustCompile("(?m)[\r\n]+^.*XXX_.*$")
|
|
str = re.ReplaceAllString(str, "")
|
|
fmt.Print(str)
|
|
}
|
|
|
|
func benchmarkHash(sszPath string, sszType string) {
|
|
switch sszType {
|
|
case "state_capella":
|
|
st := ðpb.BeaconStateCapella{}
|
|
rawFile, err := os.ReadFile(sszPath) // #nosec G304
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
|
|
startDeserialize := time.Now()
|
|
if err := st.UnmarshalSSZ(rawFile); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
deserializeDuration := time.Since(startDeserialize)
|
|
|
|
stateTrieState, err := state_native.InitializeFromProtoCapella(st)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
start := time.Now()
|
|
stat := &runtime.MemStats{}
|
|
runtime.ReadMemStats(stat)
|
|
root, err := stateTrieState.HashTreeRoot(context.Background())
|
|
if err != nil {
|
|
log.Fatal("couldn't hash")
|
|
}
|
|
newStat := &runtime.MemStats{}
|
|
runtime.ReadMemStats(newStat)
|
|
fmt.Printf("Deserialize Duration: %v, Hashing Duration: %v HTR: %#x\n", deserializeDuration, time.Since(start), root)
|
|
fmt.Printf("Total Memory Allocation Differential: %d bytes, Heap Memory Allocation Differential: %d bytes\n", int64(newStat.TotalAlloc)-int64(stat.TotalAlloc), int64(newStat.HeapAlloc)-int64(stat.HeapAlloc))
|
|
return
|
|
default:
|
|
log.Fatal("Invalid type")
|
|
}
|
|
}
|