From e636e246c5614f7f478012b2c54a5c82bae6aae0 Mon Sep 17 00:00:00 2001 From: laser Date: Thu, 19 Sep 2019 11:31:54 -0700 Subject: [PATCH 1/3] docs(jq): add jq to deps list --- README.md | 1 + 1 file changed, 1 insertion(+) diff --git a/README.md b/README.md index 566d510f6..885ccc12e 100644 --- a/README.md +++ b/README.md @@ -19,6 +19,7 @@ board](https://github.com/filecoin-project/go-lotus/projects/1). - git - bzr (some go dependency needs this) - b2sum +- jq *Building:* ``` From f7e3762cfc8f06c09f3dd14bbb5d1de36644664c Mon Sep 17 00:00:00 2001 From: whyrusleeping Date: Thu, 19 Sep 2019 13:25:18 -0700 Subject: [PATCH 2/3] add state replay api and command --- api/api.go | 7 ++++++ api/struct.go | 5 ++++ chain/stmgr/call.go | 22 ++++++++++++++++ chain/stmgr/stmgr.go | 45 ++++++++++++++++++++------------- cli/state.go | 56 +++++++++++++++++++++++++++++++++++++++++ node/impl/full/state.go | 19 ++++++++++++++ 6 files changed, 137 insertions(+), 17 deletions(-) diff --git a/api/api.go b/api/api.go index ee6c82661..c531624fe 100644 --- a/api/api.go +++ b/api/api.go @@ -108,6 +108,7 @@ type FullNode interface { StateMinerPeerID(ctx context.Context, m address.Address, ts *types.TipSet) (peer.ID, error) // if tipset is nil, we'll use heaviest StateCall(context.Context, *types.Message, *types.TipSet) (*types.MessageReceipt, error) + StateReplay(context.Context, *types.TipSet, cid.Cid) (*ReplayResults, error) StateGetActor(ctx context.Context, actor address.Address, ts *types.TipSet) (*types.Actor, error) StateReadState(ctx context.Context, act *types.Actor, ts *types.TipSet) (*ActorState, error) @@ -269,3 +270,9 @@ type RetrievalOrder struct { Miner address.Address MinerPeerID peer.ID } + +type ReplayResults struct { + Msg *types.Message + Receipt *types.MessageReceipt + Error string +} diff --git a/api/struct.go b/api/struct.go index 82f033ce6..87d891d6b 100644 --- a/api/struct.go +++ b/api/struct.go @@ -81,6 +81,7 @@ type FullNodeStruct struct { StateMinerWorker func(context.Context, address.Address, *types.TipSet) (address.Address, error) `perm:"read"` StateMinerPeerID func(ctx context.Context, m address.Address, ts *types.TipSet) (peer.ID, error) `perm:"read"` StateCall func(context.Context, *types.Message, *types.TipSet) (*types.MessageReceipt, error) `perm:"read"` + StateReplay func(context.Context, *types.TipSet, cid.Cid) (*ReplayResults, error) `perm:"read"` StateGetActor func(context.Context, address.Address, *types.TipSet) (*types.Actor, error) `perm:"read"` StateReadState func(context.Context, *types.Actor, *types.TipSet) (*ActorState, error) `perm:"read"` @@ -302,6 +303,10 @@ func (c *FullNodeStruct) StateCall(ctx context.Context, msg *types.Message, ts * return c.Internal.StateCall(ctx, msg, ts) } +func (c *FullNodeStruct) StateReplay(ctx context.Context, ts *types.TipSet, mc cid.Cid) (*ReplayResults, error) { + return c.Internal.StateReplay(ctx, ts, mc) +} + func (c *FullNodeStruct) StateGetActor(ctx context.Context, actor address.Address, ts *types.TipSet) (*types.Actor, error) { return c.Internal.StateGetActor(ctx, actor, ts) } diff --git a/chain/stmgr/call.go b/chain/stmgr/call.go index 607273724..62c7acabe 100644 --- a/chain/stmgr/call.go +++ b/chain/stmgr/call.go @@ -2,6 +2,7 @@ package stmgr import ( "context" + "fmt" "github.com/filecoin-project/go-lotus/chain/actors" "github.com/filecoin-project/go-lotus/chain/types" @@ -59,3 +60,24 @@ func (sm *StateManager) Call(ctx context.Context, msg *types.Message, ts *types. return sm.CallRaw(ctx, msg, state, ts.Height()) } + +var errHaltExecution = fmt.Errorf("halt") + +func (sm *StateManager) Replay(ctx context.Context, ts *types.TipSet, mcid cid.Cid) (*types.Message, *vm.ApplyRet, error) { + var outm *types.Message + var outr *vm.ApplyRet + + _, err := sm.computeTipSetState(ctx, ts.Blocks(), func(c cid.Cid, m *types.Message, ret *vm.ApplyRet) error { + if c == mcid { + outm = m + outr = ret + return errHaltExecution + } + return nil + }) + if err != nil && err != errHaltExecution { + return nil, nil, xerrors.Errorf("unexpected error during execution: %w", err) + } + + return outm, outr, nil +} diff --git a/chain/stmgr/stmgr.go b/chain/stmgr/stmgr.go index 447c1bd63..70511fa5e 100644 --- a/chain/stmgr/stmgr.go +++ b/chain/stmgr/stmgr.go @@ -41,6 +41,8 @@ func cidsToKey(cids []cid.Cid) string { } func (sm *StateManager) TipSetState(cids []cid.Cid) (cid.Cid, error) { + ctx := context.TODO() + ck := cidsToKey(cids) sm.stlk.Lock() cached, ok := sm.stCache[ck] @@ -49,7 +51,12 @@ func (sm *StateManager) TipSetState(cids []cid.Cid) (cid.Cid, error) { return cached, nil } - out, err := sm.computeTipSetState(cids) + ts, err := sm.cs.LoadTipSet(cids) + if err != nil { + return cid.Undef, err + } + + out, err := sm.computeTipSetState(ctx, ts.Blocks(), nil) if err != nil { return cid.Undef, err } @@ -60,31 +67,23 @@ func (sm *StateManager) TipSetState(cids []cid.Cid) (cid.Cid, error) { return out, nil } -func (sm *StateManager) computeTipSetState(cids []cid.Cid) (cid.Cid, error) { - ctx := context.TODO() - - ts, err := sm.cs.LoadTipSet(cids) - if err != nil { - log.Error("failed loading tipset: ", cids) - return cid.Undef, err +func (sm *StateManager) computeTipSetState(ctx context.Context, blks []*types.BlockHeader, cb func(cid.Cid, *types.Message, *vm.ApplyRet) error) (cid.Cid, error) { + if len(blks) == 1 && cb == nil { + return blks[0].StateRoot, nil } - if len(ts.Blocks()) == 1 { - return ts.Blocks()[0].StateRoot, nil - } - - pstate, err := sm.TipSetState(ts.Parents()) + pstate, err := sm.TipSetState(blks[0].Parents) if err != nil { return cid.Undef, xerrors.Errorf("recursive TipSetState failed: %w", err) } - vmi, err := vm.NewVM(pstate, ts.Height(), address.Undef, sm.cs) + vmi, err := vm.NewVM(pstate, blks[0].Height, address.Undef, sm.cs) if err != nil { return cid.Undef, xerrors.Errorf("instantiating VM failed: %w", err) } applied := make(map[cid.Cid]bool) - for _, b := range ts.Blocks() { + for _, b := range blks { vmi.SetBlockMiner(b.Miner) bms, sms, err := sm.cs.MessagesForBlock(b) @@ -98,10 +97,16 @@ func (sm *StateManager) computeTipSetState(cids []cid.Cid) (cid.Cid, error) { } applied[m.Cid()] = true - _, err := vmi.ApplyMessage(ctx, m) + r, err := vmi.ApplyMessage(ctx, m) if err != nil { return cid.Undef, err } + + if cb != nil { + if err := cb(m.Cid(), m, r); err != nil { + return cid.Undef, err + } + } } for _, sm := range sms { @@ -110,10 +115,16 @@ func (sm *StateManager) computeTipSetState(cids []cid.Cid) (cid.Cid, error) { } applied[sm.Cid()] = true - _, err := vmi.ApplyMessage(ctx, &sm.Message) + r, err := vmi.ApplyMessage(ctx, &sm.Message) if err != nil { return cid.Undef, err } + + if cb != nil { + if err := cb(sm.Cid(), &sm.Message, r); err != nil { + return cid.Undef, err + } + } } } diff --git a/cli/state.go b/cli/state.go index ce799093f..16396edb4 100644 --- a/cli/state.go +++ b/cli/state.go @@ -6,6 +6,8 @@ import ( "gopkg.in/urfave/cli.v2" "github.com/filecoin-project/go-lotus/chain/address" + "github.com/filecoin-project/go-lotus/chain/types" + "github.com/ipfs/go-cid" ) var stateCmd = &cli.Command{ @@ -116,3 +118,57 @@ var stateProvingSetCmd = &cli.Command{ return nil }, } + +var stateReplaySetCmd = &cli.Command{ + Name: "replay", + Usage: "Replay a particular message within a tipset", + Action: func(cctx *cli.Context) error { + if cctx.Args().Len() < 2 { + fmt.Println("usage: ") + fmt.Println("The last cid passed will be used as the message CID") + fmt.Println("All preceding ones will be used as the tipset") + return nil + } + + args := cctx.Args().Slice() + mcid, err := cid.Decode(args[len(args)-1]) + if err != nil { + return fmt.Errorf("message cid was invalid: %s", err) + } + + var tscids []cid.Cid + for _, s := range args[:len(args)-1] { + c, err := cid.Decode(s) + if err != nil { + return fmt.Errorf("tipset cid was invalid: %s", err) + } + tscids = append(tscids, c) + } + + api, err := GetFullNodeAPI(cctx) + if err != nil { + return err + } + + ctx := ReqContext(cctx) + + var headers []*types.BlockHeader + for _, c := range tscids { + h, err := api.ChainGetBlock(ctx, c) + if err != nil { + return err + } + + headers = append(headers, h) + } + + ts, err := types.NewTipSet(headers) + if err != nil { + return err + } + + api.StateReplay(ctx, ts, mcid) + + return nil + }, +} diff --git a/node/impl/full/state.go b/node/impl/full/state.go index 6b9bfe33d..57141846f 100644 --- a/node/impl/full/state.go +++ b/node/impl/full/state.go @@ -5,6 +5,7 @@ import ( "fmt" "strconv" + cid "github.com/ipfs/go-cid" "github.com/ipfs/go-hamt-ipld" cbor "github.com/ipfs/go-ipld-cbor" "github.com/libp2p/go-libp2p-core/peer" @@ -194,6 +195,24 @@ func (a *StateAPI) StateCall(ctx context.Context, msg *types.Message, ts *types. return a.StateManager.Call(ctx, msg, ts) } +func (a *StateAPI) StateReplay(ctx context.Context, ts *types.TipSet, mc cid.Cid) (*api.ReplayResults, error) { + m, r, err := a.StateManager.Replay(ctx, ts, mc) + if err != nil { + return nil, err + } + + var errstr string + if r.ActorErr != nil { + errstr = r.ActorErr.Error() + } + + return &api.ReplayResults{ + Msg: m, + Receipt: &r.MessageReceipt, + Error: errstr, + }, nil +} + func (a *StateAPI) stateForTs(ts *types.TipSet) (*state.StateTree, error) { if ts == nil { ts = a.Chain.GetHeaviestTipSet() From 61df25eecb687a5af695f17f9a210f2e8663bb19 Mon Sep 17 00:00:00 2001 From: whyrusleeping Date: Thu, 19 Sep 2019 23:46:15 -0700 Subject: [PATCH 3/3] fix import grouping --- cli/state.go | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/cli/state.go b/cli/state.go index 16396edb4..d9f07559b 100644 --- a/cli/state.go +++ b/cli/state.go @@ -3,11 +3,11 @@ package cli import ( "fmt" - "gopkg.in/urfave/cli.v2" - "github.com/filecoin-project/go-lotus/chain/address" "github.com/filecoin-project/go-lotus/chain/types" + "github.com/ipfs/go-cid" + "gopkg.in/urfave/cli.v2" ) var stateCmd = &cli.Command{