From ab61f4bb2c42b5f50057aeaf32226629f41e637c Mon Sep 17 00:00:00 2001 From: Phi Date: Tue, 30 Aug 2022 12:54:59 +0200 Subject: [PATCH 1/7] `lotus info` cmd Adds lotus info command that prints a lot of node information. --- cli/cmd.go | 1 + cli/info.go | 220 ++++++++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 221 insertions(+) create mode 100644 cli/info.go diff --git a/cli/cmd.go b/cli/cmd.go index 71524d787d2..79023917b46 100644 --- a/cli/cmd.go +++ b/cli/cmd.go @@ -69,6 +69,7 @@ var CommonCommands = []*cli.Command{ var Commands = []*cli.Command{ WithCategory("basic", sendCmd), WithCategory("basic", walletCmd), + WithCategory("basic", infoCmd), WithCategory("basic", clientCmd), WithCategory("basic", multisigCmd), WithCategory("basic", filplusCmd), diff --git a/cli/info.go b/cli/info.go new file mode 100644 index 00000000000..b88480a5a5b --- /dev/null +++ b/cli/info.go @@ -0,0 +1,220 @@ +package cli + +import ( + "fmt" + "math" + "os" + "sort" + "strings" + "text/tabwriter" + "time" + + "github.com/dustin/go-humanize" + "github.com/fatih/color" + "github.com/urfave/cli/v2" + + "github.com/filecoin-project/go-fil-markets/storagemarket" + "github.com/filecoin-project/go-state-types/big" + "github.com/filecoin-project/lotus/build" + "github.com/filecoin-project/lotus/chain/types" +) + +var infoCmd = &cli.Command{ + Name: "info", + Usage: "Print node info", + Action: infoCmdAct, +} + +func infoCmdAct(cctx *cli.Context) error { + fullapi, acloser, err := GetFullNodeAPIV1(cctx) + if err != nil { + return err + } + defer acloser() + ctx := ReqContext(cctx) + + network, err := fullapi.StateGetNetworkParams(ctx) + if err != nil { + return err + } + + fmt.Printf("Network: %s\n", network.NetworkName) + fmt.Print("Chain: ") + head, err := fullapi.ChainHead(ctx) + if err != nil { + return err + } + + switch { + case time.Now().Unix()-int64(head.MinTimestamp()) < int64(build.BlockDelaySecs*3/2): // within 1.5 epochs + fmt.Printf("[%s]", color.GreenString("sync ok")) + case time.Now().Unix()-int64(head.MinTimestamp()) < int64(build.BlockDelaySecs*5): // within 5 epochs + fmt.Printf("[%s]", color.YellowString("sync slow (%s behind)", time.Now().Sub(time.Unix(int64(head.MinTimestamp()), 0)).Truncate(time.Second))) + default: + fmt.Printf("[%s]", color.RedString("sync behind! (%s behind)", time.Now().Sub(time.Unix(int64(head.MinTimestamp()), 0)).Truncate(time.Second))) + } + basefee := head.MinTicketBlock().ParentBaseFee + gasCol := []color.Attribute{color.FgBlue} + switch { + case basefee.GreaterThan(big.NewInt(7000_000_000)): // 7 nFIL + gasCol = []color.Attribute{color.BgRed, color.FgBlack} + case basefee.GreaterThan(big.NewInt(3000_000_000)): // 3 nFIL + gasCol = []color.Attribute{color.FgRed} + case basefee.GreaterThan(big.NewInt(750_000_000)): // 750 uFIL + gasCol = []color.Attribute{color.FgYellow} + case basefee.GreaterThan(big.NewInt(100_000_000)): // 100 uFIL + gasCol = []color.Attribute{color.FgGreen} + } + fmt.Printf(" [basefee %s]", color.New(gasCol...).Sprint(types.FIL(basefee).Short())) + + status, err := fullapi.NodeStatus(ctx, true) + if err != nil { + return err + } + + fmt.Printf(" [epoch %s]\n", color.MagentaString(("%d"), status.SyncStatus.Epoch)) + fmt.Printf("Peers to: [publish messages %d] [publish blocks %d]\n", status.PeerStatus.PeersToPublishMsgs, status.PeerStatus.PeersToPublishBlocks) + + //Chain health calculated as percentage: amount of blocks in last finality / very healthy amount of blocks in a finality (900 epochs * 5 blocks per tipset) + health := (100 * (900 * status.ChainStatus.BlocksPerTipsetLastFinality) / (900 * 5)) + switch { + case health > 85: + fmt.Printf("Chain health: %.f%% [%s]\n", health, color.GreenString("healthy")) + case health < 85: + fmt.Printf("Chain health: %.f%% [%s]\n", health, color.RedString("unhealthy")) + } + + fmt.Println() + + addr, err := fullapi.WalletDefaultAddress(ctx) + if err != nil { + return err + } + + balance, err := fullapi.WalletBalance(ctx, addr) + if err != nil { + return err + } + + fmt.Printf("Default address: \n") + fmt.Printf(" %s [%s]\n", addr.String(), types.FIL(balance).Short()) + fmt.Println() + + addrs, err := fullapi.WalletList(ctx) + if err != nil { + return err + } + + totalBalance := big.Zero() + for _, addr := range addrs { + totbal, err := fullapi.WalletBalance(ctx, addr) + if err != nil { + return err + } + totalBalance = big.Add(totalBalance, totbal) + } + + switch { + case len(addrs) <= 1: + fmt.Printf("Wallet: %v address\n", len(addrs)) + case len(addrs) > 1: + fmt.Printf("Wallet: %v addresses\n", len(addrs)) + } + fmt.Printf(" Total balance: %s\n", types.FIL(totalBalance).Short()) + + mbLockedSum := big.Zero() + mbAvailableSum := big.Zero() + for _, addr := range addrs { + mbal, err := fullapi.StateMarketBalance(ctx, addr, types.EmptyTSK) + if err != nil { + if strings.Contains(err.Error(), "actor not found") { + continue + } else { + return err + } + } + mbLockedSum = big.Add(mbLockedSum, mbal.Locked) + mbAvailableSum = big.Add(mbAvailableSum, mbal.Escrow) + } + + fmt.Printf(" Market locked: %s\n", types.FIL(mbLockedSum).Short()) + fmt.Printf(" Market available: %s\n", types.FIL(mbAvailableSum).Short()) + + fmt.Println() + + chs, err := fullapi.PaychList(ctx) + if err != nil { + return err + } + + switch { + case len(chs) <= 1: + fmt.Printf("Payment Channels: %v channel\n", len(chs)) + case len(chs) > 1: + fmt.Printf("Payment Channels: %v channels\n", len(chs)) + } + fmt.Println() + + localDeals, err := fullapi.ClientListDeals(ctx) + if err != nil { + return err + } + + var totalSize uint64 + byState := map[storagemarket.StorageDealStatus][]uint64{} + for _, deal := range localDeals { + totalSize += deal.Size + byState[deal.State] = append(byState[deal.State], deal.Size) + } + + fmt.Printf("Deals: %d, %s\n", len(localDeals), types.SizeStr(types.NewInt(totalSize))) + + type stateStat struct { + state storagemarket.StorageDealStatus + count int + bytes uint64 + } + + stateStats := make([]stateStat, 0, len(byState)) + for state, deals := range byState { + if state == storagemarket.StorageDealActive { + state = math.MaxUint64 // for sort + } + + st := stateStat{ + state: state, + count: len(deals), + } + for _, b := range deals { + st.bytes += b + } + + stateStats = append(stateStats, st) + } + + sort.Slice(stateStats, func(i, j int) bool { + return int64(stateStats[i].state) < int64(stateStats[j].state) + }) + + for _, st := range stateStats { + if st.state == math.MaxUint64 { + st.state = storagemarket.StorageDealActive + } + fmt.Printf(" %s: %d deals, %s\n", storagemarket.DealStates[st.state], st.count, types.SizeStr(types.NewInt(st.bytes))) + } + + fmt.Println() + + tw := tabwriter.NewWriter(os.Stdout, 6, 6, 2, ' ', 0) + + s, err := fullapi.NetBandwidthStats(ctx) + if err != nil { + return err + } + + fmt.Printf("Bandwidth:\n") + fmt.Fprintf(tw, "\tTotalIn\tTotalOut\tRateIn\tRateOut\n") + fmt.Fprintf(tw, "\t%s\t%s\t%s/s\t%s/s\n", humanize.Bytes(uint64(s.TotalIn)), humanize.Bytes(uint64(s.TotalOut)), humanize.Bytes(uint64(s.RateIn)), humanize.Bytes(uint64(s.RateOut))) + return tw.Flush() + +} From 683900b5a28d9c0303ac50608f9ea1b9bf47d514 Mon Sep 17 00:00:00 2001 From: Phi Date: Tue, 30 Aug 2022 13:29:22 +0200 Subject: [PATCH 2/7] make docsgen-cli make docsgen-cli --- documentation/en/cli-lotus.md | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) diff --git a/documentation/en/cli-lotus.md b/documentation/en/cli-lotus.md index 5cb6b42aa8d..15178d8e0e0 100644 --- a/documentation/en/cli-lotus.md +++ b/documentation/en/cli-lotus.md @@ -18,6 +18,7 @@ COMMANDS: BASIC: send Send funds between accounts wallet Manage wallet + info Print node info client Make deals, store data, retrieve data msig Interact with a multisig wallet filplus Interact with the verified registry actor used by Filplus @@ -398,6 +399,22 @@ OPTIONS: ``` +## lotus info +``` +NAME: + lotus info - Print node info + +USAGE: + lotus info [command options] [arguments...] + +CATEGORY: + BASIC + +OPTIONS: + --help, -h show help (default: false) + +``` + ## lotus client ``` NAME: From 8a45e8c8af836d917759dd1a44701a5361a325ec Mon Sep 17 00:00:00 2001 From: Phi Date: Wed, 31 Aug 2022 10:40:50 +0200 Subject: [PATCH 3/7] Handle "Default address not set" better Only print address and get balance if err == nil, else print that the Default address is not set --- cli/info.go | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/cli/info.go b/cli/info.go index b88480a5a5b..8def9e1777a 100644 --- a/cli/info.go +++ b/cli/info.go @@ -87,17 +87,17 @@ func infoCmdAct(cctx *cli.Context) error { fmt.Println() addr, err := fullapi.WalletDefaultAddress(ctx) - if err != nil { - return err - } - - balance, err := fullapi.WalletBalance(ctx, addr) - if err != nil { - return err + if err == nil { + fmt.Printf("Default address: \n") + balance, err := fullapi.WalletBalance(ctx, addr) + if err != nil { + return err + } + fmt.Printf(" %s [%s]\n", addr.String(), types.FIL(balance).Short()) + fmt.Println() + } else { + fmt.Printf("Default address: address not set\n") } - - fmt.Printf("Default address: \n") - fmt.Printf(" %s [%s]\n", addr.String(), types.FIL(balance).Short()) fmt.Println() addrs, err := fullapi.WalletList(ctx) From 870c2f955490826920e2180b22c1eb4fb22d6ba2 Mon Sep 17 00:00:00 2001 From: Phi Date: Wed, 31 Aug 2022 10:44:59 +0200 Subject: [PATCH 4/7] Move newline Move newline --- cli/info.go | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/cli/info.go b/cli/info.go index 8def9e1777a..35707be8459 100644 --- a/cli/info.go +++ b/cli/info.go @@ -97,8 +97,8 @@ func infoCmdAct(cctx *cli.Context) error { fmt.Println() } else { fmt.Printf("Default address: address not set\n") + fmt.Println() } - fmt.Println() addrs, err := fullapi.WalletList(ctx) if err != nil { From 818153a1e71a14f913572f1b5659fb8ad250b4af Mon Sep 17 00:00:00 2001 From: Phi Date: Wed, 31 Aug 2022 13:15:28 +0200 Subject: [PATCH 5/7] Put SyncBasefeeCheck in function Put SyncBasefeeCheck in function --- cli/info.go | 56 +++++++++++++++++++++++++++++++---------------------- 1 file changed, 33 insertions(+), 23 deletions(-) diff --git a/cli/info.go b/cli/info.go index 35707be8459..b23ac9df4a1 100644 --- a/cli/info.go +++ b/cli/info.go @@ -1,6 +1,7 @@ package cli import ( + "context" "fmt" "math" "os" @@ -15,6 +16,7 @@ import ( "github.com/filecoin-project/go-fil-markets/storagemarket" "github.com/filecoin-project/go-state-types/big" + "github.com/filecoin-project/lotus/api/v1api" "github.com/filecoin-project/lotus/build" "github.com/filecoin-project/lotus/chain/types" ) @@ -40,33 +42,11 @@ func infoCmdAct(cctx *cli.Context) error { fmt.Printf("Network: %s\n", network.NetworkName) fmt.Print("Chain: ") - head, err := fullapi.ChainHead(ctx) + err = SyncBasefeeCheck(ctx, fullapi) if err != nil { return err } - switch { - case time.Now().Unix()-int64(head.MinTimestamp()) < int64(build.BlockDelaySecs*3/2): // within 1.5 epochs - fmt.Printf("[%s]", color.GreenString("sync ok")) - case time.Now().Unix()-int64(head.MinTimestamp()) < int64(build.BlockDelaySecs*5): // within 5 epochs - fmt.Printf("[%s]", color.YellowString("sync slow (%s behind)", time.Now().Sub(time.Unix(int64(head.MinTimestamp()), 0)).Truncate(time.Second))) - default: - fmt.Printf("[%s]", color.RedString("sync behind! (%s behind)", time.Now().Sub(time.Unix(int64(head.MinTimestamp()), 0)).Truncate(time.Second))) - } - basefee := head.MinTicketBlock().ParentBaseFee - gasCol := []color.Attribute{color.FgBlue} - switch { - case basefee.GreaterThan(big.NewInt(7000_000_000)): // 7 nFIL - gasCol = []color.Attribute{color.BgRed, color.FgBlack} - case basefee.GreaterThan(big.NewInt(3000_000_000)): // 3 nFIL - gasCol = []color.Attribute{color.FgRed} - case basefee.GreaterThan(big.NewInt(750_000_000)): // 750 uFIL - gasCol = []color.Attribute{color.FgYellow} - case basefee.GreaterThan(big.NewInt(100_000_000)): // 100 uFIL - gasCol = []color.Attribute{color.FgGreen} - } - fmt.Printf(" [basefee %s]", color.New(gasCol...).Sprint(types.FIL(basefee).Short())) - status, err := fullapi.NodeStatus(ctx, true) if err != nil { return err @@ -218,3 +198,33 @@ func infoCmdAct(cctx *cli.Context) error { return tw.Flush() } + +func SyncBasefeeCheck(ctx context.Context, fullapi v1api.FullNode) error { + head, err := fullapi.ChainHead(ctx) + if err != nil { + return err + } + + switch { + case time.Now().Unix()-int64(head.MinTimestamp()) < int64(build.BlockDelaySecs*3/2): // within 1.5 epochs + fmt.Printf("[%s]", color.GreenString("sync ok")) + case time.Now().Unix()-int64(head.MinTimestamp()) < int64(build.BlockDelaySecs*5): // within 5 epochs + fmt.Printf("[%s]", color.YellowString("sync slow (%s behind)", time.Now().Sub(time.Unix(int64(head.MinTimestamp()), 0)).Truncate(time.Second))) + default: + fmt.Printf("[%s]", color.RedString("sync behind! (%s behind)", time.Now().Sub(time.Unix(int64(head.MinTimestamp()), 0)).Truncate(time.Second))) + } + basefee := head.MinTicketBlock().ParentBaseFee + gasCol := []color.Attribute{color.FgBlue} + switch { + case basefee.GreaterThan(big.NewInt(7000_000_000)): // 7 nFIL + gasCol = []color.Attribute{color.BgRed, color.FgBlack} + case basefee.GreaterThan(big.NewInt(3000_000_000)): // 3 nFIL + gasCol = []color.Attribute{color.FgRed} + case basefee.GreaterThan(big.NewInt(750_000_000)): // 750 uFIL + gasCol = []color.Attribute{color.FgYellow} + case basefee.GreaterThan(big.NewInt(100_000_000)): // 100 uFIL + gasCol = []color.Attribute{color.FgGreen} + } + fmt.Printf(" [basefee %s]", color.New(gasCol...).Sprint(types.FIL(basefee).Short())) + return nil +} From a31b8a1b58ae65749650ffe1b7f04e9f5c97d894 Mon Sep 17 00:00:00 2001 From: Phi-rjan Date: Fri, 2 Sep 2022 15:12:32 +0200 Subject: [PATCH 6/7] Update cli/info.go MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Co-authored-by: Ɓukasz Magiera --- cli/info.go | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/cli/info.go b/cli/info.go index b23ac9df4a1..5a3d6da5f94 100644 --- a/cli/info.go +++ b/cli/info.go @@ -74,11 +74,10 @@ func infoCmdAct(cctx *cli.Context) error { return err } fmt.Printf(" %s [%s]\n", addr.String(), types.FIL(balance).Short()) - fmt.Println() } else { fmt.Printf("Default address: address not set\n") - fmt.Println() } + fmt.Println() addrs, err := fullapi.WalletList(ctx) if err != nil { From ab0699875a7f51c3fbcd532f812839ee883a0f46 Mon Sep 17 00:00:00 2001 From: Phi Date: Fri, 2 Sep 2022 15:20:36 +0200 Subject: [PATCH 7/7] make gen happy make gen happy not sad --- cli/info.go | 1 + 1 file changed, 1 insertion(+) diff --git a/cli/info.go b/cli/info.go index 5a3d6da5f94..0d036875a43 100644 --- a/cli/info.go +++ b/cli/info.go @@ -16,6 +16,7 @@ import ( "github.com/filecoin-project/go-fil-markets/storagemarket" "github.com/filecoin-project/go-state-types/big" + "github.com/filecoin-project/lotus/api/v1api" "github.com/filecoin-project/lotus/build" "github.com/filecoin-project/lotus/chain/types"