lotus/cli/sync.go

313 lines
6.8 KiB
Go
Raw Normal View History

package cli
import (
2019-12-04 00:25:18 +00:00
"context"
"fmt"
2019-10-13 07:33:25 +00:00
"time"
2020-09-09 07:42:03 +00:00
"github.com/filecoin-project/lotus/chain/types"
2020-09-07 03:49:10 +00:00
"github.com/filecoin-project/go-state-types/abi"
2019-10-13 07:33:25 +00:00
cid "github.com/ipfs/go-cid"
"github.com/urfave/cli/v2"
"github.com/filecoin-project/lotus/api"
"github.com/filecoin-project/lotus/build"
)
var syncCmd = &cli.Command{
Name: "sync",
Usage: "Inspect or interact with the chain syncer",
Subcommands: []*cli.Command{
syncStatusCmd,
2019-10-13 07:33:25 +00:00
syncWaitCmd,
syncMarkBadCmd,
2020-09-09 07:25:19 +00:00
syncUnmarkBadCmd,
syncCheckBadCmd,
2020-09-09 07:42:03 +00:00
syncCheckpointCmd,
},
}
var syncStatusCmd = &cli.Command{
Name: "status",
Usage: "check sync status",
Action: func(cctx *cli.Context) error {
2019-12-04 04:59:41 +00:00
apic, closer, err := GetFullNodeAPI(cctx)
if err != nil {
return err
}
2019-10-03 18:12:30 +00:00
defer closer()
ctx := ReqContext(cctx)
2019-12-04 04:59:41 +00:00
state, err := apic.SyncState(ctx)
if err != nil {
return err
}
fmt.Println("sync status:")
2019-11-16 01:05:16 +00:00
for i, ss := range state.ActiveSyncs {
fmt.Printf("worker %d:\n", i)
var base, target []cid.Cid
2019-11-20 19:44:38 +00:00
var heightDiff int64
2020-02-08 02:18:32 +00:00
var theight abi.ChainEpoch
2019-11-16 01:05:16 +00:00
if ss.Base != nil {
base = ss.Base.Cids()
2019-11-20 19:44:38 +00:00
heightDiff = int64(ss.Base.Height())
2019-11-16 01:05:16 +00:00
}
if ss.Target != nil {
target = ss.Target.Cids()
2019-11-20 19:44:38 +00:00
heightDiff = int64(ss.Target.Height()) - heightDiff
2019-12-04 04:59:41 +00:00
theight = ss.Target.Height()
2019-11-20 19:44:38 +00:00
} else {
heightDiff = 0
2019-11-16 01:05:16 +00:00
}
fmt.Printf("\tBase:\t%s\n", base)
2019-12-04 04:59:41 +00:00
fmt.Printf("\tTarget:\t%s (%d)\n", target, theight)
2019-11-20 19:44:38 +00:00
fmt.Printf("\tHeight diff:\t%d\n", heightDiff)
fmt.Printf("\tStage: %s\n", ss.Stage)
2019-11-16 01:05:16 +00:00
fmt.Printf("\tHeight: %d\n", ss.Height)
2019-12-04 04:59:41 +00:00
if ss.End.IsZero() {
if !ss.Start.IsZero() {
fmt.Printf("\tElapsed: %s\n", time.Since(ss.Start))
}
} else {
fmt.Printf("\tElapsed: %s\n", ss.End.Sub(ss.Start))
}
if ss.Stage == api.StageSyncErrored {
fmt.Printf("\tError: %s\n", ss.Message)
}
2019-11-16 01:05:16 +00:00
}
return nil
},
}
2019-10-13 07:33:25 +00:00
var syncWaitCmd = &cli.Command{
Name: "wait",
Usage: "Wait for sync to be complete",
Action: func(cctx *cli.Context) error {
napi, closer, err := GetFullNodeAPI(cctx)
if err != nil {
return err
}
defer closer()
ctx := ReqContext(cctx)
2019-12-04 00:25:18 +00:00
return SyncWait(ctx, napi)
},
}
2019-10-13 07:33:25 +00:00
var syncMarkBadCmd = &cli.Command{
2020-03-06 19:01:28 +00:00
Name: "mark-bad",
Usage: "Mark the given block as bad, will prevent syncing to a chain that contains it",
ArgsUsage: "[blockCid]",
Action: func(cctx *cli.Context) error {
napi, closer, err := GetFullNodeAPI(cctx)
if err != nil {
return err
}
defer closer()
ctx := ReqContext(cctx)
if !cctx.Args().Present() {
return fmt.Errorf("must specify block cid to mark")
}
bcid, err := cid.Decode(cctx.Args().First())
if err != nil {
return fmt.Errorf("failed to decode input as a cid: %s", err)
}
return napi.SyncMarkBad(ctx, bcid)
},
}
2020-09-09 07:25:19 +00:00
var syncUnmarkBadCmd = &cli.Command{
Name: "unmark-bad",
Usage: "Unmark the given block as bad, makes it possible to sync to a chain containing it",
ArgsUsage: "[blockCid]",
Action: func(cctx *cli.Context) error {
napi, closer, err := GetFullNodeAPI(cctx)
if err != nil {
return err
}
defer closer()
ctx := ReqContext(cctx)
if !cctx.Args().Present() {
return fmt.Errorf("must specify block cid to unmark")
}
bcid, err := cid.Decode(cctx.Args().First())
if err != nil {
return fmt.Errorf("failed to decode input as a cid: %s", err)
}
return napi.SyncUnmarkBad(ctx, bcid)
},
}
var syncCheckBadCmd = &cli.Command{
2020-03-06 19:01:28 +00:00
Name: "check-bad",
Usage: "check if the given block was marked bad, and for what reason",
ArgsUsage: "[blockCid]",
Action: func(cctx *cli.Context) error {
napi, closer, err := GetFullNodeAPI(cctx)
if err != nil {
return err
}
defer closer()
ctx := ReqContext(cctx)
if !cctx.Args().Present() {
return fmt.Errorf("must specify block cid to check")
}
bcid, err := cid.Decode(cctx.Args().First())
if err != nil {
return fmt.Errorf("failed to decode input as a cid: %s", err)
}
reason, err := napi.SyncCheckBad(ctx, bcid)
if err != nil {
return err
}
if reason == "" {
fmt.Println("block was not marked as bad")
return nil
}
fmt.Println(reason)
return nil
},
}
2020-09-09 07:42:03 +00:00
var syncCheckpointCmd = &cli.Command{
Name: "checkpoint",
Usage: "mark a certain tipset as checkpointed; the node will never fork away from this tipset",
ArgsUsage: "[tipsetKey]",
Flags: []cli.Flag{
&cli.Uint64Flag{
Name: "epoch",
Usage: "checkpoint the tipset at the given epoch",
},
},
Action: func(cctx *cli.Context) error {
napi, closer, err := GetFullNodeAPI(cctx)
if err != nil {
return err
}
defer closer()
ctx := ReqContext(cctx)
var ts *types.TipSet
if cctx.IsSet("epoch") {
ts, err = napi.ChainGetTipSetByHeight(ctx, abi.ChainEpoch(cctx.Uint64("epoch")), types.EmptyTSK)
}
if ts == nil {
ts, err = parseTipSet(ctx, napi, cctx.Args().Slice())
}
if err != nil {
return err
}
if ts == nil {
return fmt.Errorf("must pass cids for tipset to set as head, or specify epoch flag")
}
if err := napi.SyncCheckpoint(ctx, ts.Key()); err != nil {
return err
}
return nil
},
}
2019-12-04 00:25:18 +00:00
func SyncWait(ctx context.Context, napi api.FullNode) error {
2020-09-24 11:35:45 +00:00
tick := time.Second / 4
lastLines := 0
ticker := time.NewTicker(tick)
defer ticker.Stop()
samples := 8
i := 0
var app, lastApp uint64
2019-12-04 00:25:18 +00:00
for {
state, err := napi.SyncState(ctx)
if err != nil {
return err
}
head, err := napi.ChainHead(ctx)
if err != nil {
return err
}
2019-12-04 00:25:18 +00:00
working := 0
for i, ss := range state.ActiveSyncs {
switch ss.Stage {
case api.StageSyncComplete:
default:
working = i
case api.StageIdle:
// not complete, not actively working
2019-11-16 01:05:16 +00:00
}
2019-12-04 00:25:18 +00:00
}
2019-11-16 01:05:16 +00:00
2019-12-04 00:25:18 +00:00
ss := state.ActiveSyncs[working]
2019-11-16 01:05:16 +00:00
2020-09-19 06:04:48 +00:00
var baseHeight abi.ChainEpoch
2019-12-04 00:25:18 +00:00
var target []cid.Cid
2020-08-31 05:43:58 +00:00
var theight abi.ChainEpoch
2020-09-19 06:04:48 +00:00
var heightDiff int64
if ss.Base != nil {
baseHeight = ss.Base.Height()
heightDiff = int64(ss.Base.Height())
}
2019-12-04 00:25:18 +00:00
if ss.Target != nil {
target = ss.Target.Cids()
2020-08-31 05:43:58 +00:00
theight = ss.Target.Height()
2020-09-19 06:04:48 +00:00
heightDiff = int64(ss.Target.Height()) - heightDiff
} else {
heightDiff = 0
2019-12-04 00:25:18 +00:00
}
2019-10-13 07:33:25 +00:00
2020-09-24 11:35:45 +00:00
for i := 0; i < lastLines; i++ {
fmt.Print("\r\x1b[2K\x1b[A")
}
fmt.Printf("Worker: %d; Base: %d; Target: %d (diff: %d)\n", working, baseHeight, theight, heightDiff)
fmt.Printf("State: %s; Current Epoch: %d; Todo: %d\n", ss.Stage, ss.Height, theight-ss.Height)
lastLines = 2
if i%samples == 0 {
lastApp = app
app = state.VMApplied
}
if i > 0 {
fmt.Printf("Validated %d messages (%d per second)\n", state.VMApplied, (app-lastApp)*uint64(time.Second/tick)/uint64(samples))
lastLines++
}
_ = target // todo: maybe print? (creates a bunch of line wrapping issues with most tipsets)
if time.Now().Unix()-int64(head.MinTimestamp()) < int64(build.BlockDelaySecs) {
2019-12-04 00:25:18 +00:00
fmt.Println("\nDone!")
return nil
}
2019-10-13 07:33:25 +00:00
2019-12-04 00:25:18 +00:00
select {
case <-ctx.Done():
fmt.Println("\nExit by user")
return nil
2020-09-24 11:35:45 +00:00
case <-ticker.C:
2019-10-13 07:33:25 +00:00
}
2020-09-24 11:35:45 +00:00
i++
2019-12-04 00:25:18 +00:00
}
2019-10-13 07:33:25 +00:00
}