diff --git a/protocol/testutil/constants/orders.go b/protocol/testutil/constants/orders.go index 4c8f2b7b7f..bebaff3975 100644 --- a/protocol/testutil/constants/orders.go +++ b/protocol/testutil/constants/orders.go @@ -190,6 +190,13 @@ var ( Subticks: 10, GoodTilOneof: &clobtypes.Order_GoodTilBlock{GoodTilBlock: 20}, } + Order_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB30 = clobtypes.Order{ + OrderId: clobtypes.OrderId{SubaccountId: Alice_Num1, ClientId: 0, ClobPairId: 0}, + Side: clobtypes.Order_SIDE_SELL, + Quantums: 10, + Subticks: 10, + GoodTilOneof: &clobtypes.Order_GoodTilBlock{GoodTilBlock: 30}, + } Order_Alice_Num1_Id0_Clob0_Sell10_Price15_GTB20 = clobtypes.Order{ OrderId: clobtypes.OrderId{SubaccountId: Alice_Num1, ClientId: 0, ClobPairId: 0}, Side: clobtypes.Order_SIDE_SELL, @@ -197,6 +204,20 @@ var ( Subticks: 15, GoodTilOneof: &clobtypes.Order_GoodTilBlock{GoodTilBlock: 20}, } + Order_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB20 = clobtypes.Order{ + OrderId: clobtypes.OrderId{SubaccountId: Alice_Num1, ClientId: 2, ClobPairId: 1}, + Side: clobtypes.Order_SIDE_BUY, + Quantums: 10, + Subticks: 10, + GoodTilOneof: &clobtypes.Order_GoodTilBlock{GoodTilBlock: 20}, + } + Order_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB26 = clobtypes.Order{ + OrderId: clobtypes.OrderId{SubaccountId: Alice_Num1, ClientId: 2, ClobPairId: 1}, + Side: clobtypes.Order_SIDE_BUY, + Quantums: 10, + Subticks: 10, + GoodTilOneof: &clobtypes.Order_GoodTilBlock{GoodTilBlock: 26}, + } Order_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20 = clobtypes.Order{ OrderId: clobtypes.OrderId{SubaccountId: Alice_Num1, ClientId: 1, ClobPairId: 1}, Side: clobtypes.Order_SIDE_SELL, @@ -365,6 +386,20 @@ var ( Subticks: 1_000_000_000_000, GoodTilOneof: &clobtypes.Order_GoodTilBlock{GoodTilBlock: 20}, } + Order_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20 = clobtypes.Order{ + OrderId: clobtypes.OrderId{SubaccountId: Alice_Num1, ClientId: 3, ClobPairId: 0}, + Side: clobtypes.Order_SIDE_SELL, + Quantums: 100, + Subticks: 1_000_000_000_000, + GoodTilOneof: &clobtypes.Order_GoodTilBlock{GoodTilBlock: 20}, + } + Order_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23 = clobtypes.Order{ + OrderId: clobtypes.OrderId{SubaccountId: Alice_Num1, ClientId: 5, ClobPairId: 1}, + Side: clobtypes.Order_SIDE_BUY, + Quantums: 10, + Subticks: 15, + GoodTilOneof: &clobtypes.Order_GoodTilBlock{GoodTilBlock: 23}, + } Order_Bob_Num0_Id0_Clob1_Sell10_Price15_GTB20 = clobtypes.Order{ OrderId: clobtypes.OrderId{SubaccountId: Bob_Num0, ClientId: 0, ClobPairId: 1}, Side: clobtypes.Order_SIDE_SELL, @@ -1480,6 +1515,14 @@ var ( GoodTilOneof: &clobtypes.Order_GoodTilBlock{GoodTilBlock: 20}, TimeInForce: clobtypes.Order_TIME_IN_FORCE_POST_ONLY, } + Order_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO = clobtypes.Order{ + OrderId: clobtypes.OrderId{SubaccountId: Alice_Num1, ClientId: 4, ClobPairId: 1}, + Side: clobtypes.Order_SIDE_SELL, + Quantums: 10, + Subticks: 15, + GoodTilOneof: &clobtypes.Order_GoodTilBlock{GoodTilBlock: 20}, + TimeInForce: clobtypes.Order_TIME_IN_FORCE_POST_ONLY, + } Order_Bob_Num0_Id4_Clob1_Buy20_Price35_GTB22_PO = clobtypes.Order{ OrderId: clobtypes.OrderId{SubaccountId: Bob_Num0, ClientId: 4, ClobPairId: 1}, Side: clobtypes.Order_SIDE_BUY, diff --git a/protocol/x/clob/e2e/batch_cancel_test.go b/protocol/x/clob/e2e/batch_cancel_test.go index b39f487afb..199f47afbe 100644 --- a/protocol/x/clob/e2e/batch_cancel_test.go +++ b/protocol/x/clob/e2e/batch_cancel_test.go @@ -3,11 +3,19 @@ package clob_test import ( "testing" + "github.com/cometbft/cometbft/crypto/tmhash" + "github.com/cometbft/cometbft/types" "github.com/stretchr/testify/require" + "github.com/dydxprotocol/v4-chain/protocol/indexer" + "github.com/dydxprotocol/v4-chain/protocol/indexer/msgsender" + "github.com/dydxprotocol/v4-chain/protocol/indexer/off_chain_updates" + ocutypes "github.com/dydxprotocol/v4-chain/protocol/indexer/off_chain_updates/types" + indexersharedtypes "github.com/dydxprotocol/v4-chain/protocol/indexer/shared/types" testapp "github.com/dydxprotocol/v4-chain/protocol/testutil/app" "github.com/dydxprotocol/v4-chain/protocol/testutil/constants" clobtypes "github.com/dydxprotocol/v4-chain/protocol/x/clob/types" + satypes "github.com/dydxprotocol/v4-chain/protocol/x/subaccounts/types" ) // These tests are the same as the e2e tests for single order cancellations. @@ -281,3 +289,601 @@ func TestBatchCancelSingleCancelFunctionality(t *testing.T) { }) } } + +var ( + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20 = *clobtypes.NewMsgPlaceOrder( + testapp.MustScaleOrder( + constants.Order_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20, + testapp.DefaultGenesis(), + ), + ) + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB30 = *clobtypes.NewMsgPlaceOrder( + testapp.MustScaleOrder( + constants.Order_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB30, + testapp.DefaultGenesis(), + ), + ) + PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20 = *clobtypes.NewMsgPlaceOrder( + testapp.MustScaleOrder( + constants.Order_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20, + testapp.DefaultGenesis(), + ), + ) + PlaceOrder_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB20 = *clobtypes.NewMsgPlaceOrder( + testapp.MustScaleOrder( + constants.Order_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB20, + testapp.DefaultGenesis(), + ), + ) + PlaceOrder_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB26 = *clobtypes.NewMsgPlaceOrder( + testapp.MustScaleOrder( + constants.Order_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB26, + testapp.DefaultGenesis(), + ), + ) + PlaceOrder_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20 = *clobtypes.NewMsgPlaceOrder( + testapp.MustScaleOrder( + constants.Order_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20, + testapp.DefaultGenesis(), + ), + ) + PlaceOrder_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO = *clobtypes.NewMsgPlaceOrder( + testapp.MustScaleOrder( + constants.Order_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO, + testapp.DefaultGenesis(), + ), + ) + PlaceOrder_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23 = *clobtypes.NewMsgPlaceOrder( + testapp.MustScaleOrder( + constants.Order_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23, + testapp.DefaultGenesis(), + ), + ) +) + +// Tests cancelling multiple orders. +func TestBatchCancelBatchFunctionality(t *testing.T) { + tests := map[string]struct { + firstBlockOrders []clobtypes.MsgPlaceOrder + firstBlockBatchCancel []clobtypes.MsgBatchCancel + secondBlockOrders map[clobtypes.MsgPlaceOrder]bool + secondBlockBatchCancel []clobtypes.MsgBatchCancel + + expectedOrderIdsInMemclob map[clobtypes.OrderId]bool + expectedCancelExpirationsInMemclob map[clobtypes.OrderId]uint32 + }{ + "Cancel a batch of orders, one not cancelled": { + firstBlockOrders: []clobtypes.MsgPlaceOrder{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20, + PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20, + PlaceOrder_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB20, // not cancelled + PlaceOrder_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20, + PlaceOrder_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO, + PlaceOrder_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23, + }, + firstBlockBatchCancel: []clobtypes.MsgBatchCancel{ + { + SubaccountId: PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId.SubaccountId, + ShortTermCancels: []clobtypes.OrderBatch{ + { + ClobPairId: 0, + ClientIds: []uint32{0, 3}, + }, + { + ClobPairId: 1, + ClientIds: []uint32{1, 4, 5}, + }, + }, + GoodTilBlock: 25, + }, + }, + expectedOrderIdsInMemclob: map[clobtypes.OrderId]bool{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId: false, + constants.Order_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.OrderId: false, + PlaceOrder_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB20.Order.OrderId: true, + constants.Order_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20.OrderId: false, + constants.Order_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO.OrderId: false, + PlaceOrder_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23.Order.OrderId: false, + }, + expectedCancelExpirationsInMemclob: map[clobtypes.OrderId]uint32{ + constants.Order_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.OrderId: 25, + constants.Order_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.OrderId: 25, + constants.Order_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20.OrderId: 25, + constants.Order_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO.OrderId: 25, + PlaceOrder_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23.Order.OrderId: 25, + }, + }, + "Cancel a batch of orders, one cancel gtb is for an order with greater gtb": { + firstBlockOrders: []clobtypes.MsgPlaceOrder{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20, + PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20, + PlaceOrder_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB26, // gtb 26 is > cancel gtb + PlaceOrder_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20, + PlaceOrder_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO, + PlaceOrder_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23, + }, + firstBlockBatchCancel: []clobtypes.MsgBatchCancel{ + { + SubaccountId: PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId.SubaccountId, + ShortTermCancels: []clobtypes.OrderBatch{ + { + ClobPairId: 0, + ClientIds: []uint32{0, 3}, + }, + { + ClobPairId: 1, + ClientIds: []uint32{1, 4, 5}, + }, + }, + GoodTilBlock: 25, + }, + }, + expectedOrderIdsInMemclob: map[clobtypes.OrderId]bool{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId: false, + constants.Order_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.OrderId: false, + PlaceOrder_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB26.Order.OrderId: true, + constants.Order_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20.OrderId: false, + constants.Order_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO.OrderId: false, + PlaceOrder_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23.Order.OrderId: false, + }, + expectedCancelExpirationsInMemclob: map[clobtypes.OrderId]uint32{ + constants.Order_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.OrderId: 25, + constants.Order_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.OrderId: 25, + // no cancel for client_id 5 since cancel gtb < existing order gtb + PlaceOrder_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB26.Order.OrderId: 0, + constants.Order_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20.OrderId: 25, + constants.Order_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO.OrderId: 25, + PlaceOrder_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23.Order.OrderId: 25, + }, + }, + "Cancel two batch of orders, overwriting cancel gtb with higher values": { + firstBlockOrders: []clobtypes.MsgPlaceOrder{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20, + PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20, + PlaceOrder_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB26, + PlaceOrder_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20, + PlaceOrder_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO, + PlaceOrder_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23, + }, + firstBlockBatchCancel: []clobtypes.MsgBatchCancel{ + { + SubaccountId: PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId.SubaccountId, + ShortTermCancels: []clobtypes.OrderBatch{ + { + ClobPairId: 0, + ClientIds: []uint32{0, 3}, + }, + { + ClobPairId: 1, + ClientIds: []uint32{1, 4, 5}, + }, + }, + GoodTilBlock: 25, + }, + }, + secondBlockBatchCancel: []clobtypes.MsgBatchCancel{ + { + SubaccountId: PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId.SubaccountId, + ShortTermCancels: []clobtypes.OrderBatch{ + { + ClobPairId: 0, + ClientIds: []uint32{0, 3}, + }, + { + ClobPairId: 1, + ClientIds: []uint32{1, 2, 4, 5}, + }, + }, + GoodTilBlock: 30, + }, + }, + expectedOrderIdsInMemclob: map[clobtypes.OrderId]bool{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId: false, + constants.Order_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.OrderId: false, + PlaceOrder_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB26.Order.OrderId: false, + constants.Order_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20.OrderId: false, + constants.Order_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO.OrderId: false, + PlaceOrder_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23.Order.OrderId: false, + }, + expectedCancelExpirationsInMemclob: map[clobtypes.OrderId]uint32{ + constants.Order_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.OrderId: 30, + constants.Order_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.OrderId: 30, + PlaceOrder_Alice_Num1_Id2_Clob1_Buy10_Price10_GTB26.Order.OrderId: 30, + constants.Order_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20.OrderId: 30, + constants.Order_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO.OrderId: 30, + PlaceOrder_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23.Order.OrderId: 30, + }, + }, + "Batch cancels prevent new orders with lower gtb from being placed": { + firstBlockOrders: []clobtypes.MsgPlaceOrder{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20, + PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20, + PlaceOrder_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20, + }, + firstBlockBatchCancel: []clobtypes.MsgBatchCancel{ + { + SubaccountId: PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId.SubaccountId, + ShortTermCancels: []clobtypes.OrderBatch{ + { + ClobPairId: 0, + ClientIds: []uint32{0, 3}, + }, + { + ClobPairId: 1, + ClientIds: []uint32{1}, + }, + }, + GoodTilBlock: 25, + }, + }, + secondBlockOrders: map[clobtypes.MsgPlaceOrder]bool{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20: false, + PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20: false, + PlaceOrder_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20: false, + }, + expectedOrderIdsInMemclob: map[clobtypes.OrderId]bool{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId: false, + constants.Order_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.OrderId: false, + constants.Order_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20.OrderId: false, + }, + expectedCancelExpirationsInMemclob: map[clobtypes.OrderId]uint32{ + constants.Order_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.OrderId: 25, + constants.Order_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.OrderId: 25, + constants.Order_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20.OrderId: 25, + }, + }, + "Batch cancel nonexistent orders": { + firstBlockOrders: []clobtypes.MsgPlaceOrder{}, + firstBlockBatchCancel: []clobtypes.MsgBatchCancel{ + { + SubaccountId: PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId.SubaccountId, + ShortTermCancels: []clobtypes.OrderBatch{ + { + ClobPairId: 0, + ClientIds: []uint32{0, 3}, + }, + { + ClobPairId: 1, + ClientIds: []uint32{1, 4, 5}, + }, + }, + GoodTilBlock: 25, + }, + }, + secondBlockOrders: map[clobtypes.MsgPlaceOrder]bool{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20: false, + PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20: false, + PlaceOrder_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20: false, + }, + expectedOrderIdsInMemclob: map[clobtypes.OrderId]bool{}, + expectedCancelExpirationsInMemclob: map[clobtypes.OrderId]uint32{ + constants.Order_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.OrderId: 25, + constants.Order_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.OrderId: 25, + constants.Order_Alice_Num1_Id3_Clob0_Sell100_Price100000_GTB20.OrderId: 25, + constants.Order_Alice_Num1_Id4_Clob1_Sell10_Price15_GTB20_PO.OrderId: 25, + PlaceOrder_Alice_Num1_Id5_Clob1_Buy10_Price15_GTB23.Order.OrderId: 25, + }, + }, + "Batch cancel does not prevent orders with higher gtb from being placed": { + firstBlockOrders: []clobtypes.MsgPlaceOrder{}, + firstBlockBatchCancel: []clobtypes.MsgBatchCancel{ + { + SubaccountId: PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId.SubaccountId, + ShortTermCancels: []clobtypes.OrderBatch{ + { + ClobPairId: 0, + ClientIds: []uint32{0, 3}, + }, + { + ClobPairId: 1, + ClientIds: []uint32{1, 4, 5}, + }, + }, + GoodTilBlock: 25, + }, + }, + secondBlockOrders: map[clobtypes.MsgPlaceOrder]bool{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB30: true, + }, + expectedOrderIdsInMemclob: map[clobtypes.OrderId]bool{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB30.Order.OrderId: true, + }, + expectedCancelExpirationsInMemclob: map[clobtypes.OrderId]uint32{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB30.Order.OrderId: 25, + }, + }, + } + + for name, tc := range tests { + t.Run(name, func(t *testing.T) { + tApp := testapp.NewTestAppBuilder(t).WithGenesisDocFn(func() (genesis types.GenesisDoc) { + genesis = testapp.DefaultGenesis() + testapp.UpdateGenesisDocWithAppStateForModule( + &genesis, + func(genesisState *clobtypes.GenesisState) { + genesisState.EquityTierLimitConfig = clobtypes.EquityTierLimitConfiguration{} + }, + ) + testapp.UpdateGenesisDocWithAppStateForModule( + &genesis, + func(genesisState *satypes.GenesisState) { + genesisState.Subaccounts = []satypes.Subaccount{ + constants.Alice_Num1_100_000USD, + } + }, + ) + return genesis + }).WithCrashingAppCheckTxNonDeterminismChecksEnabled(false).Build() + _ = tApp.InitChain() + + // Advance block to 10 + ctx := tApp.AdvanceToBlock(10, testapp.AdvanceToBlockOptions{}) + + // Place first block orders and cancels + for _, order := range tc.firstBlockOrders { + for _, checkTx := range testapp.MustMakeCheckTxsWithClobMsg(ctx, tApp.App, order) { + resp := tApp.CheckTx(checkTx) + require.Conditionf(t, resp.IsOK, "Expected CheckTx to succeed. Response: %+v", resp) + } + } + for _, batch := range tc.firstBlockBatchCancel { + for _, checkTx := range testapp.MustMakeCheckTxsWithClobMsg(ctx, tApp.App, batch) { + tApp.CheckTx(checkTx) + } + } + + // Advance block to 15 + ctx = tApp.AdvanceToBlock(15, testapp.AdvanceToBlockOptions{}) + + // Place second block orders and cancels + for order, shouldSucceed := range tc.secondBlockOrders { + for _, checkTx := range testapp.MustMakeCheckTxsWithClobMsg(ctx, tApp.App, order) { + resp := tApp.CheckTx(checkTx) + require.Equal(t, shouldSucceed, resp.IsOK(), "Expected CheckTx to succeed. Response: %+v", resp) + } + } + for _, batch := range tc.secondBlockBatchCancel { + for _, checkTx := range testapp.MustMakeCheckTxsWithClobMsg(ctx, tApp.App, batch) { + tApp.CheckTx(checkTx) + } + } + + // Verify expectations + for orderId, shouldHaveOrder := range tc.expectedOrderIdsInMemclob { + _, exists := tApp.App.ClobKeeper.MemClob.GetOrder(ctx, orderId) + require.Equal(t, shouldHaveOrder, exists) + } + for orderId, expectedCancelExpirationBlock := range tc.expectedCancelExpirationsInMemclob { + cancelExpirationBlock, exists := tApp.App.ClobKeeper.MemClob.GetCancelOrder(ctx, orderId) + if expectedCancelExpirationBlock > 0 { + require.True(t, exists) + require.Equal(t, expectedCancelExpirationBlock, cancelExpirationBlock) + } else { + require.False(t, exists) + } + } + }) + } +} + +// Tests emitting offchain updates. +func TestBatchCancelOffchainUpdates(t *testing.T) { + tApp := testapp.NewTestAppBuilder(t).Build() + ctx := tApp.InitChain() + + batchCancel := clobtypes.MsgBatchCancel{ + SubaccountId: PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId.SubaccountId, + ShortTermCancels: []clobtypes.OrderBatch{ + { + ClobPairId: 0, + ClientIds: []uint32{0, 3}, + }, + { + ClobPairId: 1, + ClientIds: []uint32{1, 2}, + }, + }, + GoodTilBlock: 25, + } + + CheckTx_PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20 := testapp.MustMakeCheckTx( + ctx, + tApp.App, + testapp.MustMakeCheckTxOptions{ + AccAddressForSigning: constants.Alice_Num1.Owner, + }, + &PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20, + ) + CheckTx_PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20 := testapp.MustMakeCheckTx( + ctx, + tApp.App, + testapp.MustMakeCheckTxOptions{ + AccAddressForSigning: constants.Alice_Num1.Owner, + }, + &PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20, + ) + CheckTx_BatchCancel := testapp.MustMakeCheckTx( + ctx, + tApp.App, + testapp.MustMakeCheckTxOptions{ + AccAddressForSigning: constants.Alice_Num1.Owner, + }, + &batchCancel, + ) + + tests := map[string]struct { + firstBlockOrders []clobtypes.MsgPlaceOrder + firstBlockBatchCancel []clobtypes.MsgBatchCancel + + expectedOffchainUpdates []msgsender.Message + }{ + "Cancel a batch of orders and check offchain updates": { + firstBlockOrders: []clobtypes.MsgPlaceOrder{ + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20, + PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20, + }, + firstBlockBatchCancel: []clobtypes.MsgBatchCancel{ + { + SubaccountId: PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId.SubaccountId, + ShortTermCancels: []clobtypes.OrderBatch{ + { + ClobPairId: 0, + ClientIds: []uint32{0, 3}, + }, + { + ClobPairId: 1, + ClientIds: []uint32{1, 2}, + }, + }, + GoodTilBlock: 25, + }, + }, + expectedOffchainUpdates: []msgsender.Message{ + off_chain_updates.MustCreateOrderPlaceMessage( + ctx, + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order, + ).AddHeader(msgsender.MessageHeader{ + Key: msgsender.TransactionHashHeaderKey, + Value: tmhash.Sum(CheckTx_PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Tx), + }), + off_chain_updates.MustCreateOrderUpdateMessage( + ctx, + PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Order.OrderId, + 0, + ).AddHeader(msgsender.MessageHeader{ + Key: msgsender.TransactionHashHeaderKey, + Value: tmhash.Sum(CheckTx_PlaceOrder_Alice_Num1_Id0_Clob0_Sell10_Price10_GTB20.Tx), + }), + off_chain_updates.MustCreateOrderPlaceMessage( + ctx, + PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.Order, + ).AddHeader(msgsender.MessageHeader{ + Key: msgsender.TransactionHashHeaderKey, + Value: tmhash.Sum(CheckTx_PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.Tx), + }), + off_chain_updates.MustCreateOrderUpdateMessage( + ctx, + PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.Order.OrderId, + 0, + ).AddHeader(msgsender.MessageHeader{ + Key: msgsender.TransactionHashHeaderKey, + Value: tmhash.Sum(CheckTx_PlaceOrder_Alice_Num1_Id1_Clob1_Sell10_Price15_GTB20.Tx), + }), + // 4 removals from the batch remove operation + off_chain_updates.MustCreateOrderRemoveMessageWithReason( + ctx, + // Order id for the first cancel in the batch. + clobtypes.OrderId{ + SubaccountId: constants.Alice_Num1, + ClientId: 0, + OrderFlags: clobtypes.OrderIdFlags_ShortTerm, + ClobPairId: 0, + }, + indexersharedtypes.OrderRemovalReason_ORDER_REMOVAL_REASON_USER_CANCELED, + ocutypes.OrderRemoveV1_ORDER_REMOVAL_STATUS_BEST_EFFORT_CANCELED, + ).AddHeader(msgsender.MessageHeader{ + Key: msgsender.TransactionHashHeaderKey, + Value: tmhash.Sum(CheckTx_BatchCancel.Tx), + }), + off_chain_updates.MustCreateOrderRemoveMessageWithReason( + ctx, + // Order id for the second cancel in the batch. + clobtypes.OrderId{ + SubaccountId: constants.Alice_Num1, + ClientId: 3, + OrderFlags: clobtypes.OrderIdFlags_ShortTerm, + ClobPairId: 0, + }, + indexersharedtypes.OrderRemovalReason_ORDER_REMOVAL_REASON_USER_CANCELED, + ocutypes.OrderRemoveV1_ORDER_REMOVAL_STATUS_BEST_EFFORT_CANCELED, + ).AddHeader(msgsender.MessageHeader{ + Key: msgsender.TransactionHashHeaderKey, + Value: tmhash.Sum(CheckTx_BatchCancel.Tx), + }), + off_chain_updates.MustCreateOrderRemoveMessageWithReason( + ctx, + // Order id for the third cancel in the batch. + clobtypes.OrderId{ + SubaccountId: constants.Alice_Num1, + ClientId: 1, + OrderFlags: clobtypes.OrderIdFlags_ShortTerm, + ClobPairId: 1, + }, + indexersharedtypes.OrderRemovalReason_ORDER_REMOVAL_REASON_USER_CANCELED, + ocutypes.OrderRemoveV1_ORDER_REMOVAL_STATUS_BEST_EFFORT_CANCELED, + ).AddHeader(msgsender.MessageHeader{ + Key: msgsender.TransactionHashHeaderKey, + Value: tmhash.Sum(CheckTx_BatchCancel.Tx), + }), + off_chain_updates.MustCreateOrderRemoveMessageWithReason( + ctx, + // Order id for the fourth cancel in the batch. + clobtypes.OrderId{ + SubaccountId: constants.Alice_Num1, + ClientId: 2, + OrderFlags: clobtypes.OrderIdFlags_ShortTerm, + ClobPairId: 1, + }, + indexersharedtypes.OrderRemovalReason_ORDER_REMOVAL_REASON_USER_CANCELED, + ocutypes.OrderRemoveV1_ORDER_REMOVAL_STATUS_BEST_EFFORT_CANCELED, + ).AddHeader(msgsender.MessageHeader{ + Key: msgsender.TransactionHashHeaderKey, + Value: tmhash.Sum(CheckTx_BatchCancel.Tx), + }), + }, + }, + } + + for name, tc := range tests { + t.Run(name, func(t *testing.T) { + msgSender := msgsender.NewIndexerMessageSenderInMemoryCollector() + appOpts := map[string]interface{}{ + indexer.MsgSenderInstanceForTest: msgSender, + } + tApp := testapp.NewTestAppBuilder(t).WithGenesisDocFn(func() (genesis types.GenesisDoc) { + genesis = testapp.DefaultGenesis() + testapp.UpdateGenesisDocWithAppStateForModule( + &genesis, + func(genesisState *satypes.GenesisState) { + genesisState.Subaccounts = []satypes.Subaccount{ + constants.Alice_Num1_100_000USD, + } + }, + ) + testapp.UpdateGenesisDocWithAppStateForModule( + &genesis, + func(genesisState *clobtypes.GenesisState) { + genesisState.EquityTierLimitConfig = clobtypes.EquityTierLimitConfiguration{} + }, + ) + return genesis + }).WithCrashingAppCheckTxNonDeterminismChecksEnabled(false).WithAppOptions(appOpts).Build() + _ = tApp.InitChain() + + // Advance block to 10 + ctx := tApp.AdvanceToBlock(10, testapp.AdvanceToBlockOptions{}) + // Clear any messages produced prior to these checkTx calls. + msgSender.Clear() + + // Place first block orders and cancels + for _, order := range tc.firstBlockOrders { + for _, checkTx := range testapp.MustMakeCheckTxsWithClobMsg(ctx, tApp.App, order) { + resp := tApp.CheckTx(checkTx) + require.Conditionf(t, resp.IsOK, "Expected CheckTx to succeed. Response: %+v", resp) + } + } + for _, batch := range tc.firstBlockBatchCancel { + for _, checkTx := range testapp.MustMakeCheckTxsWithClobMsg(ctx, tApp.App, batch) { + tApp.CheckTx(checkTx) + } + } + + // Verify offchain messages + require.ElementsMatch( + t, + tc.expectedOffchainUpdates[:4], + msgSender.GetOffchainMessages()[:4], + ) + }) + } +} diff --git a/protocol/x/clob/keeper/orders.go b/protocol/x/clob/keeper/orders.go index db709c469d..f65aff2fce 100644 --- a/protocol/x/clob/keeper/orders.go +++ b/protocol/x/clob/keeper/orders.go @@ -109,11 +109,10 @@ func (k Keeper) BatchCancelShortTermOrder( log.Error, err, ) } else { - success = append(failure, clientId) + success = append(success, clientId) } } } - return success, failure, nil }