From ed9c857f490a2292348f0211a50ffaff45f39c44 Mon Sep 17 00:00:00 2001 From: Chen Chen <34592639+envestcc@users.noreply.github.com> Date: Tue, 30 Jan 2024 21:24:00 +0800 Subject: [PATCH] [staking] Handling CandidateEndorsement Action (#4020) --- action/candidate_endorsement.go | 4 +- .../staking/handler_candidate_endorsement.go | 86 ++++ .../handler_candidate_endorsement_test.go | 429 ++++++++++++++++++ action/protocol/staking/protocol.go | 24 +- blockchain/genesis/genesis.go | 18 +- pkg/util/byteutil/byteutil.go | 8 + 6 files changed, 549 insertions(+), 20 deletions(-) create mode 100644 action/protocol/staking/handler_candidate_endorsement.go create mode 100644 action/protocol/staking/handler_candidate_endorsement_test.go diff --git a/action/candidate_endorsement.go b/action/candidate_endorsement.go index 48ea3fc1cd..cb4331175e 100644 --- a/action/candidate_endorsement.go +++ b/action/candidate_endorsement.go @@ -29,8 +29,8 @@ func (act *CandidateEndorsement) BucketIndex() uint64 { return act.bucketIndex } -// Endorse returns true if the action is to endorse a candidate -func (act *CandidateEndorsement) Endorse() bool { +// IsEndorse returns true if the action is to endorse a candidate +func (act *CandidateEndorsement) IsEndorse() bool { return act.endorse } diff --git a/action/protocol/staking/handler_candidate_endorsement.go b/action/protocol/staking/handler_candidate_endorsement.go new file mode 100644 index 0000000000..5ce083c182 --- /dev/null +++ b/action/protocol/staking/handler_candidate_endorsement.go @@ -0,0 +1,86 @@ +package staking + +import ( + "context" + + "github.com/iotexproject/iotex-address/address" + "github.com/pkg/errors" + + "github.com/iotexproject/iotex-core/action" + "github.com/iotexproject/iotex-core/action/protocol" + "github.com/iotexproject/iotex-core/pkg/util/byteutil" +) + +const ( + handleCandidateEndorsement = "candidateEndorsement" +) + +func (p *Protocol) handleCandidateEndorsement(ctx context.Context, act *action.CandidateEndorsement, csm CandidateStateManager) (*receiptLog, []*action.TransactionLog, error) { + actCtx := protocol.MustGetActionCtx(ctx) + featureCtx := protocol.MustGetFeatureCtx(ctx) + log := newReceiptLog(p.addr.String(), handleCandidateEndorsement, featureCtx.NewStakingReceiptFormat) + + bucket, rErr := p.fetchBucket(csm, act.BucketIndex()) + if rErr != nil { + return log, nil, rErr + } + cand := csm.GetByOwner(bucket.Candidate) + if cand == nil { + return log, nil, errCandNotExist + } + log.AddTopics(byteutil.Uint64ToBytesBigEndian(bucket.Index), bucket.Candidate.Bytes(), []byte{byteutil.BoolToByte(act.IsEndorse())}) + + esm := NewEndorsementStateManager(csm.SM()) + expireHeight := uint64(0) + if act.IsEndorse() { + // handle endorsement + if err := p.validateEndorsement(ctx, csm, esm, actCtx.Caller, bucket, cand); err != nil { + return log, nil, err + } + expireHeight = uint64(endorsementNotExpireHeight) + } else { + // handle withdrawal + if err := p.validateEndorsementWithdrawal(ctx, esm, actCtx.Caller, bucket); err != nil { + return log, nil, err + } + // expire immediately if the bucket is not self-staked + // otherwise, expire after withdraw waiting period + expireHeight = protocol.MustGetBlockCtx(ctx).BlockHeight + if csm.ContainsSelfStakingBucket(bucket.Index) { + expireHeight += p.config.EndorsementWithdrawWaitingBlocks + } + } + // update endorsement state + if err := esm.Put(bucket.Index, &Endorsement{ + ExpireHeight: expireHeight, + }); err != nil { + return log, nil, errors.Wrapf(err, "failed to put endorsement with bucket index %d", bucket.Index) + } + return log, nil, nil +} + +func (p *Protocol) validateEndorsement(ctx context.Context, csm CandidateStateManager, esm *EndorsementStateManager, caller address.Address, bucket *VoteBucket, cand *Candidate) ReceiptError { + if err := validateBucketOwner(bucket, caller); err != nil { + return err + } + if err := validateBucketMinAmount(bucket, p.config.RegistrationConsts.MinSelfStake); err != nil { + return err + } + if err := validateBucketStake(bucket, true); err != nil { + return err + } + if err := validateBucketCandidate(bucket, cand.Owner); err != nil { + return err + } + if err := validateBucketSelfStake(csm, bucket, false); err != nil { + return err + } + return validateBucketEndorsement(esm, bucket, false, protocol.MustGetBlockCtx(ctx).BlockHeight) +} + +func (p *Protocol) validateEndorsementWithdrawal(ctx context.Context, esm *EndorsementStateManager, caller address.Address, bucket *VoteBucket) ReceiptError { + if err := validateBucketOwner(bucket, caller); err != nil { + return err + } + return validateBucketEndorsement(esm, bucket, true, protocol.MustGetBlockCtx(ctx).BlockHeight) +} diff --git a/action/protocol/staking/handler_candidate_endorsement_test.go b/action/protocol/staking/handler_candidate_endorsement_test.go new file mode 100644 index 0000000000..a3357ebbac --- /dev/null +++ b/action/protocol/staking/handler_candidate_endorsement_test.go @@ -0,0 +1,429 @@ +package staking + +import ( + "context" + "fmt" + "math/big" + "testing" + + "github.com/golang/mock/gomock" + "github.com/iotexproject/iotex-address/address" + "github.com/iotexproject/iotex-proto/golang/iotextypes" + "github.com/mohae/deepcopy" + "github.com/pkg/errors" + "github.com/stretchr/testify/require" + + "github.com/iotexproject/iotex-core/action" + "github.com/iotexproject/iotex-core/action/protocol" + accountutil "github.com/iotexproject/iotex-core/action/protocol/account/util" + "github.com/iotexproject/iotex-core/blockchain/genesis" + "github.com/iotexproject/iotex-core/pkg/unit" + "github.com/iotexproject/iotex-core/test/identityset" +) + +type appendAction struct { + act func() action.Action + status iotextypes.ReceiptStatus + validator func(t *testing.T) +} + +func TestProtocol_HandleCandidateEndorsement(t *testing.T) { + require := require.New(t) + ctrl := gomock.NewController(t) + initBucketCfgs := []*bucketConfig{ + {identityset.Address(1), identityset.Address(1), "1", 1, true, false, nil, 0}, + {identityset.Address(1), identityset.Address(1), "1200000000000000000000000", 30, true, false, nil, 0}, + {identityset.Address(1), identityset.Address(1), "1200000000000000000000000", 30, true, false, &timeBeforeBlockII, 0}, + {identityset.Address(2), identityset.Address(1), "1200000000000000000000000", 30, true, true, nil, 0}, + {identityset.Address(1), identityset.Address(2), "1200000000000000000000000", 30, true, false, nil, 0}, + {identityset.Address(2), identityset.Address(1), "1200000000000000000000000", 30, true, false, nil, 0}, + {identityset.Address(2), identityset.Address(2), "1200000000000000000000000", 30, true, true, nil, 0}, + {identityset.Address(1), identityset.Address(2), "1200000000000000000000000", 91, true, false, nil, endorsementNotExpireHeight}, + {identityset.Address(1), identityset.Address(2), "1200000000000000000000000", 91, true, false, nil, 1}, + {identityset.Address(2), identityset.Address(2), "1200000000000000000000000", 91, true, false, nil, 0}, + {identityset.Address(1), identityset.Address(1), "1200000000000000000000000", 30, false, false, nil, 0}, + } + initCandidateCfgs := []*candidateConfig{ + {identityset.Address(1), identityset.Address(7), identityset.Address(1), "test1"}, + {identityset.Address(2), identityset.Address(8), identityset.Address(1), "test2"}, + {identityset.Address(3), identityset.Address(9), identityset.Address(11), "test3"}, + } + initTestStateFromIds := func(bucketCfgIdx, candCfgIds []uint64) (protocol.StateManager, *Protocol, []*VoteBucket, []*Candidate) { + bucketCfgs := []*bucketConfig{} + for _, idx := range bucketCfgIdx { + bucketCfgs = append(bucketCfgs, initBucketCfgs[idx]) + } + candCfgs := []*candidateConfig{} + for _, idx := range candCfgIds { + candCfgs = append(candCfgs, initCandidateCfgs[idx]) + } + return initTestState(t, ctrl, bucketCfgs, candCfgs) + } + sm, p, _, _ := initTestState(t, ctrl, initBucketCfgs, initCandidateCfgs) + + tests := []struct { + name string + // params + initBucketCfgIds []uint64 + initCandidateCfgIds []uint64 + initBalance int64 + caller address.Address + nonce uint64 + gasLimit uint64 + blkGasLimit uint64 + gasPrice *big.Int + bucketID uint64 + endorse bool + newProtocol bool + append *appendAction + // expect + err error + status iotextypes.ReceiptStatus + expectCandidates []expectCandidate + expectBuckets []expectBucket + }{ + { + "endorse candidate with invalid bucket index", + []uint64{0, 1}, + []uint64{0, 1}, + 1300000, + identityset.Address(1), + 1, + uint64(1000000), + uint64(1000000), + big.NewInt(1000), + 2, + true, + true, + nil, + nil, + iotextypes.ReceiptStatus_ErrInvalidBucketIndex, + []expectCandidate{}, + nil, + }, + { + "endorse candidate with invalid bucket owner", + []uint64{0, 1}, + []uint64{0, 1}, + 1300000, + identityset.Address(2), + 1, + uint64(1000000), + uint64(1000000), + big.NewInt(1000), + 1, + true, + true, + nil, + nil, + iotextypes.ReceiptStatus_ErrUnauthorizedOperator, + []expectCandidate{}, + nil, + }, + { + "endorse candidate with invalid bucket amount", + []uint64{0, 1}, + []uint64{0, 1}, + 1000, + identityset.Address(1), + 1, + uint64(1000000), + uint64(1000000), + big.NewInt(1000), + 0, + true, + true, + nil, + nil, + iotextypes.ReceiptStatus_ErrInvalidBucketAmount, + []expectCandidate{}, + nil, + }, + { + "endorse candidate with self-staked bucket", + []uint64{0, 3}, + []uint64{0, 1}, + 1300000, + identityset.Address(1), + 1, + uint64(1000000), + uint64(1000000), + big.NewInt(1000), + 1, + true, + true, + nil, + nil, + iotextypes.ReceiptStatus_ErrInvalidBucketType, + []expectCandidate{}, + nil, + }, + { + "endorse candidate with invalid bucket candidate", + []uint64{0, 4}, + []uint64{0, 1}, + 1300000, + identityset.Address(1), + 1, + uint64(1000000), + uint64(1000000), + big.NewInt(1000), + 1, + true, + true, + nil, + nil, + iotextypes.ReceiptStatus_ErrUnauthorizedOperator, + []expectCandidate{}, + nil, + }, + { + "endorse candidate with endorsed bucket", + []uint64{0, 7}, + []uint64{0, 1}, + 1300000, + identityset.Address(2), + 1, + uint64(1000000), + uint64(1000000), + big.NewInt(1000), + 1, + true, + true, + nil, + nil, + iotextypes.ReceiptStatus_ErrInvalidBucketType, + []expectCandidate{}, + nil, + }, + { + "endorse candidate with unstaked bucket", + []uint64{0, 2}, + []uint64{0, 1}, + 1300000, + identityset.Address(1), + 1, + uint64(1000000), + uint64(1000000), + big.NewInt(1000), + 1, + true, + true, + nil, + nil, + iotextypes.ReceiptStatus_ErrInvalidBucketType, + []expectCandidate{}, + nil, + }, + { + "endorse candidate with expired endorsement", + []uint64{0, 8}, + []uint64{0, 1}, + 1300000, + identityset.Address(2), + 1, + uint64(1000000), + uint64(1000000), + big.NewInt(1000), + 1, + true, + true, + nil, + nil, + iotextypes.ReceiptStatus_Success, + []expectCandidate{ + {identityset.Address(1), candidateNoSelfStakeBucketIndex, "0", "1542516163985454635820817"}, + }, + []expectBucket{ + {0, identityset.Address(1)}, + {1, identityset.Address(1)}, + }, + }, + { + "endorse candidate with valid bucket", + []uint64{0, 1}, + []uint64{0, 1}, + 1300000, + identityset.Address(1), + 1, + uint64(1000000), + uint64(1000000), + big.NewInt(1000), + 1, + true, + true, + nil, + nil, + iotextypes.ReceiptStatus_Success, + []expectCandidate{}, + nil, + }, + { + "unendorse a valid bucket", + []uint64{0, 9}, + []uint64{0, 1}, + 1300000, + identityset.Address(2), + 1, + uint64(1000000), + uint64(1000000), + big.NewInt(1000), + 1, + true, + true, + &appendAction{ + func() action.Action { + act := action.NewCandidateEndorsement(0, uint64(1000000), big.NewInt(1000), 1, false) + return act + }, + iotextypes.ReceiptStatus_Success, + func(t *testing.T) { + csm, err := NewCandidateStateManager(sm, false) + require.NoError(err) + esm := NewEndorsementStateManager(csm.SM()) + bucket, err := csm.getBucket(1) + require.NoError(err) + endorsement, err := esm.Get(bucket.Index) + require.NoError(err) + require.NotNil(endorsement) + require.Equal(uint64(1), endorsement.ExpireHeight) + }, + }, + nil, + iotextypes.ReceiptStatus_Success, + []expectCandidate{}, + nil, + }, + { + "unendorse a self-staked bucket", + []uint64{0, 10}, + []uint64{0, 1}, + 1300000, + identityset.Address(1), + 1, + uint64(1000000), + uint64(1000000), + big.NewInt(1000), + 1, + true, + true, + &appendAction{ + func() action.Action { + act := action.NewCandidateEndorsement(0, uint64(1000000), big.NewInt(1000), 1, false) + return act + }, + iotextypes.ReceiptStatus_Success, + func(t *testing.T) { + csm, err := NewCandidateStateManager(sm, false) + require.NoError(err) + esm := NewEndorsementStateManager(csm.SM()) + bucket, err := csm.getBucket(1) + require.NoError(err) + endorsement, err := esm.Get(bucket.Index) + require.NoError(err) + require.NotNil(endorsement) + require.Equal(uint64(1), endorsement.ExpireHeight) + }, + }, + nil, + iotextypes.ReceiptStatus_Success, + []expectCandidate{}, + nil, + }, + } + + for _, test := range tests { + t.Run(test.name, func(t *testing.T) { + nonce := test.nonce + if test.newProtocol { + sm, p, _, _ = initTestStateFromIds(test.initBucketCfgIds, test.initCandidateCfgIds) + } + require.NoError(setupAccount(sm, test.caller, test.initBalance)) + act := action.NewCandidateEndorsement(nonce, test.gasLimit, test.gasPrice, test.bucketID, test.endorse) + IntrinsicGas, _ := act.IntrinsicGas() + ctx := protocol.WithActionCtx(context.Background(), protocol.ActionCtx{ + Caller: test.caller, + GasPrice: test.gasPrice, + IntrinsicGas: IntrinsicGas, + Nonce: nonce, + }) + ctx = protocol.WithBlockCtx(ctx, protocol.BlockCtx{ + BlockHeight: 1, + BlockTimeStamp: timeBlock, + GasLimit: test.blkGasLimit, + }) + cfg := deepcopy.Copy(genesis.Default).(genesis.Genesis) + cfg.ToBeEnabledBlockHeight = 1 + ctx = genesis.WithGenesisContext(ctx, genesis.Default) + ctx = protocol.WithFeatureCtx(protocol.WithFeatureWithHeightCtx(ctx)) + require.Equal(test.err, errors.Cause(p.Validate(ctx, act, sm))) + if test.err != nil { + return + } + r, err := p.Handle(ctx, act, sm) + require.NoError(err) + if r != nil { + require.Equal(uint64(test.status), r.Status) + } else { + require.Equal(test.status, iotextypes.ReceiptStatus_Failure) + } + var appendIntrinsicGas uint64 + if test.append != nil { + nonce = nonce + 1 + appendIntrinsicGas, _ = act.IntrinsicGas() + ctx := protocol.WithActionCtx(context.Background(), protocol.ActionCtx{ + Caller: test.caller, + GasPrice: test.gasPrice, + IntrinsicGas: IntrinsicGas, + Nonce: nonce, + }) + ctx = protocol.WithBlockCtx(ctx, protocol.BlockCtx{ + BlockHeight: 1, + BlockTimeStamp: timeBlock, + GasLimit: test.blkGasLimit, + }) + ctx = genesis.WithGenesisContext(ctx, cfg) + ctx = protocol.WithFeatureCtx(protocol.WithFeatureWithHeightCtx(ctx)) + r, err = p.Handle(ctx, test.append.act(), sm) + require.NoError(err) + if r != nil { + require.Equal(uint64(test.append.status), r.Status, fmt.Sprintf("except :%d, actual:%d", test.append.status, r.Status)) + if test.append.validator != nil { + test.append.validator(t) + } + } else { + require.Equal(test.status, iotextypes.ReceiptStatus_Failure) + } + } + + if test.err == nil && test.status == iotextypes.ReceiptStatus_Success { + // check candidate + csm, err := NewCandidateStateManager(sm, false) + require.NoError(err) + for _, expectCand := range test.expectCandidates { + candidate := csm.GetByOwner(expectCand.owner) + require.NotNil(candidate) + require.Equal(expectCand.candSelfStakeIndex, candidate.SelfStakeBucketIdx) + require.Equal(expectCand.candSelfStakeAmountStr, candidate.SelfStake.String()) + require.Equal(expectCand.candVoteStr, candidate.Votes.String()) + } + // check buckets + for _, expectBkt := range test.expectBuckets { + bkt, err := csm.getBucket(expectBkt.id) + require.NoError(err) + require.Equal(expectBkt.candidate, bkt.Candidate) + } + + // test staker's account + caller, err := accountutil.LoadAccount(sm, test.caller) + require.NoError(err) + actCost, err := act.Cost() + actCost.Add(actCost, big.NewInt(0).Mul(test.gasPrice, big.NewInt(0).SetUint64(appendIntrinsicGas))) + require.NoError(err) + total := big.NewInt(0) + require.Equal(unit.ConvertIotxToRau(test.initBalance), total.Add(total, caller.Balance).Add(total, actCost)) + require.Equal(nonce+1, caller.PendingNonce()) + } + }) + } +} diff --git a/action/protocol/staking/protocol.go b/action/protocol/staking/protocol.go index acb3d1fa80..70bdfaf68d 100644 --- a/action/protocol/staking/protocol.go +++ b/action/protocol/staking/protocol.go @@ -86,12 +86,13 @@ type ( // Configuration is the staking protocol configuration. Configuration struct { - VoteWeightCalConsts genesis.VoteWeightCalConsts - RegistrationConsts RegistrationConsts - WithdrawWaitingPeriod time.Duration - MinStakeAmount *big.Int - BootstrapCandidates []genesis.BootstrapCandidate - PersistStakingPatchBlock uint64 + VoteWeightCalConsts genesis.VoteWeightCalConsts + RegistrationConsts RegistrationConsts + WithdrawWaitingPeriod time.Duration + MinStakeAmount *big.Int + BootstrapCandidates []genesis.BootstrapCandidate + PersistStakingPatchBlock uint64 + EndorsementWithdrawWaitingBlocks uint64 } // DepositGas deposits gas to some pool @@ -155,10 +156,11 @@ func NewProtocol( Fee: regFee, MinSelfStake: minSelfStake, }, - WithdrawWaitingPeriod: cfg.Staking.WithdrawWaitingPeriod, - MinStakeAmount: minStakeAmount, - BootstrapCandidates: cfg.Staking.BootstrapCandidates, - PersistStakingPatchBlock: cfg.PersistStakingPatchBlock, + WithdrawWaitingPeriod: cfg.Staking.WithdrawWaitingPeriod, + MinStakeAmount: minStakeAmount, + BootstrapCandidates: cfg.Staking.BootstrapCandidates, + PersistStakingPatchBlock: cfg.PersistStakingPatchBlock, + EndorsementWithdrawWaitingBlocks: cfg.Staking.EndorsementWithdrawWaitingBlocks, }, depositGas: depositGas, candBucketsIndexer: candBucketsIndexer, @@ -420,6 +422,8 @@ func (p *Protocol) handle(ctx context.Context, act action.Action, csm CandidateS rLog, err = p.handleCandidateUpdate(ctx, act, csm) case *action.CandidateActivate: rLog, tLogs, err = p.handleCandidateActivate(ctx, act, csm) + case *action.CandidateEndorsement: + rLog, tLogs, err = p.handleCandidateEndorsement(ctx, act, csm) default: return nil, nil } diff --git a/blockchain/genesis/genesis.go b/blockchain/genesis/genesis.go index 21bcbcd574..52655243e1 100644 --- a/blockchain/genesis/genesis.go +++ b/blockchain/genesis/genesis.go @@ -114,9 +114,10 @@ func defaultConfig() Genesis { Fee: unit.ConvertIotxToRau(100).String(), MinSelfStake: unit.ConvertIotxToRau(1200000).String(), }, - WithdrawWaitingPeriod: 3 * 24 * time.Hour, - MinStakeAmount: unit.ConvertIotxToRau(100).String(), - BootstrapCandidates: []BootstrapCandidate{}, + WithdrawWaitingPeriod: 3 * 24 * time.Hour, + MinStakeAmount: unit.ConvertIotxToRau(100).String(), + BootstrapCandidates: []BootstrapCandidate{}, + EndorsementWithdrawWaitingBlocks: 24 * 60 * 60 / 5, }, } } @@ -344,11 +345,12 @@ type ( } // Staking contains the configs for staking protocol Staking struct { - VoteWeightCalConsts VoteWeightCalConsts `yaml:"voteWeightCalConsts"` - RegistrationConsts RegistrationConsts `yaml:"registrationConsts"` - WithdrawWaitingPeriod time.Duration `yaml:"withdrawWaitingPeriod"` - MinStakeAmount string `yaml:"minStakeAmount"` - BootstrapCandidates []BootstrapCandidate `yaml:"bootstrapCandidates"` + VoteWeightCalConsts VoteWeightCalConsts `yaml:"voteWeightCalConsts"` + RegistrationConsts RegistrationConsts `yaml:"registrationConsts"` + WithdrawWaitingPeriod time.Duration `yaml:"withdrawWaitingPeriod"` + MinStakeAmount string `yaml:"minStakeAmount"` + BootstrapCandidates []BootstrapCandidate `yaml:"bootstrapCandidates"` + EndorsementWithdrawWaitingBlocks uint64 `yaml:"endorsementWithdrawWaitingBlocks"` } // VoteWeightCalConsts contains the configs for calculating vote weight diff --git a/pkg/util/byteutil/byteutil.go b/pkg/util/byteutil/byteutil.go index 5a60dc4c6a..2f867dce80 100644 --- a/pkg/util/byteutil/byteutil.go +++ b/pkg/util/byteutil/byteutil.go @@ -59,3 +59,11 @@ func Uint64ToBytesBigEndian(value uint64) []byte { func BytesToUint64BigEndian(value []byte) uint64 { return binary.BigEndian.Uint64(value) } + +// BoolToByte converts bool to byte +func BoolToByte(value bool) byte { + if value { + return 1 + } + return 0 +}