From 180ab5228aa00e6bb664352745049f5ea712d745 Mon Sep 17 00:00:00 2001 From: Jeremy Chou Date: Sat, 28 May 2022 00:35:15 +0800 Subject: [PATCH] [ioctl] Refactor nodedelegate command in new ioctl (#3363) * refactor nodedelegate command in new ioctl --- ioctl/newcmd/bc/bc.go | 19 ++- ioctl/newcmd/node/nodedelegate.go | 187 +++++++++++++------------ ioctl/newcmd/node/nodedelegate_test.go | 20 ++- 3 files changed, 127 insertions(+), 99 deletions(-) diff --git a/ioctl/newcmd/bc/bc.go b/ioctl/newcmd/bc/bc.go index f3c220c8f8..5ca5de2de0 100644 --- a/ioctl/newcmd/bc/bc.go +++ b/ioctl/newcmd/bc/bc.go @@ -104,8 +104,7 @@ func GetChainMeta(client ioctl.Client) (*iotextypes.ChainMeta, error) { } // GetEpochMeta gets blockchain epoch meta -func GetEpochMeta(epochNum uint64, client ioctl.Client) (*iotexapi.GetEpochMetaResponse, error) { - +func GetEpochMeta(client ioctl.Client, epochNum uint64) (*iotexapi.GetEpochMetaResponse, error) { var endpoint string var insecure bool apiServiceClient, err := client.APIServiceClient(ioctl.APIServiceConfig{ @@ -116,7 +115,12 @@ func GetEpochMeta(epochNum uint64, client ioctl.Client) (*iotexapi.GetEpochMetaR return nil, err } - epochMetaresponse, err := apiServiceClient.GetEpochMeta(context.Background(), &iotexapi.GetEpochMetaRequest{}) + ctx := context.Background() + jwtMD, err := util.JwtAuth() + if err == nil { + ctx = metautils.NiceMD(jwtMD).ToOutgoing(ctx) + } + epochMetaresponse, err := apiServiceClient.GetEpochMeta(ctx, &iotexapi.GetEpochMetaRequest{}) if err != nil { sta, ok := status.FromError(err) if ok { @@ -128,7 +132,7 @@ func GetEpochMeta(epochNum uint64, client ioctl.Client) (*iotexapi.GetEpochMetaR } // GetProbationList gets probation list -func GetProbationList(epochNum uint64, client ioctl.Client) (*iotexapi.ReadStateResponse, error) { +func GetProbationList(client ioctl.Client, epochNum uint64) (*iotexapi.ReadStateResponse, error) { var endpoint string var insecure bool apiServiceClient, err := client.APIServiceClient(ioctl.APIServiceConfig{ @@ -139,13 +143,18 @@ func GetProbationList(epochNum uint64, client ioctl.Client) (*iotexapi.ReadState return nil, err } + ctx := context.Background() + jwtMD, err := util.JwtAuth() + if err == nil { + ctx = metautils.NiceMD(jwtMD).ToOutgoing(ctx) + } request := &iotexapi.ReadStateRequest{ ProtocolID: []byte("poll"), MethodName: []byte("ProbationListByEpoch"), Arguments: [][]byte{[]byte(strconv.FormatUint(epochNum, 10))}, } - response, err := apiServiceClient.ReadState(context.Background(), request) + response, err := apiServiceClient.ReadState(ctx, request) if err != nil { sta, ok := status.FromError(err) if ok && sta.Code() == codes.NotFound { diff --git a/ioctl/newcmd/node/nodedelegate.go b/ioctl/newcmd/node/nodedelegate.go index 0fc15f4849..d151f5d7f0 100644 --- a/ioctl/newcmd/node/nodedelegate.go +++ b/ioctl/newcmd/node/nodedelegate.go @@ -1,3 +1,9 @@ +// Copyright (c) 2022 IoTeX +// This is an alpha (internal) release and is not suitable for production. This source code is provided 'as is' and no +// warranties are given as to title or non-infringement, merchantability or fitness for purpose and, to the extent +// permitted by law, all liability for your use of the code is disclaimed. This source code is governed by Apache +// License 2.0 that can be found in the LICENSE file. + package node import ( @@ -7,7 +13,9 @@ import ( "strconv" "strings" + "github.com/grpc-ecosystem/go-grpc-middleware/util/metautils" "github.com/iotexproject/iotex-proto/golang/iotexapi" + "github.com/pkg/errors" "github.com/spf13/cobra" "google.golang.org/grpc/codes" "google.golang.org/grpc/status" @@ -16,7 +24,6 @@ import ( "github.com/iotexproject/iotex-core/ioctl" "github.com/iotexproject/iotex-core/ioctl/config" "github.com/iotexproject/iotex-core/ioctl/newcmd/bc" - "github.com/iotexproject/iotex-core/ioctl/output" "github.com/iotexproject/iotex-core/ioctl/util" "github.com/iotexproject/iotex-core/state" ) @@ -42,8 +49,6 @@ var ( ) var ( - _epochNum uint64 - _nextEpoch bool _nodeStatus map[bool]string _probatedStatus map[bool]string ) @@ -72,14 +77,18 @@ type delegatesMessage struct { } // NewNodeDelegateCmd represents the node delegate command -func NewNodeDelegateCmd(c ioctl.Client) *cobra.Command { - var endpoint string - var insecure bool +func NewNodeDelegateCmd(client ioctl.Client) *cobra.Command { + var ( + epochNum uint64 + nextEpoch bool + endpoint string + insecure bool + ) - use, _ := c.SelectTranslation(_delegateUses) - short, _ := c.SelectTranslation(_delegateShorts) - flagEpochNumUsage, _ := c.SelectTranslation(_flagEpochNumUsages) - flagNextEpochUsage, _ := c.SelectTranslation(_flagNextEpochUsages) + use, _ := client.SelectTranslation(_delegateUses) + short, _ := client.SelectTranslation(_delegateShorts) + flagEpochNumUsage, _ := client.SelectTranslation(_flagEpochNumUsages) + flagNextEpochUsage, _ := client.SelectTranslation(_flagNextEpochUsages) cmd := &cobra.Command{ Use: use, @@ -88,48 +97,52 @@ func NewNodeDelegateCmd(c ioctl.Client) *cobra.Command { RunE: func(cmd *cobra.Command, args []string) error { cmd.SilenceUsage = true var err error - if _nextEpoch { + + if nextEpoch { //nextDelegates //deprecated: It won't be able to query next delegate after Easter height, because it will be determined at the end of the epoch. - apiServiceClient, err := c.APIServiceClient(ioctl.APIServiceConfig{ + apiServiceClient, err := client.APIServiceClient(ioctl.APIServiceConfig{ Endpoint: endpoint, Insecure: insecure, }) if err != nil { return err } - chainMeta, err := bc.GetChainMeta(c) + chainMeta, err := bc.GetChainMeta(client) if err != nil { - return output.NewError(0, "failed to get chain meta", err) + return errors.Wrap(err, "failed to get chain meta") } - _epochNum = chainMeta.Epoch.Num + 1 - message := nextDelegatesMessage{Epoch: int(_epochNum)} + epochNum = chainMeta.GetEpoch().GetNum() + 1 + message := nextDelegatesMessage{Epoch: int(epochNum)} ctx := context.Background() + jwtMD, err := util.JwtAuth() + if err == nil { + ctx = metautils.NiceMD(jwtMD).ToOutgoing(ctx) + } abpResponse, err := apiServiceClient.ReadState( ctx, &iotexapi.ReadStateRequest{ ProtocolID: []byte("poll"), MethodName: []byte("ActiveBlockProducersByEpoch"), - Arguments: [][]byte{[]byte(strconv.FormatUint(_epochNum, 10))}, + Arguments: [][]byte{[]byte(strconv.FormatUint(epochNum, 10))}, }, ) - if err != nil { sta, ok := status.FromError(err) if ok && sta.Code() == codes.NotFound { message.Determined = false - fmt.Println(message.String()) + cmd.Println(message.String(epochNum)) return nil } else if ok { - return output.NewError(output.APIError, sta.Message(), nil) + return errors.New(sta.Message()) } - return output.NewError(output.NetworkError, "failed to invoke ReadState api", err) + return errors.Wrap(err, "failed to invoke ReadState api") } message.Determined = true - var ABPs state.CandidateList - if err := ABPs.Deserialize(abpResponse.Data); err != nil { - return output.NewError(output.SerializationError, "failed to deserialize active BPs", err) + var abps state.CandidateList + if err := abps.Deserialize(abpResponse.Data); err != nil { + return errors.Wrap(err, "failed to deserialize active bps") } bpResponse, err := apiServiceClient.ReadState( @@ -137,27 +150,27 @@ func NewNodeDelegateCmd(c ioctl.Client) *cobra.Command { &iotexapi.ReadStateRequest{ ProtocolID: []byte("poll"), MethodName: []byte("BlockProducersByEpoch"), - Arguments: [][]byte{[]byte(strconv.FormatUint(_epochNum, 10))}, + Arguments: [][]byte{[]byte(strconv.FormatUint(epochNum, 10))}, }, ) if err != nil { sta, ok := status.FromError(err) if ok { - return output.NewError(output.APIError, sta.Message(), nil) + return errors.New(sta.Message()) } - return output.NewError(output.NetworkError, "failed to invoke ReadState api", err) + return errors.Wrap(err, "failed to invoke ReadState api") } - var BPs state.CandidateList - if err := BPs.Deserialize(bpResponse.Data); err != nil { - return output.NewError(output.SerializationError, "failed to deserialize BPs", err) + var bps state.CandidateList + if err := bps.Deserialize(bpResponse.Data); err != nil { + return errors.Wrap(err, "failed to deserialize bps") } isActive := make(map[string]bool) - for _, abp := range ABPs { + for _, abp := range abps { isActive[abp.Address] = true } - aliases := c.AliasMap() - for rank, bp := range BPs { + aliases := client.AliasMap() + for rank, bp := range bps { votes := big.NewInt(0).SetBytes(bp.Votes.Bytes()) message.Delegates = append(message.Delegates, delegate{ Address: bp.Address, @@ -167,42 +180,41 @@ func NewNodeDelegateCmd(c ioctl.Client) *cobra.Command { Votes: util.RauToString(votes, util.IotxDecimalNum), }) } - fmt.Println(message.String()) + cmd.Println(message.String(epochNum)) } else { - if _epochNum == 0 { - chainMeta, err := bc.GetChainMeta(c) + if epochNum == 0 { + chainMeta, err := bc.GetChainMeta(client) if err != nil { - return output.NewError(0, "failed to get chain meta", err) + return errors.Wrap(err, "failed to get chain meta") } - _epochNum = chainMeta.Epoch.Num + epochNum = chainMeta.GetEpoch().GetNum() } - response, err := bc.GetEpochMeta(_epochNum, c) - + response, err := bc.GetEpochMeta(client, epochNum) if err != nil { - return output.NewError(0, "failed to get epoch meta", err) + return errors.Wrap(err, "failed to get epoch meta") } epochData := response.EpochData - aliases := c.AliasMap() + aliases := client.AliasMap() message := delegatesMessage{ Epoch: int(epochData.Num), StartBlock: int(epochData.Height), TotalBlocks: int(response.TotalBlocks), } - probationListRes, err := bc.GetProbationList(_epochNum, c) + probationListRes, err := bc.GetProbationList(client, epochNum) if err != nil { - return output.NewError(0, "failed to get probation list", err) + return errors.Wrap(err, "failed to get probation list") } probationList := &vote.ProbationList{} if probationListRes != nil { if err := probationList.Deserialize(probationListRes.Data); err != nil { - return output.NewError(output.SerializationError, "failed to deserialize probation list", err) + return errors.Wrap(err, "failed to deserialize probation list") } } for rank, bp := range response.BlockProducersInfo { votes, ok := new(big.Int).SetString(bp.Votes, 10) if !ok { - return output.NewError(output.ConvertError, "failed to convert votes into big int", nil) + return errors.New("failed to convert votes into big int") } isProbated := false if _, ok := probationList.ProbationInfo[bp.Address]; ok { @@ -220,63 +232,60 @@ func NewNodeDelegateCmd(c ioctl.Client) *cobra.Command { } message.Delegates = append(message.Delegates, delegate) } - fmt.Println(message.String()) + cmd.Println(message.String()) } - - return output.PrintError(err) + if err != nil { + cmd.Println(err) + } + return nil }, } - cmd.Flags().Uint64VarP(&_epochNum, "epoch-num", "e", 0, + cmd.Flags().Uint64VarP(&epochNum, "epoch-num", "e", 0, flagEpochNumUsage) - cmd.Flags().BoolVarP(&_nextEpoch, "next-epoch", "n", false, + cmd.Flags().BoolVarP(&nextEpoch, "next-epoch", "n", false, flagNextEpochUsage) _nodeStatus = map[bool]string{true: "active", false: ""} _probatedStatus = map[bool]string{true: "probated", false: ""} return cmd } -func (m *nextDelegatesMessage) String() string { - if output.Format == "" { - if !m.Determined { - return fmt.Sprintf("delegates of upcoming epoch #%d are not determined", _epochNum) - } - aliasLen := 5 - for _, bp := range m.Delegates { - if len(bp.Alias) > aliasLen { - aliasLen = len(bp.Alias) - } - } - lines := []string{fmt.Sprintf("Epoch: %d\n", _epochNum)} - formatTitleString := "%-41s %-4s %-" + strconv.Itoa(aliasLen) + "s %-6s %s" - formatDataString := "%-41s %4d %-" + strconv.Itoa(aliasLen) + "s %-6s %s" - lines = append(lines, fmt.Sprintf(formatTitleString, "Address", "Rank", "Alias", "Status", "Votes")) - for _, bp := range m.Delegates { - lines = append(lines, fmt.Sprintf(formatDataString, bp.Address, bp.Rank, - bp.Alias, _nodeStatus[bp.Active], bp.Votes)) +func (m *nextDelegatesMessage) String(epochNum uint64) string { + if !m.Determined { + return fmt.Sprintf("delegates of upcoming epoch #%d are not determined", epochNum) + } + aliasLen := 5 + for _, bp := range m.Delegates { + if len(bp.Alias) > aliasLen { + aliasLen = len(bp.Alias) } - return strings.Join(lines, "\n") } - return output.FormatString(output.Result, m) + lines := []string{fmt.Sprintf("Epoch: %d\n", epochNum)} + formatTitleString := "%-41s %-4s %-" + strconv.Itoa(aliasLen) + "s %-6s %s" + formatDataString := "%-41s %4d %-" + strconv.Itoa(aliasLen) + "s %-6s %s" + lines = append(lines, fmt.Sprintf(formatTitleString, "Address", "Rank", "Alias", "Status", "Votes")) + for _, bp := range m.Delegates { + lines = append(lines, fmt.Sprintf(formatDataString, bp.Address, bp.Rank, + bp.Alias, _nodeStatus[bp.Active], bp.Votes)) + } + return strings.Join(lines, "\n") } + func (m *delegatesMessage) String() string { - if output.Format == "" { - aliasLen := 5 - for _, bp := range m.Delegates { - if len(bp.Alias) > aliasLen { - aliasLen = len(bp.Alias) - } + aliasLen := 5 + for _, bp := range m.Delegates { + if len(bp.Alias) > aliasLen { + aliasLen = len(bp.Alias) } - lines := []string{fmt.Sprintf("Epoch: %d, Start block height: %d,Total blocks in epoch: %d\n", - m.Epoch, m.StartBlock, m.TotalBlocks)} - formatTitleString := "%-41s %-4s %-" + strconv.Itoa(aliasLen) + "s %-6s %-6s %-12s %s" - formatDataString := "%-41s %4d %-" + strconv.Itoa(aliasLen) + "s %-6s %-6d %-12s %s" - lines = append(lines, fmt.Sprintf(formatTitleString, - "Address", "Rank", "Alias", "Status", "Blocks", "ProbatedStatus", "Votes")) - for _, bp := range m.Delegates { - lines = append(lines, fmt.Sprintf(formatDataString, bp.Address, bp.Rank, - bp.Alias, _nodeStatus[bp.Active], bp.Production, _probatedStatus[bp.ProbatedStatus], bp.Votes)) - } - return strings.Join(lines, "\n") } - return output.FormatString(output.Result, m) + lines := []string{fmt.Sprintf("Epoch: %d, Start block height: %d,Total blocks in epoch: %d\n", + m.Epoch, m.StartBlock, m.TotalBlocks)} + formatTitleString := "%-41s %-4s %-" + strconv.Itoa(aliasLen) + "s %-6s %-6s %-12s %s" + formatDataString := "%-41s %4d %-" + strconv.Itoa(aliasLen) + "s %-6s %-6d %-12s %s" + lines = append(lines, fmt.Sprintf(formatTitleString, + "Address", "Rank", "Alias", "Status", "Blocks", "ProbatedStatus", "Votes")) + for _, bp := range m.Delegates { + lines = append(lines, fmt.Sprintf(formatDataString, bp.Address, bp.Rank, + bp.Alias, _nodeStatus[bp.Active], bp.Production, _probatedStatus[bp.ProbatedStatus], bp.Votes)) + } + return strings.Join(lines, "\n") } diff --git a/ioctl/newcmd/node/nodedelegate_test.go b/ioctl/newcmd/node/nodedelegate_test.go index c4592543e3..29717b2827 100644 --- a/ioctl/newcmd/node/nodedelegate_test.go +++ b/ioctl/newcmd/node/nodedelegate_test.go @@ -15,7 +15,7 @@ import ( ) func TestNewNodeDelegateCmd(t *testing.T) { - + require := require.New(t) ctrl := gomock.NewController(t) client := mock_ioctlclient.NewMockClient(ctrl) @@ -80,9 +80,19 @@ func TestNewNodeDelegateCmd(t *testing.T) { probationList := &iotexapi.ReadStateResponse{} apiServiceClient.EXPECT().ReadState(gomock.Any(), gomock.Any()).Return(probationList, nil).AnyTimes() - cmd := NewNodeDelegateCmd(client) + t.Run("next epoch", func(t *testing.T) { + cmd := NewNodeDelegateCmd(client) + result, err := util.ExecuteCmd(cmd, "-n", "-e", "1") + require.NoError(err) + require.Contains(result, "Epoch: 7001") + }) + + t.Run("get zero delegate epoch", func(t *testing.T) { + cmd := NewNodeDelegateCmd(client) + result, err := util.ExecuteCmd(cmd) + require.NoError(err) + require.Contains(result, "io1kr8c6krd7dhxaaqwdkr6erqgu4z0scug3drgja") + require.Contains(result, "109510794.521770016955545668") - result, err := util.ExecuteCmd(cmd) - require.NotNil(t, result) - require.NoError(t, err) + }) }