Merge pull request #1662 from filecoin-project/feat/compute-state-html
compute-state: Add html output option
This commit is contained in:
commit
f7e6e45442
262
cli/state.go
262
cli/state.go
@ -8,29 +8,83 @@ import (
|
|||||||
"reflect"
|
"reflect"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
"github.com/filecoin-project/specs-actors/actors/builtin"
|
"github.com/docker/go-units"
|
||||||
|
"github.com/ipfs/go-cid"
|
||||||
|
"github.com/libp2p/go-libp2p-core/peer"
|
||||||
|
"github.com/multiformats/go-multihash"
|
||||||
|
cbg "github.com/whyrusleeping/cbor-gen"
|
||||||
|
"golang.org/x/xerrors"
|
||||||
|
"gopkg.in/urfave/cli.v2"
|
||||||
|
|
||||||
"github.com/filecoin-project/go-address"
|
"github.com/filecoin-project/go-address"
|
||||||
"github.com/filecoin-project/specs-actors/actors/abi"
|
"github.com/filecoin-project/specs-actors/actors/abi"
|
||||||
|
"github.com/filecoin-project/specs-actors/actors/builtin"
|
||||||
|
"github.com/filecoin-project/specs-actors/actors/builtin/account"
|
||||||
|
"github.com/filecoin-project/specs-actors/actors/builtin/cron"
|
||||||
|
init_ "github.com/filecoin-project/specs-actors/actors/builtin/init"
|
||||||
"github.com/filecoin-project/specs-actors/actors/builtin/market"
|
"github.com/filecoin-project/specs-actors/actors/builtin/market"
|
||||||
miner2 "github.com/filecoin-project/specs-actors/actors/builtin/miner"
|
miner2 "github.com/filecoin-project/specs-actors/actors/builtin/miner"
|
||||||
samsig "github.com/filecoin-project/specs-actors/actors/builtin/multisig"
|
"github.com/filecoin-project/specs-actors/actors/builtin/multisig"
|
||||||
"github.com/filecoin-project/specs-actors/actors/builtin/paych"
|
"github.com/filecoin-project/specs-actors/actors/builtin/paych"
|
||||||
"github.com/filecoin-project/specs-actors/actors/builtin/power"
|
"github.com/filecoin-project/specs-actors/actors/builtin/power"
|
||||||
"github.com/libp2p/go-libp2p-core/peer"
|
"github.com/filecoin-project/specs-actors/actors/builtin/reward"
|
||||||
"golang.org/x/xerrors"
|
"github.com/filecoin-project/specs-actors/actors/builtin/verifreg"
|
||||||
|
"github.com/filecoin-project/specs-actors/actors/util/adt"
|
||||||
|
|
||||||
"github.com/filecoin-project/lotus/api"
|
"github.com/filecoin-project/lotus/api"
|
||||||
"github.com/filecoin-project/lotus/chain/types"
|
"github.com/filecoin-project/lotus/chain/types"
|
||||||
"github.com/filecoin-project/lotus/miner"
|
"github.com/filecoin-project/lotus/miner"
|
||||||
|
|
||||||
"github.com/docker/go-units"
|
|
||||||
"github.com/ipfs/go-cid"
|
|
||||||
cbg "github.com/whyrusleeping/cbor-gen"
|
|
||||||
"gopkg.in/urfave/cli.v2"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type methodMeta struct {
|
||||||
|
name string
|
||||||
|
|
||||||
|
params reflect.Type
|
||||||
|
ret reflect.Type
|
||||||
|
}
|
||||||
|
|
||||||
|
var methods = map[cid.Cid][]methodMeta{}
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
cidToMethods := map[cid.Cid][2]interface{}{
|
||||||
|
// builtin.SystemActorCodeID: {builtin.MethodsSystem, system.Actor{} }- apparently it doesn't have methods
|
||||||
|
builtin.InitActorCodeID: {builtin.MethodsInit, init_.Actor{}},
|
||||||
|
builtin.CronActorCodeID: {builtin.MethodsCron, cron.Actor{}},
|
||||||
|
builtin.AccountActorCodeID: {builtin.MethodsAccount, account.Actor{}},
|
||||||
|
builtin.StoragePowerActorCodeID: {builtin.MethodsPower, power.Actor{}},
|
||||||
|
builtin.StorageMinerActorCodeID: {builtin.MethodsMiner, miner2.Actor{}},
|
||||||
|
builtin.StorageMarketActorCodeID: {builtin.MethodsMarket, market.Actor{}},
|
||||||
|
builtin.PaymentChannelActorCodeID: {builtin.MethodsPaych, paych.Actor{}},
|
||||||
|
builtin.MultisigActorCodeID: {builtin.MethodsMultisig, multisig.Actor{}},
|
||||||
|
builtin.RewardActorCodeID: {builtin.MethodsReward, reward.Actor{}},
|
||||||
|
builtin.VerifiedRegistryActorCodeID: {builtin.MethodsVerifiedRegistry, verifreg.Actor{}},
|
||||||
|
}
|
||||||
|
|
||||||
|
for c, m := range cidToMethods {
|
||||||
|
rt := reflect.TypeOf(m[0])
|
||||||
|
nf := rt.NumField()
|
||||||
|
|
||||||
|
methods[c] = append(methods[c], methodMeta{
|
||||||
|
name: "Send",
|
||||||
|
params: reflect.TypeOf(new(adt.EmptyValue)),
|
||||||
|
ret: reflect.TypeOf(new(adt.EmptyValue)),
|
||||||
|
})
|
||||||
|
|
||||||
|
exports := m[1].(abi.Invokee).Exports()
|
||||||
|
for i := 0; i < nf; i++ {
|
||||||
|
export := reflect.TypeOf(exports[i+1])
|
||||||
|
|
||||||
|
methods[c] = append(methods[c], methodMeta{
|
||||||
|
name: rt.Field(i).Name,
|
||||||
|
params: export.In(1),
|
||||||
|
ret: export.Out(0),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var stateCmd = &cli.Command{
|
var stateCmd = &cli.Command{
|
||||||
Name: "state",
|
Name: "state",
|
||||||
Usage: "Interact with and query filecoin chain state",
|
Usage: "Interact with and query filecoin chain state",
|
||||||
@ -769,6 +823,10 @@ var stateComputeStateCmd = &cli.Command{
|
|||||||
Name: "show-trace",
|
Name: "show-trace",
|
||||||
Usage: "print out full execution trace for given tipset",
|
Usage: "print out full execution trace for given tipset",
|
||||||
},
|
},
|
||||||
|
&cli.BoolFlag{
|
||||||
|
Name: "html",
|
||||||
|
Usage: "generate html report",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
Action: func(cctx *cli.Context) error {
|
Action: func(cctx *cli.Context) error {
|
||||||
api, closer, err := GetFullNodeAPI(cctx)
|
api, closer, err := GetFullNodeAPI(cctx)
|
||||||
@ -818,6 +876,25 @@ var stateComputeStateCmd = &cli.Command{
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if cctx.Bool("html") {
|
||||||
|
codeCache := map[address.Address]cid.Cid{}
|
||||||
|
getCode := func(addr address.Address) (cid.Cid, error) {
|
||||||
|
if c, found := codeCache[addr]; found {
|
||||||
|
return c, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
c, err := api.StateGetActor(ctx, addr, ts.Key())
|
||||||
|
if err != nil {
|
||||||
|
return cid.Cid{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
codeCache[addr] = c.Code
|
||||||
|
return c.Code, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return computeStateHtml(stout, getCode)
|
||||||
|
}
|
||||||
|
|
||||||
fmt.Println("computed state cid: ", stout.Root)
|
fmt.Println("computed state cid: ", stout.Root)
|
||||||
if cctx.Bool("show-trace") {
|
if cctx.Bool("show-trace") {
|
||||||
for _, ir := range stout.Trace {
|
for _, ir := range stout.Trace {
|
||||||
@ -836,6 +913,171 @@ func printInternalExecutions(prefix string, trace []*types.ExecutionResult) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func codeStr(c cid.Cid) string {
|
||||||
|
cmh, err := multihash.Decode(c.Hash())
|
||||||
|
if err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
return string(cmh.Digest)
|
||||||
|
}
|
||||||
|
|
||||||
|
func computeStateHtml(o *api.ComputeStateOutput, getCode func(addr address.Address) (cid.Cid, error)) error {
|
||||||
|
fmt.Printf(`<html>
|
||||||
|
<head>
|
||||||
|
<style>
|
||||||
|
html, body { font-family: monospace; }
|
||||||
|
a:link, a:visited { color: #004; }
|
||||||
|
pre { background: #ccc; }
|
||||||
|
small { color: #444; }
|
||||||
|
.call { color: #00a; }
|
||||||
|
.params { background: #dfd; }
|
||||||
|
.ret { background: #ddf; }
|
||||||
|
.error { color: red; }
|
||||||
|
.exit0 { color: green; }
|
||||||
|
.exec {
|
||||||
|
padding-left: 15px;
|
||||||
|
border-left: 2.5px solid;
|
||||||
|
margin-bottom: 45px;
|
||||||
|
}
|
||||||
|
.exec:hover {
|
||||||
|
background: #eee;
|
||||||
|
}
|
||||||
|
.slow-true-false { color: #660; }
|
||||||
|
.slow-true-true { color: #f80; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div>State CID: <b>%s</b></div>
|
||||||
|
<div>Calls</div>`, o.Root)
|
||||||
|
|
||||||
|
for _, ir := range o.Trace {
|
||||||
|
toCode, err := getCode(ir.Msg.To)
|
||||||
|
if err != nil {
|
||||||
|
return xerrors.Errorf("getting code for %s: %w", toCode, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
params, err := jsonParams(toCode, ir.Msg.Method, ir.Msg.Params)
|
||||||
|
if err != nil {
|
||||||
|
return xerrors.Errorf("decoding params: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(ir.Msg.Params) != 0 {
|
||||||
|
params = `<div><pre class="params">` + params + `</pre></div>`
|
||||||
|
} else {
|
||||||
|
params = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
ret, err := jsonReturn(toCode, ir.Msg.Method, ir.MsgRct.Return)
|
||||||
|
if err != nil {
|
||||||
|
return xerrors.Errorf("decoding return value: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(ir.MsgRct.Return) == 0 {
|
||||||
|
ret = "</div>"
|
||||||
|
} else {
|
||||||
|
ret = `, Return</div><div><pre class="ret">` + ret + `</pre></div>`
|
||||||
|
}
|
||||||
|
|
||||||
|
slow := ir.Duration > 10*time.Millisecond
|
||||||
|
veryslow := ir.Duration > 50*time.Millisecond
|
||||||
|
|
||||||
|
fmt.Printf(`<div class="exec">
|
||||||
|
<div><h2 class="call">%s:%s</h2></div>
|
||||||
|
<div><b>%s</b> -> <b>%s</b> (%s FIL), M%d</div>
|
||||||
|
<div><small>Msg CID: %s</small></div>
|
||||||
|
%s
|
||||||
|
<div><span class="slow-%t-%t">Took %s</span>, <span class="exit%d">Exit: <b>%d</b></span>%s
|
||||||
|
`, codeStr(toCode), methods[toCode][ir.Msg.Method].name, ir.Msg.From, ir.Msg.To, types.FIL(ir.Msg.Value), ir.Msg.Method, ir.Msg.Cid(), params, slow, veryslow, ir.Duration, ir.MsgRct.ExitCode, ir.MsgRct.ExitCode, ret)
|
||||||
|
if ir.MsgRct.ExitCode != 0 {
|
||||||
|
fmt.Printf(`<div class="error">Error: <pre>%s</pre></div>`, ir.Error)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(ir.InternalExecutions) > 0 {
|
||||||
|
fmt.Println("<div>Internal executions:</div>")
|
||||||
|
if err := printInternalExecutionsHtml(ir.InternalExecutions, getCode); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fmt.Println("</div>")
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Printf(`</body>
|
||||||
|
</html>`)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func printInternalExecutionsHtml(trace []*types.ExecutionResult, getCode func(addr address.Address) (cid.Cid, error)) error {
|
||||||
|
for _, im := range trace {
|
||||||
|
toCode, err := getCode(im.Msg.To)
|
||||||
|
if err != nil {
|
||||||
|
return xerrors.Errorf("getting code for %s: %w", toCode, err)
|
||||||
|
}
|
||||||
|
|
||||||
|
params, err := jsonParams(toCode, im.Msg.Method, im.Msg.Params)
|
||||||
|
if err != nil {
|
||||||
|
return xerrors.Errorf("decoding params: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(im.Msg.Params) != 0 {
|
||||||
|
params = `<div><pre class="params">` + params + `</pre></div>`
|
||||||
|
} else {
|
||||||
|
params = ""
|
||||||
|
}
|
||||||
|
|
||||||
|
ret, err := jsonReturn(toCode, im.Msg.Method, im.MsgRct.Return)
|
||||||
|
if err != nil {
|
||||||
|
return xerrors.Errorf("decoding return value: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(im.MsgRct.Return) == 0 {
|
||||||
|
ret = "</div>"
|
||||||
|
} else {
|
||||||
|
ret = `, Return</div><div><pre class="ret">` + ret + `</pre></div>`
|
||||||
|
}
|
||||||
|
|
||||||
|
fmt.Printf(`<div class="exec">
|
||||||
|
<div><h4 class="call">%s:%s</h4></div>
|
||||||
|
<div><b>%s</b> -> <b>%s</b> (%s FIL), M%d</div>
|
||||||
|
%s
|
||||||
|
<div><span class="exit%d">Exit: <b>%d</b></span>%s
|
||||||
|
`, codeStr(toCode), methods[toCode][im.Msg.Method].name, im.Msg.From, im.Msg.To, types.FIL(im.Msg.Value), im.Msg.Method, params, im.MsgRct.ExitCode, im.MsgRct.ExitCode, ret)
|
||||||
|
if im.MsgRct.ExitCode != 0 {
|
||||||
|
fmt.Printf(`<div class="error">Error: <pre>%s</pre></div>`, im.Error)
|
||||||
|
}
|
||||||
|
if len(im.Subcalls) > 0 {
|
||||||
|
fmt.Println("<div>Subcalls:</div>")
|
||||||
|
if err := printInternalExecutionsHtml(im.Subcalls, getCode); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
fmt.Println("</div>")
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func jsonParams(code cid.Cid, method abi.MethodNum, params []byte) (string, error) {
|
||||||
|
re := reflect.New(methods[code][method].params.Elem())
|
||||||
|
p := re.Interface().(cbg.CBORUnmarshaler)
|
||||||
|
if err := p.UnmarshalCBOR(bytes.NewReader(params)); err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
b, err := json.MarshalIndent(p, "", " ")
|
||||||
|
return string(b), err
|
||||||
|
}
|
||||||
|
|
||||||
|
func jsonReturn(code cid.Cid, method abi.MethodNum, ret []byte) (string, error) {
|
||||||
|
re := reflect.New(methods[code][method].ret.Elem())
|
||||||
|
p := re.Interface().(cbg.CBORUnmarshaler)
|
||||||
|
if err := p.UnmarshalCBOR(bytes.NewReader(ret)); err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
|
||||||
|
b, err := json.MarshalIndent(p, "", " ")
|
||||||
|
return string(b), err
|
||||||
|
}
|
||||||
|
|
||||||
var stateWaitMsgCmd = &cli.Command{
|
var stateWaitMsgCmd = &cli.Command{
|
||||||
Name: "wait-msg",
|
Name: "wait-msg",
|
||||||
Usage: "Wait for a message to appear on chain",
|
Usage: "Wait for a message to appear on chain",
|
||||||
@ -1072,7 +1314,7 @@ func parseParamsForMethod(act cid.Cid, method uint64, args []string) ([]byte, er
|
|||||||
case builtin.StoragePowerActorCodeID:
|
case builtin.StoragePowerActorCodeID:
|
||||||
f = power.Actor{}.Exports()[method]
|
f = power.Actor{}.Exports()[method]
|
||||||
case builtin.MultisigActorCodeID:
|
case builtin.MultisigActorCodeID:
|
||||||
f = samsig.Actor{}.Exports()[method]
|
f = multisig.Actor{}.Exports()[method]
|
||||||
case builtin.PaymentChannelActorCodeID:
|
case builtin.PaymentChannelActorCodeID:
|
||||||
f = paych.Actor{}.Exports()[method]
|
f = paych.Actor{}.Exports()[method]
|
||||||
default:
|
default:
|
||||||
|
1
go.mod
1
go.mod
@ -89,7 +89,6 @@ require (
|
|||||||
github.com/libp2p/go-libp2p-yamux v0.2.7
|
github.com/libp2p/go-libp2p-yamux v0.2.7
|
||||||
github.com/libp2p/go-maddr-filter v0.0.5
|
github.com/libp2p/go-maddr-filter v0.0.5
|
||||||
github.com/minio/blake2b-simd v0.0.0-20160723061019-3f5f724cb5b1
|
github.com/minio/blake2b-simd v0.0.0-20160723061019-3f5f724cb5b1
|
||||||
github.com/minio/sha256-simd v0.1.1
|
|
||||||
github.com/mitchellh/go-homedir v1.1.0
|
github.com/mitchellh/go-homedir v1.1.0
|
||||||
github.com/multiformats/go-base32 v0.0.3
|
github.com/multiformats/go-base32 v0.0.3
|
||||||
github.com/multiformats/go-multiaddr v0.2.1
|
github.com/multiformats/go-multiaddr v0.2.1
|
||||||
|
Loading…
Reference in New Issue
Block a user