From 940423716aece9e991e63cd95724cff49b49eb97 Mon Sep 17 00:00:00 2001 From: "ev.lekht" Date: Mon, 4 Nov 2024 19:21:59 +0400 Subject: [PATCH 1/3] [PVM] wip --- .../txs/executor/camino_tx_executor.go | 69 +-- .../txs/executor/camino_tx_executor_test.go | 586 +++++++++++------- 2 files changed, 405 insertions(+), 250 deletions(-) diff --git a/vms/platformvm/txs/executor/camino_tx_executor.go b/vms/platformvm/txs/executor/camino_tx_executor.go index 050212808e1a..3df02599482f 100644 --- a/vms/platformvm/txs/executor/camino_tx_executor.go +++ b/vms/platformvm/txs/executor/camino_tx_executor.go @@ -7,7 +7,6 @@ import ( "bytes" "errors" "fmt" - "reflect" "github.com/ava-labs/avalanchego/chains/atomic" "github.com/ava-labs/avalanchego/database" @@ -106,6 +105,7 @@ var ( errWrongAdminProposal = errors.New("this type of proposal can't be admin-proposal") errNotPermittedToCreateProposal = errors.New("don't have permission to create proposal of this type") errZeroDepositOfferLimits = errors.New("deposit offer TotalMaxAmount and TotalMaxRewardAmount are zero") + errAddrStateNotChanged = errors.New("address state wasn't changed") ErrInvalidProposal = errors.New("proposal is semantically invalid") ) @@ -504,19 +504,21 @@ func (e *CaminoProposalTxExecutor) RewardValidatorTx(tx *txs.RewardValidatorTx) return errWrongCredentialsNumber } - ins, outs, err := e.FlowChecker.Unlock(e.OnCommitState, []ids.ID{tx.TxID}, locked.StateBonded) + expectedIns, expectedOuts, err := e.FlowChecker.Unlock( + e.OnCommitState, + []ids.ID{tx.TxID}, + locked.StateBonded, + ) if err != nil { return err } - expectedTx := &txs.CaminoRewardValidatorTx{ - RewardValidatorTx: *tx, - Ins: ins, - Outs: outs, + if !inputsAreEqual(caminoTx.Ins, expectedIns) { + return fmt.Errorf("%w: invalid inputs", errInvalidSystemTxBody) } - if !reflect.DeepEqual(caminoTx, expectedTx) { - return errInvalidSystemTxBody + if !outputsAreEqual(caminoTx.Outs, expectedOuts) { + return fmt.Errorf("%w: invalid outputs", errInvalidSystemTxBody) } var currentStakerToRemove *state.Staker @@ -591,33 +593,30 @@ func (e *CaminoProposalTxExecutor) RewardValidatorTx(tx *txs.RewardValidatorTx) } else { e.OnCommitState.DeleteDeferredValidator(stakerToRemove) e.OnAbortState.DeleteDeferredValidator(stakerToRemove) - // Reset deferred bit on node owner address for onCommitState - nodeOwnerAddressOnCommit, err := e.OnCommitState.GetShortIDLink( - ids.ShortID(stakerToRemove.NodeID), - state.ShortLinkKeyRegisterNode, - ) - if err != nil { - return err - } - nodeOwnerAddressStateOnCommit, err := e.OnCommitState.GetAddressStates(nodeOwnerAddressOnCommit) - if err != nil { - return err - } - e.OnCommitState.SetAddressStates(nodeOwnerAddressOnCommit, nodeOwnerAddressStateOnCommit&^as.AddressStateNodeDeferred) - // Reset deferred bit on node owner address for onAbortState - nodeOwnerAddressOnAbort, err := e.OnAbortState.GetShortIDLink( - ids.ShortID(stakerToRemove.NodeID), - state.ShortLinkKeyRegisterNode, - ) - if err != nil { - return err - } - nodeOwnerAddressStateOnAbort, err := e.OnAbortState.GetAddressStates(nodeOwnerAddressOnAbort) - if err != nil { - return err + if e.Config.IsBerlinPhaseActivated(currentChainTime) { + // Reset deferred bit on node owner address + nodeOwnerAddress, err := e.OnCommitState.GetShortIDLink( + ids.ShortID(stakerToRemove.NodeID), + state.ShortLinkKeyRegisterNode, + ) + if err != nil { + return err + } + nodeOwnerAddressState, err := e.OnCommitState.GetAddressStates(nodeOwnerAddress) + if err != nil { + return err + } + + // TODO @evlekht if after Berlin we don't have any txs that hit this if, we can move it up, before phase if block + if nodeOwnerAddressState.IsNot(as.AddressStateNodeDeferred) { + return errAddrStateNotChanged + } + + addrState := nodeOwnerAddressState &^ as.AddressStateNodeDeferred + e.OnCommitState.SetAddressStates(nodeOwnerAddress, addrState) + e.OnAbortState.SetAddressStates(nodeOwnerAddress, addrState) } - e.OnAbortState.SetAddressStates(nodeOwnerAddressOnAbort, nodeOwnerAddressStateOnAbort&^as.AddressStateNodeDeferred) } txID := e.Tx.ID() @@ -2069,8 +2068,6 @@ func (e *CaminoStandardTxExecutor) FinishProposalsTx(tx *txs.FinishProposalsTx) return err } - // TODO @evlekht change rewardValidator tx in the same manner - if !inputsAreEqual(tx.Ins, expectedIns) { return fmt.Errorf("%w: invalid inputs", errInvalidSystemTxBody) } @@ -2354,6 +2351,8 @@ func (e *CaminoStandardTxExecutor) AddressStateTx(tx *txs.AddressStateTx) error // Set the new state if changed if addrState != newAddrState { e.State.SetAddressStates(tx.Address, newAddrState) + // } else { + // return errAddrStateNotChanged } // Consume the UTXOS diff --git a/vms/platformvm/txs/executor/camino_tx_executor_test.go b/vms/platformvm/txs/executor/camino_tx_executor_test.go index 9bd77bb6b746..d05dc513a335 100644 --- a/vms/platformvm/txs/executor/camino_tx_executor_test.go +++ b/vms/platformvm/txs/executor/camino_tx_executor_test.go @@ -1220,270 +1220,426 @@ func TestCaminoAddSubnetValidatorTxNodeSig(t *testing.T) { } } -func TestCaminoRewardValidatorTx(t *testing.T) { +func TestCaminoStandardTxExecutorCaminoRewardValidatorTx(t *testing.T) { caminoGenesisConf := api.Camino{ VerifyNodeSignature: true, LockModeBondDeposit: true, } + caminoConfig := &state.CaminoConfig{ + VerifyNodeSignature: caminoGenesisConf.VerifyNodeSignature, + LockModeBondDeposit: caminoGenesisConf.LockModeBondDeposit, + } - env := newCaminoEnvironment(t, test.PhaseLast, caminoGenesisConf) + nodeOwnerAddr := ids.ShortID{1, 1, 1} - currentStakerIterator, err := env.state.GetCurrentStakerIterator() - require.NoError(t, err) - require.True(t, currentStakerIterator.Next()) - stakerToRemove := currentStakerIterator.Value() - currentStakerIterator.Release() + addValidatorTxID := ids.ID{1, 1} + wrongAddValidatorTxID := ids.ID{2, 2} + depositTxID := ids.ID{3, 3} - stakerToRemoveTxIntf, _, err := env.state.GetTx(stakerToRemove.TxID) - require.NoError(t, err) - stakerToRemoveTx := stakerToRemoveTxIntf.Unsigned.(*txs.CaminoAddValidatorTx) - ins, outs, err := env.utxosHandler.Unlock(env.state, []ids.ID{stakerToRemove.TxID}, locked.StateBonded) - require.NoError(t, err) + _, bondOwnerAddr, bondOwner := generate.KeyAndOwner(t, test.Keys[0]) + _, depositBondOwnerAddr, depositBondOwner := generate.KeyAndOwner(t, test.Keys[1]) - // UTXOs before reward - innerOut := stakerToRemoveTx.Outs[0].Out.(*locked.Out) - secpOut := innerOut.TransferableOut.(*secp256k1fx.TransferOutput) - stakeOwnersAddresses := secpOut.AddressesSet() - stakeOwners := secpOut.OutputOwners - utxosBeforeReward, err := avax.GetAllUTXOs(env.state, stakeOwnersAddresses) - require.NoError(t, err) + bondedUTXO := generate.UTXO(ids.ID{3}, test.AVAXAssetID, 7, bondOwner, ids.Empty, addValidatorTxID, true) + depositedBondedUTXO := generate.UTXO(ids.ID{4}, test.AVAXAssetID, 13, depositBondOwner, depositTxID, addValidatorTxID, true) - unlockedUTXOTxID := ids.Empty - for _, utxo := range utxosBeforeReward { - if _, ok := utxo.Out.(*locked.Out); !ok { - unlockedUTXOTxID = utxo.TxID - break - } + ins := []*avax.TransferableInput{ + generate.InFromUTXO(t, bondedUTXO, []uint32{}, false), + generate.InFromUTXO(t, depositedBondedUTXO, []uint32{}, false), + } + outs := []*avax.TransferableOutput{ + generate.Out(test.AVAXAssetID, 7, bondOwner, ids.Empty, ids.Empty), + generate.Out(test.AVAXAssetID, 13, depositBondOwner, depositTxID, ids.Empty), } - require.NotEqual(t, ids.Empty, unlockedUTXOTxID) - type testCase struct { - ins []*avax.TransferableInput - outs []*avax.TransferableOutput - preExecute func(*testing.T, *txs.Tx) - generateUTXOsAfterReward func(ids.ID) []*avax.UTXO - expectedErr error + caminoRewardValidatorTx := &txs.CaminoRewardValidatorTx{ + RewardValidatorTx: txs.RewardValidatorTx{TxID: addValidatorTxID}, + Ins: ins, + Outs: outs, } - tests := map[string]testCase{ - "Reward before end time": { - ins: ins, - outs: outs, - preExecute: func(t *testing.T, tx *txs.Tx) {}, - generateUTXOsAfterReward: func(txID ids.ID) []*avax.UTXO { - return utxosBeforeReward - }, - expectedErr: errRemoveValidatorToEarly, - }, - "Wrong validator": { - ins: ins, - outs: outs, - preExecute: func(t *testing.T, tx *txs.Tx) { - rewValTx := tx.Unsigned.(*txs.CaminoRewardValidatorTx) - rewValTx.RewardValidatorTx.TxID = ids.GenerateTestID() + emptyState := func(_ *testing.T, ctrl *gomock.Controller, _ *config.Config, _ test.Phase, _ *txs.CaminoRewardValidatorTx, _ ids.ID) *state.MockDiff { + s := state.NewMockDiff(ctrl) + return s + } + + tests := map[string]struct { + onCommitState func(*testing.T, *gomock.Controller, *config.Config, test.Phase, *txs.CaminoRewardValidatorTx, ids.ID) *state.MockDiff + onAbortState func(*testing.T, *gomock.Controller, *config.Config, test.Phase, *txs.CaminoRewardValidatorTx, ids.ID) *state.MockDiff + phase test.Phase + tx *txs.CaminoRewardValidatorTx + signers [][]*secp256k1.PrivateKey + expectedErr error + }{ + "Not zero credentials": { + onCommitState: func(_ *testing.T, ctrl *gomock.Controller, _ *config.Config, _ test.Phase, _ *txs.CaminoRewardValidatorTx, _ ids.ID) *state.MockDiff { + s := state.NewMockDiff(ctrl) + s.EXPECT().CaminoConfig().Return(caminoConfig, nil) + return s + }, + onAbortState: emptyState, + phase: test.PhaseLast, + tx: caminoRewardValidatorTx, + signers: [][]*secp256k1.PrivateKey{{}}, + expectedErr: errWrongCredentialsNumber, + }, + "Invalid inputs (one excess)": { + onCommitState: func(t *testing.T, ctrl *gomock.Controller, cfg *config.Config, phase test.Phase, tx *txs.CaminoRewardValidatorTx, _ ids.ID) *state.MockDiff { + s := state.NewMockDiff(ctrl) + s.EXPECT().CaminoConfig().Return(caminoConfig, nil) + expect.Unlock(t, s, + []ids.ID{tx.TxID}, + []ids.ShortID{bondOwnerAddr, depositBondOwnerAddr}, + []*avax.UTXO{bondedUTXO, depositedBondedUTXO}, + locked.StateBonded, + ) + return s }, - generateUTXOsAfterReward: func(txID ids.ID) []*avax.UTXO { - return utxosBeforeReward + onAbortState: emptyState, + phase: test.PhaseLast, + tx: &txs.CaminoRewardValidatorTx{ + RewardValidatorTx: txs.RewardValidatorTx{TxID: addValidatorTxID}, + Ins: append( + ins, + generate.In(test.AVAXAssetID, 3, ids.Empty, ids.Empty, []uint32{}), // excess input + ), + Outs: outs, }, - expectedErr: database.ErrNotFound, + expectedErr: errInvalidSystemTxBody, }, - "No zero credentials": { - ins: ins, - outs: outs, - preExecute: func(t *testing.T, tx *txs.Tx) { - tx.Creds = append(tx.Creds, &secp256k1fx.Credential{}) + "Invalid inputs (wrong amount)": { + onCommitState: func(t *testing.T, ctrl *gomock.Controller, cfg *config.Config, phase test.Phase, tx *txs.CaminoRewardValidatorTx, _ ids.ID) *state.MockDiff { + s := state.NewMockDiff(ctrl) + s.EXPECT().CaminoConfig().Return(caminoConfig, nil) + expect.Unlock(t, s, + []ids.ID{tx.TxID}, + []ids.ShortID{bondOwnerAddr, depositBondOwnerAddr}, + []*avax.UTXO{bondedUTXO, depositedBondedUTXO}, + locked.StateBonded, + ) + return s }, - generateUTXOsAfterReward: func(txID ids.ID) []*avax.UTXO { - return utxosBeforeReward + onAbortState: emptyState, + phase: test.PhaseLast, + tx: &txs.CaminoRewardValidatorTx{ + RewardValidatorTx: txs.RewardValidatorTx{TxID: addValidatorTxID}, + Ins: func() []*avax.TransferableInput { + wrongIns := make([]*avax.TransferableInput, len(ins)) + copy(wrongIns, ins) + wrongIns[0] = &avax.TransferableInput{ + UTXOID: ins[0].UTXOID, + Asset: ins[0].Asset, + In: &locked.In{ + IDs: ins[0].In.(*locked.In).IDs, + TransferableIn: &secp256k1fx.TransferInput{ + Amt: ins[0].In.Amount() + 1, // wrong amount + }, + }, + } + return wrongIns + }(), + Outs: outs, }, - expectedErr: errWrongCredentialsNumber, + expectedErr: errInvalidSystemTxBody, }, - "Invalid inputs (one excess)": { - ins: append(ins, &avax.TransferableInput{In: &secp256k1fx.TransferInput{}}), - outs: outs, - preExecute: func(t *testing.T, tx *txs.Tx) {}, - generateUTXOsAfterReward: func(txID ids.ID) []*avax.UTXO { - return utxosBeforeReward + "Invalid outputs (one excess)": { + onCommitState: func(t *testing.T, ctrl *gomock.Controller, cfg *config.Config, phase test.Phase, tx *txs.CaminoRewardValidatorTx, _ ids.ID) *state.MockDiff { + s := state.NewMockDiff(ctrl) + s.EXPECT().CaminoConfig().Return(caminoConfig, nil) + expect.Unlock(t, s, + []ids.ID{tx.TxID}, + []ids.ShortID{bondOwnerAddr, depositBondOwnerAddr}, + []*avax.UTXO{bondedUTXO, depositedBondedUTXO}, + locked.StateBonded, + ) + return s + }, + onAbortState: emptyState, + phase: test.PhaseLast, + tx: &txs.CaminoRewardValidatorTx{ + RewardValidatorTx: txs.RewardValidatorTx{TxID: addValidatorTxID}, + Ins: ins, + Outs: append( + outs, + generate.Out(test.AVAXAssetID, 3, bondOwner, ids.Empty, ids.Empty), // excess output + ), }, expectedErr: errInvalidSystemTxBody, }, - "Invalid inputs (wrong amount)": { - ins: func() []*avax.TransferableInput { - tempIns := make([]*avax.TransferableInput, len(ins)) - inputLockIDs := locked.IDs{} - if lockedIn, ok := ins[0].In.(*locked.In); ok { - inputLockIDs = lockedIn.IDs - } - tempIns[0] = &avax.TransferableInput{ - UTXOID: ins[0].UTXOID, - Asset: ins[0].Asset, - In: &locked.In{ - IDs: inputLockIDs, - TransferableIn: &secp256k1fx.TransferInput{ - Amt: ins[0].In.Amount() - 1, + "Invalid outputs (wrong amount)": { + onCommitState: func(t *testing.T, ctrl *gomock.Controller, cfg *config.Config, phase test.Phase, tx *txs.CaminoRewardValidatorTx, _ ids.ID) *state.MockDiff { + s := state.NewMockDiff(ctrl) + s.EXPECT().CaminoConfig().Return(caminoConfig, nil) + expect.Unlock(t, s, + []ids.ID{tx.TxID}, + []ids.ShortID{bondOwnerAddr, depositBondOwnerAddr}, + []*avax.UTXO{bondedUTXO, depositedBondedUTXO}, + locked.StateBonded, + ) + return s + }, + onAbortState: emptyState, + phase: test.PhaseLast, + tx: &txs.CaminoRewardValidatorTx{ + RewardValidatorTx: txs.RewardValidatorTx{TxID: addValidatorTxID}, + Ins: ins, + Outs: func() []*avax.TransferableOutput { + wrongOuts := make([]*avax.TransferableOutput, len(outs)) + copy(wrongOuts, outs) + wrongOuts[0] = &avax.TransferableOutput{ + Asset: outs[0].Asset, + Out: &secp256k1fx.TransferOutput{ + Amt: outs[0].Out.Amount() + 1, // wrong amount + OutputOwners: bondOwner, }, - }, - } - return tempIns - }(), - outs: outs, - preExecute: func(t *testing.T, tx *txs.Tx) {}, - generateUTXOsAfterReward: func(txID ids.ID) []*avax.UTXO { - return utxosBeforeReward + } + return wrongOuts + }(), }, expectedErr: errInvalidSystemTxBody, }, - "Invalid outs (one excess)": { - ins: ins, - outs: append(outs, &avax.TransferableOutput{Out: &secp256k1fx.TransferOutput{}}), - preExecute: func(t *testing.T, tx *txs.Tx) {}, - generateUTXOsAfterReward: func(txID ids.ID) []*avax.UTXO { - return utxosBeforeReward + "Wrong staker": { + onCommitState: func(t *testing.T, ctrl *gomock.Controller, cfg *config.Config, phase test.Phase, tx *txs.CaminoRewardValidatorTx, _ ids.ID) *state.MockDiff { + stakerToRemove := &state.Staker{TxID: addValidatorTxID} + + currentStakerIterator := state.NewMockStakerIterator(ctrl) + currentStakerIterator.EXPECT().Next().Return(true) + currentStakerIterator.EXPECT().Value().Return(stakerToRemove) + currentStakerIterator.EXPECT().Release() + + deferredStakerIterator := state.NewMockStakerIterator(ctrl) + deferredStakerIterator.EXPECT().Next().Return(false) + + s := state.NewMockDiff(ctrl) + s.EXPECT().CaminoConfig().Return(caminoConfig, nil) + expect.Unlock(t, s, + []ids.ID{tx.TxID}, + []ids.ShortID{bondOwnerAddr, depositBondOwnerAddr}, + []*avax.UTXO{bondedUTXO, depositedBondedUTXO}, + locked.StateBonded, + ) + s.EXPECT().GetCurrentStakerIterator().Return(currentStakerIterator, nil) + s.EXPECT().GetDeferredStakerIterator().Return(deferredStakerIterator, nil) + return s }, - expectedErr: errInvalidSystemTxBody, + onAbortState: emptyState, + phase: test.PhaseLast, + tx: &txs.CaminoRewardValidatorTx{ + RewardValidatorTx: txs.RewardValidatorTx{TxID: wrongAddValidatorTxID}, // not next current/deferred staker addValidatorTx + Ins: ins, + Outs: outs, + }, + expectedErr: errRemoveWrongValidator, }, - "Invalid outs (wrong amount)": { - ins: ins, - outs: func() []*avax.TransferableOutput { - tempOuts := make([]*avax.TransferableOutput, len(outs)) - copy(tempOuts, outs) - validOut := tempOuts[0].Out - if lockedOut, ok := validOut.(*locked.Out); ok { - validOut = lockedOut.TransferableOut + "Remove before end time": { + onCommitState: func(t *testing.T, ctrl *gomock.Controller, cfg *config.Config, phase test.Phase, tx *txs.CaminoRewardValidatorTx, _ ids.ID) *state.MockDiff { + chainTime := test.PhaseTime(t, phase, cfg) + stakerToRemove := &state.Staker{ + TxID: tx.TxID, + EndTime: chainTime.Add(1), // too early for removal } - secpOut, ok := validOut.(*secp256k1fx.TransferOutput) - require.True(t, ok) - var invalidOut avax.TransferableOut = &secp256k1fx.TransferOutput{ - Amt: secpOut.Amt - 1, - OutputOwners: secpOut.OutputOwners, - } - if lockedOut, ok := validOut.(*locked.Out); ok { - invalidOut = &locked.Out{ - IDs: lockedOut.IDs, - TransferableOut: invalidOut, - } + currentStakerIterator := state.NewMockStakerIterator(ctrl) + currentStakerIterator.EXPECT().Next().Return(true) + currentStakerIterator.EXPECT().Value().Return(stakerToRemove) + currentStakerIterator.EXPECT().Release() + + deferredStakerIterator := state.NewMockStakerIterator(ctrl) + deferredStakerIterator.EXPECT().Next().Return(false) + + s := state.NewMockDiff(ctrl) + s.EXPECT().CaminoConfig().Return(caminoConfig, nil) + expect.Unlock(t, s, + []ids.ID{tx.TxID}, + []ids.ShortID{bondOwnerAddr, depositBondOwnerAddr}, + []*avax.UTXO{bondedUTXO, depositedBondedUTXO}, + locked.StateBonded, + ) + s.EXPECT().GetCurrentStakerIterator().Return(currentStakerIterator, nil) + s.EXPECT().GetDeferredStakerIterator().Return(deferredStakerIterator, nil) + s.EXPECT().GetTimestamp().Return(chainTime) + return s + }, + onAbortState: emptyState, + phase: test.PhaseLast, + tx: caminoRewardValidatorTx, + expectedErr: errRemoveValidatorToEarly, + }, + "OK": { + onCommitState: func(t *testing.T, ctrl *gomock.Controller, cfg *config.Config, phase test.Phase, tx *txs.CaminoRewardValidatorTx, txID ids.ID) *state.MockDiff { + chainTime := test.PhaseTime(t, phase, cfg) + stakerToRemove := &state.Staker{ + TxID: tx.TxID, + EndTime: chainTime, } - tempOuts[0] = &avax.TransferableOutput{ - Asset: avax.Asset{ID: env.ctx.AVAXAssetID}, - Out: invalidOut, + + currentStakerIterator := state.NewMockStakerIterator(ctrl) + currentStakerIterator.EXPECT().Next().Return(true) + currentStakerIterator.EXPECT().Value().Return(stakerToRemove) + currentStakerIterator.EXPECT().Release() + + deferredStakerIterator := state.NewMockStakerIterator(ctrl) + deferredStakerIterator.EXPECT().Next().Return(false) + + s := state.NewMockDiff(ctrl) + s.EXPECT().CaminoConfig().Return(caminoConfig, nil) + expect.Unlock(t, s, + []ids.ID{tx.TxID}, + []ids.ShortID{bondOwnerAddr, depositBondOwnerAddr}, + []*avax.UTXO{bondedUTXO, depositedBondedUTXO}, + locked.StateBonded, + ) + s.EXPECT().GetCurrentStakerIterator().Return(currentStakerIterator, nil) + s.EXPECT().GetDeferredStakerIterator().Return(deferredStakerIterator, nil) + s.EXPECT().GetTimestamp().Return(chainTime) + s.EXPECT().GetTx(tx.TxID).Return(&txs.Tx{Unsigned: &txs.CaminoAddValidatorTx{}}, status.Committed, nil) + s.EXPECT().DeleteCurrentValidator(stakerToRemove) + expect.ConsumeUTXOs(t, s, tx.Ins) + expect.ProduceUTXOs(t, s, tx.Outs, txID, 0) + return s + }, + onAbortState: func(_ *testing.T, ctrl *gomock.Controller, cfg *config.Config, phase test.Phase, tx *txs.CaminoRewardValidatorTx, txID ids.ID) *state.MockDiff { + chainTime := test.PhaseTime(t, phase, cfg) + stakerToRemove := &state.Staker{ + TxID: tx.TxID, + EndTime: chainTime, } - return tempOuts - }(), - preExecute: func(t *testing.T, tx *txs.Tx) {}, - generateUTXOsAfterReward: func(txID ids.ID) []*avax.UTXO { - return utxosBeforeReward + + s := state.NewMockDiff(ctrl) + s.EXPECT().DeleteCurrentValidator(stakerToRemove) + expect.ConsumeUTXOs(t, s, tx.Ins) + expect.ProduceUTXOs(t, s, tx.Outs, txID, 0) + return s }, - expectedErr: errInvalidSystemTxBody, + phase: test.PhaseLast, + tx: caminoRewardValidatorTx, }, - } + "OK: deferred": { + onCommitState: func(t *testing.T, ctrl *gomock.Controller, cfg *config.Config, phase test.Phase, tx *txs.CaminoRewardValidatorTx, txID ids.ID) *state.MockDiff { + chainTime := test.PhaseTime(t, phase, cfg) + stakerToRemove := &state.Staker{ + TxID: tx.TxID, + EndTime: chainTime, + } - execute := func(t *testing.T, tt testCase) (CaminoProposalTxExecutor, *txs.Tx) { - tx := &txs.Tx{Unsigned: &txs.CaminoRewardValidatorTx{ - RewardValidatorTx: txs.RewardValidatorTx{TxID: stakerToRemove.TxID}, - Ins: tt.ins, - Outs: tt.outs, - }} - require.NoError(t, tx.Initialize(txs.Codec)) + currentStakerIterator := state.NewMockStakerIterator(ctrl) + currentStakerIterator.EXPECT().Next().Return(false) - tt.preExecute(t, tx) + deferredStakerIterator := state.NewMockStakerIterator(ctrl) + deferredStakerIterator.EXPECT().Next().Return(true) + deferredStakerIterator.EXPECT().Value().Return(stakerToRemove) + deferredStakerIterator.EXPECT().Release() + + s := state.NewMockDiff(ctrl) + s.EXPECT().CaminoConfig().Return(caminoConfig, nil) + expect.Unlock(t, s, + []ids.ID{tx.TxID}, + []ids.ShortID{bondOwnerAddr, depositBondOwnerAddr}, + []*avax.UTXO{bondedUTXO, depositedBondedUTXO}, + locked.StateBonded, + ) + s.EXPECT().GetCurrentStakerIterator().Return(currentStakerIterator, nil) + s.EXPECT().GetDeferredStakerIterator().Return(deferredStakerIterator, nil) + s.EXPECT().GetTimestamp().Return(chainTime) + s.EXPECT().GetTx(tx.TxID).Return(&txs.Tx{Unsigned: &txs.CaminoAddValidatorTx{}}, status.Committed, nil) + s.EXPECT().DeleteDeferredValidator(stakerToRemove) - onCommitState, err := state.NewDiff(lastAcceptedID, env) - require.NoError(t, err) + s.EXPECT().GetShortIDLink(ids.ShortID(stakerToRemove.NodeID), state.ShortLinkKeyRegisterNode).Return(nodeOwnerAddr, nil) + nodeOwnerAddrState := as.AddressStateNodeDeferred | as.AddressStateConsortium + s.EXPECT().GetAddressStates(nodeOwnerAddr).Return(nodeOwnerAddrState, nil) + s.EXPECT().SetAddressStates(nodeOwnerAddr, nodeOwnerAddrState & ^as.AddressStateNodeDeferred) - onAbortState, err := state.NewDiff(lastAcceptedID, env) - require.NoError(t, err) + expect.ConsumeUTXOs(t, s, tx.Ins) + expect.ProduceUTXOs(t, s, tx.Outs, txID, 0) + return s + }, + onAbortState: func(_ *testing.T, ctrl *gomock.Controller, cfg *config.Config, phase test.Phase, tx *txs.CaminoRewardValidatorTx, txID ids.ID) *state.MockDiff { + chainTime := test.PhaseTime(t, phase, cfg) + stakerToRemove := &state.Staker{ + TxID: tx.TxID, + EndTime: chainTime, + } - txExecutor := CaminoProposalTxExecutor{ - ProposalTxExecutor{ - OnCommitState: onCommitState, - OnAbortState: onAbortState, - Backend: &env.backend, - Tx: tx, + s := state.NewMockDiff(ctrl) + s.EXPECT().DeleteDeferredValidator(stakerToRemove) + s.EXPECT().SetAddressStates(nodeOwnerAddr, as.AddressStateConsortium) + expect.ConsumeUTXOs(t, s, tx.Ins) + expect.ProduceUTXOs(t, s, tx.Outs, txID, 0) + return s }, - } - err = tx.Unsigned.Visit(&txExecutor) - require.ErrorIs(t, err, tt.expectedErr) - return txExecutor, tx - } + phase: test.PhaseLast, + tx: caminoRewardValidatorTx, + }, + "OK: deferred before Berlin": { + onCommitState: func(t *testing.T, ctrl *gomock.Controller, cfg *config.Config, phase test.Phase, tx *txs.CaminoRewardValidatorTx, txID ids.ID) *state.MockDiff { + chainTime := test.PhaseTime(t, phase, cfg) + stakerToRemove := &state.Staker{ + TxID: tx.TxID, + EndTime: chainTime, + } - // Asserting UTXO changes - assertBalance := func(t *testing.T, tt testCase, tx *txs.Tx) { - onCommitUTXOs, err := avax.GetAllUTXOs(env.state, stakeOwnersAddresses) - require.NoError(t, err) - utxosAfterReward := tt.generateUTXOsAfterReward(tx.ID()) - require.Equal(t, onCommitUTXOs, utxosAfterReward) - } + currentStakerIterator := state.NewMockStakerIterator(ctrl) + currentStakerIterator.EXPECT().Next().Return(false) - // Asserting that staker is removed - assertNextStaker := func(t *testing.T) { - nextStakerIterator, err := env.state.GetCurrentStakerIterator() - require.NoError(t, err) - require.True(t, nextStakerIterator.Next()) - nextStakerToRemove := nextStakerIterator.Value() - nextStakerIterator.Release() - require.NotEqual(t, nextStakerToRemove.TxID, stakerToRemove.TxID) - } + deferredStakerIterator := state.NewMockStakerIterator(ctrl) + deferredStakerIterator.EXPECT().Next().Return(true) + deferredStakerIterator.EXPECT().Value().Return(stakerToRemove) + deferredStakerIterator.EXPECT().Release() + + s := state.NewMockDiff(ctrl) + s.EXPECT().CaminoConfig().Return(caminoConfig, nil) + expect.Unlock(t, s, + []ids.ID{tx.TxID}, + []ids.ShortID{bondOwnerAddr, depositBondOwnerAddr}, + []*avax.UTXO{bondedUTXO, depositedBondedUTXO}, + locked.StateBonded, + ) + s.EXPECT().GetCurrentStakerIterator().Return(currentStakerIterator, nil) + s.EXPECT().GetDeferredStakerIterator().Return(deferredStakerIterator, nil) + s.EXPECT().GetTimestamp().Return(chainTime) + s.EXPECT().GetTx(tx.TxID).Return(&txs.Tx{Unsigned: &txs.CaminoAddValidatorTx{}}, status.Committed, nil) + s.EXPECT().DeleteDeferredValidator(stakerToRemove) - for name, tt := range tests { - t.Run(name+" On abort", func(t *testing.T) { - txExecutor, tx := execute(t, tt) - require.NoError(t, txExecutor.OnAbortState.Apply(env.state)) - env.state.SetHeight(uint64(1)) - require.NoError(t, env.state.Commit()) - assertBalance(t, tt, tx) - }) - t.Run(name+" On commit", func(t *testing.T) { - txExecutor, tx := execute(t, tt) - require.NoError(t, txExecutor.OnCommitState.Apply(env.state)) - env.state.SetHeight(uint64(1)) - require.NoError(t, env.state.Commit()) - assertBalance(t, tt, tx) - }) - } + expect.ConsumeUTXOs(t, s, tx.Ins) + expect.ProduceUTXOs(t, s, tx.Outs, txID, 0) + return s + }, + onAbortState: func(_ *testing.T, ctrl *gomock.Controller, cfg *config.Config, phase test.Phase, tx *txs.CaminoRewardValidatorTx, txID ids.ID) *state.MockDiff { + chainTime := test.PhaseTime(t, phase, cfg) + stakerToRemove := &state.Staker{ + TxID: tx.TxID, + EndTime: chainTime, + } - happyPathTest := testCase{ - ins: ins, - outs: outs, - preExecute: func(t *testing.T, tx *txs.Tx) { - env.state.SetTimestamp(stakerToRemove.EndTime) - }, - generateUTXOsAfterReward: func(txID ids.ID) []*avax.UTXO { - return []*avax.UTXO{ - generate.UTXO(txID, env.ctx.AVAXAssetID, test.ValidatorWeight, stakeOwners, ids.Empty, ids.Empty, true), - generate.UTXOWithIndex(unlockedUTXOTxID, 2, env.ctx.AVAXAssetID, test.PreFundedBalance, stakeOwners, ids.Empty, ids.Empty, true), - } + s := state.NewMockDiff(ctrl) + s.EXPECT().DeleteDeferredValidator(stakerToRemove) + expect.ConsumeUTXOs(t, s, tx.Ins) + expect.ProduceUTXOs(t, s, tx.Outs, txID, 0) + return s + }, + phase: test.PhaseAthens, + tx: caminoRewardValidatorTx, }, - expectedErr: nil, } + for name, tt := range tests { + t.Run(name, func(t *testing.T) { + ctrl := gomock.NewController(t) + backend := newExecutorBackend(t, caminoGenesisConf, tt.phase, nil) - t.Run("Happy path on commit", func(t *testing.T) { - txExecutor, tx := execute(t, happyPathTest) - require.NoError(t, txExecutor.OnCommitState.Apply(env.state)) - env.state.SetHeight(uint64(1)) - require.NoError(t, env.state.Commit()) - assertBalance(t, happyPathTest, tx) - assertNextStaker(t) - }) - - // We need to start again the environment because the staker is already removed from the previous test - env = newCaminoEnvironment(t, test.PhaseLast, caminoGenesisConf) + tx, err := txs.NewSigned(tt.tx, txs.Codec, tt.signers) + require.NoError(t, err) - t.Run("Happy path on abort", func(t *testing.T) { - // utxoids are polluted with cached ids, need to clean this non-exported field - for _, in := range ins { - in.UTXOID = avax.UTXOID{ - TxID: in.TxID, - OutputIndex: in.OutputIndex, + txExecutor := CaminoProposalTxExecutor{ + ProposalTxExecutor{ + OnCommitState: tt.onCommitState(t, ctrl, backend.Config, tt.phase, tt.tx, tx.ID()), + OnAbortState: tt.onAbortState(t, ctrl, backend.Config, tt.phase, tt.tx, tx.ID()), + Backend: backend, + Tx: tx, + }, } - } - txExecutor, tx := execute(t, happyPathTest) - require.NoError(t, txExecutor.OnAbortState.Apply(env.state)) - env.state.SetHeight(uint64(1)) - require.NoError(t, env.state.Commit()) - assertBalance(t, happyPathTest, tx) - assertNextStaker(t) - }) + err = tx.Unsigned.Visit(&txExecutor) + require.ErrorIs(t, err, tt.expectedErr) + }) + } } func TestCaminoStandardTxExecutorAddressStateTx(t *testing.T) { From d1b9a3df6b8a60d76f60b8b4a3f2242ae8cc3994 Mon Sep 17 00:00:00 2001 From: "ev.lekht" Date: Mon, 4 Nov 2024 20:26:28 +0400 Subject: [PATCH 2/3] [PVM] wip --- .../txs/executor/camino_tx_executor.go | 4 +- .../txs/executor/camino_tx_executor_test.go | 55 ++++++++++++++++++- 2 files changed, 55 insertions(+), 4 deletions(-) diff --git a/vms/platformvm/txs/executor/camino_tx_executor.go b/vms/platformvm/txs/executor/camino_tx_executor.go index 3df02599482f..bbf283aaf45d 100644 --- a/vms/platformvm/txs/executor/camino_tx_executor.go +++ b/vms/platformvm/txs/executor/camino_tx_executor.go @@ -2351,8 +2351,8 @@ func (e *CaminoStandardTxExecutor) AddressStateTx(tx *txs.AddressStateTx) error // Set the new state if changed if addrState != newAddrState { e.State.SetAddressStates(tx.Address, newAddrState) - // } else { - // return errAddrStateNotChanged + } else if isBerlinPhase { + return errAddrStateNotChanged } // Consume the UTXOS diff --git a/vms/platformvm/txs/executor/camino_tx_executor_test.go b/vms/platformvm/txs/executor/camino_tx_executor_test.go index d05dc513a335..98751669807b 100644 --- a/vms/platformvm/txs/executor/camino_tx_executor_test.go +++ b/vms/platformvm/txs/executor/camino_tx_executor_test.go @@ -1673,6 +1673,7 @@ func TestCaminoStandardTxExecutorAddressStateTx(t *testing.T) { remove bool currentTargetAddrState as.AddressState executorAddrState as.AddressState + lastValidPhase test.Phase } type testCase struct { @@ -1984,6 +1985,48 @@ func TestCaminoStandardTxExecutorAddressStateTx(t *testing.T) { } } + testCaseFailNoOp := map[codec.UpgradeVersionID]testCaseSimpleFunc{} + testCaseFailNoOp[codec.UpgradeVersion0] = failCaseSimpleNoOp + testCaseFailNoOp[codec.UpgradeVersion1] = func( + t *testing.T, + phase test.Phase, + ) { + if phase < test.PhaseBerlin { + return + } + testCaseName := fmt.Sprintf("%d_%s/Upgrade 1/NoOp addr state modification is forbidden", phase, test.PhaseName(t, phase)) + require.NotContains(t, testCases, testCaseName, testCaseName) + testCases[testCaseName] = testCase{ + state: func(t *testing.T, c *gomock.Controller, utx *txs.AddressStateTx, txID ids.ID, cfg *config.Config) *state.MockDiff { + s := state.NewMockDiff(c) + s.EXPECT().GetTimestamp().Return(test.PhaseTime(t, phase, cfg)) + + expect.VerifyMultisigPermission(t, s, []ids.ShortID{utx.Executor}, nil) + s.EXPECT().GetAddressStates(utx.Executor).Return(as.AddressStateRoleKYCAdmin, nil) + + s.EXPECT().GetBaseFee().Return(defaultTxFee, nil) + expect.VerifyLock(t, s, utx.Ins, []*avax.UTXO{feeUTXO}, []ids.ShortID{feeOwnerAddr}, nil) + + s.EXPECT().GetAddressStates(utx.Address).Return(as.AddressStateConsortium, nil) + return s + }, + utx: &txs.AddressStateTx{ + UpgradeVersionID: codec.UpgradeVersion1, + BaseTx: baseTx, + Address: otherAddr, + StateBit: as.AddressStateBitKYCVerified, + Remove: true, + Executor: executorAddr, + ExecutorAuth: &secp256k1fx.Input{SigIndices: []uint32{0}}, + }, + phase: phase, + signers: [][]*secp256k1.PrivateKey{ + {feeOwnerKey}, {executorKey}, + }, + expectedErr: errAddrStateNotChanged, + } + } + testCaseOK := map[codec.UpgradeVersionID]testCaseFunc{} testCaseOK[codec.UpgradeVersion0] = func( t *testing.T, @@ -1991,6 +2034,9 @@ func TestCaminoStandardTxExecutorAddressStateTx(t *testing.T) { testCaseName string, phase test.Phase, ) { + if phase > tt.lastValidPhase { + return + } testCaseName = fmt.Sprintf("%d_%s/Upgrade 0/%s", phase, test.PhaseName(t, phase), testCaseName) require.NotContains(t, testCases, testCaseName, testCaseName) targetAddr := otherAddr @@ -2060,6 +2106,9 @@ func TestCaminoStandardTxExecutorAddressStateTx(t *testing.T) { testCaseName string, phase test.Phase, ) { + if phase > tt.lastValidPhase { + return + } testCaseName = fmt.Sprintf("%d_%s/Upgrade 1/%s", phase, test.PhaseName(t, phase), testCaseName) require.NotContains(t, testCases, testCaseName, testCaseName) targetAddr := otherAddr @@ -2192,6 +2241,7 @@ func TestCaminoStandardTxExecutorAddressStateTx(t *testing.T) { txStateBit: as.AddressStateBitRoleKYCAdmin, currentTargetAddrState: as.AddressStateRoleKYCAdmin, executorAddrState: as.AddressStateRoleAdmin, + lastValidPhase: test.PhaseAthens, }, "OK: modifying executors own address state": { selfModify: true, @@ -2210,6 +2260,7 @@ func TestCaminoStandardTxExecutorAddressStateTx(t *testing.T) { remove: true, currentTargetAddrState: as.AddressStateConsortium, executorAddrState: as.AddressStateRoleAdmin, + lastValidPhase: test.PhaseAthens, }, } @@ -2305,14 +2356,14 @@ func TestCaminoStandardTxExecutorAddressStateTx(t *testing.T) { } for phase := test.PhaseFirst; phase <= test.PhaseLast; phase++ { - txUpgrades := txUpgradeMatrix[phase] - for _, txUpgrade := range txUpgrades { + for _, txUpgrade := range txUpgradeMatrix[phase] { for name, tt := range simpleOKCases { testCaseOK[txUpgrade](t, tt, name, phase) } testCaseOKMultiInput[txUpgrade](t, phase) // noop for upgr1 testCaseFailAdminSelfRemove[txUpgrade](t, phase) testCaseFailWrongExecutorCredential[txUpgrade](t, phase) // noop for upgr0 or sunrise phase + testCaseFailNoOp[txUpgrade](t, phase) // noop for before berlin testCaseFailMultisigAlias[txUpgrade](t, phase) // noop for upgr1 } } From cefb0c5c2c46fa2f0a1ec3f943fb0378e28c22b5 Mon Sep 17 00:00:00 2001 From: "ev.lekht" Date: Tue, 5 Nov 2024 17:02:11 +0400 Subject: [PATCH 3/3] review fixes --- vms/platformvm/txs/executor/camino_tx_executor.go | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/vms/platformvm/txs/executor/camino_tx_executor.go b/vms/platformvm/txs/executor/camino_tx_executor.go index bbf283aaf45d..4aff5cd4c5a5 100644 --- a/vms/platformvm/txs/executor/camino_tx_executor.go +++ b/vms/platformvm/txs/executor/camino_tx_executor.go @@ -609,8 +609,11 @@ func (e *CaminoProposalTxExecutor) RewardValidatorTx(tx *txs.RewardValidatorTx) } // TODO @evlekht if after Berlin we don't have any txs that hit this if, we can move it up, before phase if block + // This is sanity check for consistency between address state + // and validator presence in deferred stakers if nodeOwnerAddressState.IsNot(as.AddressStateNodeDeferred) { - return errAddrStateNotChanged + // should never happen + return fmt.Errorf("deferred node owner address state doesn't have AddressStateNodeDeferred bit: %w", errAddrStateNotChanged) } addrState := nodeOwnerAddressState &^ as.AddressStateNodeDeferred