package gateio import ( "bytes" "context" "fmt" "log" "os" "slices" "strconv" "strings" "sync" "testing" "time" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" "github.com/thrasher-corp/gocryptotrader/common" "github.com/thrasher-corp/gocryptotrader/common/key" "github.com/thrasher-corp/gocryptotrader/core" "github.com/thrasher-corp/gocryptotrader/currency" "github.com/thrasher-corp/gocryptotrader/encoding/json" "github.com/thrasher-corp/gocryptotrader/exchange/websocket" "github.com/thrasher-corp/gocryptotrader/exchanges/account" "github.com/thrasher-corp/gocryptotrader/exchanges/asset" "github.com/thrasher-corp/gocryptotrader/exchanges/fundingrate" "github.com/thrasher-corp/gocryptotrader/exchanges/futures" "github.com/thrasher-corp/gocryptotrader/exchanges/kline" "github.com/thrasher-corp/gocryptotrader/exchanges/order" "github.com/thrasher-corp/gocryptotrader/exchanges/request" "github.com/thrasher-corp/gocryptotrader/exchanges/sharedtestvalues" "github.com/thrasher-corp/gocryptotrader/exchanges/subscription" testexch "github.com/thrasher-corp/gocryptotrader/internal/testing/exchange" testsubs "github.com/thrasher-corp/gocryptotrader/internal/testing/subscriptions" "github.com/thrasher-corp/gocryptotrader/portfolio/withdraw" "github.com/thrasher-corp/gocryptotrader/types" ) // Please supply your own APIKEYS here for due diligence testing const ( apiKey = "" apiSecret = "" canManipulateRealOrders = false ) var e *Exchange func TestMain(m *testing.M) { e = new(Exchange) if err := testexch.Setup(e); err != nil { log.Fatalf("Gateio Setup error: %s", err) } if apiKey != "" && apiSecret != "" { e.API.AuthenticatedSupport = true e.API.AuthenticatedWebsocketSupport = true e.SetCredentials(apiKey, apiSecret, "", "", "", "") } os.Exit(m.Run()) } func TestUpdateTradablePairs(t *testing.T) { t.Parallel() testexch.UpdatePairsOnce(t, e) } func TestCancelAllExchangeOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.CancelAllOrders(t.Context(), nil) require.ErrorIs(t, err, order.ErrCancelOrderIsNil) r := &order.Cancel{ OrderID: "1", AccountID: "1", } for _, a := range e.GetAssetTypes(false) { r.AssetType = a r.Pair = currency.EMPTYPAIR _, err = e.CancelAllOrders(t.Context(), r) assert.ErrorIs(t, err, currency.ErrCurrencyPairEmpty) r.Pair = getPair(t, a) _, err = e.CancelAllOrders(t.Context(), r) require.NoError(t, err) } } func TestGetAccountInfo(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) for _, a := range e.GetAssetTypes(false) { _, err := e.UpdateAccountInfo(t.Context(), a) assert.NoErrorf(t, err, "UpdateAccountInfo should not error for asset %s", a) } } func TestWithdraw(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) cryptocurrencyChains, err := e.GetAvailableTransferChains(t.Context(), currency.BTC) require.NoError(t, err, "GetAvailableTransferChains must not error") require.NotEmpty(t, cryptocurrencyChains, "GetAvailableTransferChains must return some chains") withdrawCryptoRequest := withdraw.Request{ Exchange: e.Name, Amount: 1, Currency: currency.BTC, Description: "WITHDRAW IT ALL", Crypto: withdraw.CryptoRequest{ Address: core.BitcoinDonationAddress, Chain: cryptocurrencyChains[0], }, } _, err = e.WithdrawCryptocurrencyFunds(t.Context(), &withdrawCryptoRequest) require.NoError(t, err) } func TestGetOrderInfo(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) for _, a := range e.GetAssetTypes(false) { _, err := e.GetOrderInfo(t.Context(), "917591554", getPair(t, a), a) require.NoErrorf(t, err, "GetOrderInfo must not error for asset %s", a) } } func TestUpdateTicker(t *testing.T) { t.Parallel() for _, a := range e.GetAssetTypes(false) { _, err := e.UpdateTicker(t.Context(), getPair(t, a), a) assert.NoErrorf(t, err, "UpdateTicker should not error for %s", a) } } func TestListSpotCurrencies(t *testing.T) { t.Parallel() if _, err := e.ListSpotCurrencies(t.Context()); err != nil { t.Errorf("%s ListAllCurrencies() error %v", e.Name, err) } } func TestGetCurrencyDetail(t *testing.T) { t.Parallel() if _, err := e.GetCurrencyDetail(t.Context(), currency.BTC); err != nil { t.Errorf("%s GetCurrencyDetail() error %v", e.Name, err) } } func TestListAllCurrencyPairs(t *testing.T) { t.Parallel() if _, err := e.ListSpotCurrencyPairs(t.Context()); err != nil { t.Errorf("%s ListAllCurrencyPairs() error %v", e.Name, err) } } func TestGetCurrencyPairDetal(t *testing.T) { t.Parallel() if _, err := e.GetCurrencyPairDetail(t.Context(), currency.Pair{Base: currency.BTC, Quote: currency.USDT, Delimiter: currency.UnderscoreDelimiter}.String()); err != nil { t.Errorf("%s GetCurrencyPairDetal() error %v", e.Name, err) } } func TestGetTickers(t *testing.T) { t.Parallel() if _, err := e.GetTickers(t.Context(), "BTC_USDT", ""); err != nil { t.Errorf("%s GetTickers() error %v", e.Name, err) } } func TestGetTicker(t *testing.T) { t.Parallel() if _, err := e.GetTicker(t.Context(), currency.Pair{Base: currency.BTC, Delimiter: currency.UnderscoreDelimiter, Quote: currency.USDT}.String(), utc8TimeZone); err != nil { t.Errorf("%s GetTicker() error %v", e.Name, err) } } func TestGetOrderbook(t *testing.T) { t.Parallel() _, err := e.GetOrderbook(t.Context(), getPair(t, asset.Spot).String(), "0.1", 10, false) assert.NoError(t, err, "GetOrderbook should not error") } func TestGetMarketTrades(t *testing.T) { t.Parallel() if _, err := e.GetMarketTrades(t.Context(), getPair(t, asset.Spot), 0, "", true, time.Time{}, time.Time{}, 1); err != nil { t.Errorf("%s GetMarketTrades() error %v", e.Name, err) } } func TestCandlestickUnmarshalJSON(t *testing.T) { t.Parallel() data := []byte(`[["1738108800","229534412.73508700","103734.3","104779.9","101336.6","101343.8","2232.94510000","true"],["1738195200","178316032.62306100","104718.6","106467.1","103286.4","103734.4","1695.00787000","true"],["1738281600","231315376.16747100","102431","106042.7","101555.9","104718.6","2228.03609000","true"]]`) var targets []Candlestick err := json.Unmarshal(data, &targets) require.NoError(t, err) require.Len(t, targets, 3) assert.Equal(t, Candlestick{ Timestamp: types.Time(time.Unix(1738108800, 0)), QuoteCcyVolume: 229534412.73508700, ClosePrice: 103734.3, HighestPrice: 104779.9, LowestPrice: 101336.6, OpenPrice: 101343.8, BaseCcyAmount: 2232.94510000, WindowClosed: true, }, targets[0]) } func TestGetCandlesticks(t *testing.T) { t.Parallel() if _, err := e.GetCandlesticks(t.Context(), getPair(t, asset.Spot), 0, time.Time{}, time.Time{}, kline.OneDay); err != nil { t.Errorf("%s GetCandlesticks() error %v", e.Name, err) } } func TestGetTradingFeeRatio(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetTradingFeeRatio(t.Context(), currency.Pair{Base: currency.BTC, Quote: currency.USDT, Delimiter: currency.UnderscoreDelimiter}); err != nil { t.Errorf("%s GetTradingFeeRatio() error %v", e.Name, err) } } func TestGetSpotAccounts(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetSpotAccounts(t.Context(), currency.BTC); err != nil { t.Errorf("%s GetSpotAccounts() error %v", e.Name, err) } } func TestCreateBatchOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.CreateBatchOrders(t.Context(), []CreateOrderRequest{ { CurrencyPair: getPair(t, asset.Spot), Side: "sell", Amount: 0.001, Price: 12349, Account: e.assetTypeToString(asset.Spot), Type: "limit", }, { CurrencyPair: currency.Pair{Base: currency.BTC, Quote: currency.USDT, Delimiter: currency.UnderscoreDelimiter}, Side: "buy", Amount: 1, Price: 1234567789, Account: e.assetTypeToString(asset.Spot), Type: "limit", }, }) assert.NoError(t, err, "CreateBatchOrders should not error") } func TestGetSpotOpenOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetSpotOpenOrders(t.Context(), 0, 0, false); err != nil { t.Errorf("%s GetSpotOpenOrders() error %v", e.Name, err) } } func TestSpotClosePositionWhenCrossCurrencyDisabled(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.SpotClosePositionWhenCrossCurrencyDisabled(t.Context(), &ClosePositionRequestParam{ Amount: 0.1, Price: 1234567384, CurrencyPair: getPair(t, asset.Spot), }); err != nil { t.Errorf("%s SpotClosePositionWhenCrossCurrencyDisabled() error %v", e.Name, err) } } func TestCreateSpotOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.PlaceSpotOrder(t.Context(), &CreateOrderRequest{ CurrencyPair: getPair(t, asset.Spot), Side: "buy", Amount: 1, Price: 900000, Account: e.assetTypeToString(asset.Spot), Type: "limit", }) assert.NoError(t, err, "PlaceSpotOrder should not error") } func TestGetSpotOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetSpotOrders(t.Context(), currency.Pair{Base: currency.BTC, Quote: currency.USDT, Delimiter: currency.UnderscoreDelimiter}, statusOpen, 0, 0) assert.NoError(t, err, "GetSpotOrders should not error") } func TestCancelAllOpenOrdersSpecifiedCurrencyPair(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CancelAllOpenOrdersSpecifiedCurrencyPair(t.Context(), getPair(t, asset.Spot), order.Sell, asset.Empty); err != nil { t.Errorf("%s CancelAllOpenOrdersSpecifiedCurrencyPair() error %v", e.Name, err) } } func TestCancelBatchOrdersWithIDList(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CancelBatchOrdersWithIDList(t.Context(), []CancelOrderByIDParam{ { CurrencyPair: getPair(t, asset.Spot), ID: "1234567", }, { CurrencyPair: currency.Pair{Base: currency.BTC, Quote: currency.USDT, Delimiter: currency.UnderscoreDelimiter}, ID: "something", }, }); err != nil { t.Errorf("%s CancelBatchOrderWithIDList() error %v", e.Name, err) } } func TestGetSpotOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetSpotOrder(t.Context(), "1234", currency.Pair{ Base: currency.BTC, Delimiter: currency.UnderscoreDelimiter, Quote: currency.USDT, }, asset.Spot); err != nil { t.Errorf("%s GetSpotOrder() error %v", e.Name, err) } } func TestAmendSpotOrder(t *testing.T) { t.Parallel() _, err := e.AmendSpotOrder(t.Context(), "", getPair(t, asset.Spot), false, &PriceAndAmount{ Price: 1000, }) assert.ErrorIs(t, err, errInvalidOrderID) _, err = e.AmendSpotOrder(t.Context(), "123", currency.EMPTYPAIR, false, &PriceAndAmount{ Price: 1000, }) assert.ErrorIs(t, err, currency.ErrCurrencyPairEmpty) sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err = e.AmendSpotOrder(t.Context(), "123", getPair(t, asset.Spot), false, &PriceAndAmount{ Price: 1000, }) if err != nil { t.Error(err) } } func TestCancelSingleSpotOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CancelSingleSpotOrder(t.Context(), "1234", getPair(t, asset.Spot).String(), false); err != nil { t.Errorf("%s CancelSingleSpotOrder() error %v", e.Name, err) } } func TestGetMySpotTradingHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetMySpotTradingHistory(t.Context(), currency.Pair{Base: currency.BTC, Quote: currency.USDT, Delimiter: currency.UnderscoreDelimiter}, "", 0, 0, false, time.Time{}, time.Time{}) require.NoError(t, err) } func TestGetServerTime(t *testing.T) { t.Parallel() if _, err := e.GetServerTime(t.Context(), asset.Spot); err != nil { t.Errorf("%s GetServerTime() error %v", e.Name, err) } } func TestCountdownCancelorder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CountdownCancelorders(t.Context(), CountdownCancelOrderParam{ Timeout: 10, CurrencyPair: currency.Pair{Base: currency.BTC, Quote: currency.ETH, Delimiter: currency.UnderscoreDelimiter}, }); err != nil { t.Errorf("%s CountdownCancelorder() error %v", e.Name, err) } } func TestCreatePriceTriggeredOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CreatePriceTriggeredOrder(t.Context(), &PriceTriggeredOrderParam{ Trigger: TriggerPriceInfo{ Price: 123, Rule: ">=", Expiration: 3600, }, Put: PutOrderData{ Type: "limit", Side: "sell", Price: 2312312, Amount: 30, TimeInForce: "gtc", }, Market: currency.Pair{Base: currency.GT, Quote: currency.USDT, Delimiter: currency.UnderscoreDelimiter}, }); err != nil { t.Errorf("%s CreatePriceTriggeredOrder() error %v", e.Name, err) } } func TestGetPriceTriggeredOrderList(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetPriceTriggeredOrderList(t.Context(), statusOpen, currency.EMPTYPAIR, asset.Empty, 0, 0) assert.NoError(t, err, "GetPriceTriggeredOrderList should not error") } func TestCancelAllOpenOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CancelMultipleSpotOpenOrders(t.Context(), currency.EMPTYPAIR, asset.CrossMargin); err != nil { t.Errorf("%s CancelAllOpenOrders() error %v", e.Name, err) } } func TestGetSinglePriceTriggeredOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetSinglePriceTriggeredOrder(t.Context(), "1234"); err != nil { t.Errorf("%s GetSinglePriceTriggeredOrder() error %v", e.Name, err) } } func TestCancelPriceTriggeredOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.CancelPriceTriggeredOrder(t.Context(), "1234"); err != nil { t.Errorf("%s CancelPriceTriggeredOrder() error %v", e.Name, err) } } func TestGetMarginAccountList(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetMarginAccountList(t.Context(), currency.EMPTYPAIR); err != nil { t.Errorf("%s GetMarginAccountList() error %v", e.Name, err) } } func TestListMarginAccountBalanceChangeHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.ListMarginAccountBalanceChangeHistory(t.Context(), currency.BTC, currency.Pair{ Base: currency.BTC, Delimiter: currency.UnderscoreDelimiter, Quote: currency.USDT, }, time.Time{}, time.Time{}, 0, 0); err != nil { t.Errorf("%s ListMarginAccountBalanceChangeHistory() error %v", e.Name, err) } } func TestGetMarginFundingAccountList(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetMarginFundingAccountList(t.Context(), currency.BTC); err != nil { t.Errorf("%s GetMarginFundingAccountList %v", e.Name, err) } } func TestMarginLoan(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.MarginLoan(t.Context(), &MarginLoanRequestParam{ Side: "borrow", Amount: 1, Currency: currency.BTC, CurrencyPair: currency.Pair{Base: currency.BTC, Quote: currency.USDT, Delimiter: currency.UnderscoreDelimiter}, Days: 10, Rate: 0.0002, }); err != nil { t.Errorf("%s MarginLoan() error %v", e.Name, err) } } func TestGetMarginAllLoans(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetMarginAllLoans(t.Context(), statusOpen, "lend", "", currency.BTC, currency.Pair{Base: currency.BTC, Delimiter: currency.UnderscoreDelimiter, Quote: currency.USDT}, false, 0, 0) assert.NoError(t, err, "GetMarginAllLoans should not error") } func TestMergeMultipleLendingLoans(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.MergeMultipleLendingLoans(t.Context(), currency.USDT, []string{"123", "23423"}); err != nil { t.Errorf("%s MergeMultipleLendingLoans() error %v", e.Name, err) } } func TestRetriveOneSingleLoanDetail(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.RetriveOneSingleLoanDetail(t.Context(), "borrow", "123"); err != nil { t.Errorf("%s RetriveOneSingleLoanDetail() error %v", e.Name, err) } } func TestModifyALoan(t *testing.T) { t.Parallel() _, err := e.ModifyALoan(t.Context(), "1234", &ModifyLoanRequestParam{ Currency: currency.BTC, Side: "borrow", AutoRenew: false, }) assert.ErrorIs(t, err, currency.ErrCurrencyPairEmpty) sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.ModifyALoan(t.Context(), "1234", &ModifyLoanRequestParam{ Currency: currency.BTC, Side: "borrow", AutoRenew: false, CurrencyPair: currency.Pair{Base: currency.BTC, Quote: currency.USDT, Delimiter: currency.UnderscoreDelimiter}, }); err != nil { t.Errorf("%s ModifyALoan() error %v", e.Name, err) } } func TestCancelLendingLoan(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.CancelLendingLoan(t.Context(), currency.BTC, "1234"); err != nil { t.Errorf("%s CancelLendingLoan() error %v", e.Name, err) } } func TestRepayALoan(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.RepayALoan(t.Context(), "1234", &RepayLoanRequestParam{ CurrencyPair: currency.NewBTCUSDT(), Currency: currency.BTC, Mode: "all", }); err != nil { t.Errorf("%s RepayALoan() error %v", e.Name, err) } } func TestListLoanRepaymentRecords(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.ListLoanRepaymentRecords(t.Context(), "1234"); err != nil { t.Errorf("%s LoanRepaymentRecord() error %v", e.Name, err) } } func TestListRepaymentRecordsOfSpecificLoan(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.ListRepaymentRecordsOfSpecificLoan(t.Context(), "1234", "", 0, 0); err != nil { t.Errorf("%s error while ListRepaymentRecordsOfSpecificLoan() %v", e.Name, err) } } func TestGetOneSingleloanRecord(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetOneSingleLoanRecord(t.Context(), "1234", "123"); err != nil { t.Errorf("%s error while GetOneSingleloanRecord() %v", e.Name, err) } } func TestModifyALoanRecord(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.ModifyALoanRecord(t.Context(), "1234", &ModifyLoanRequestParam{ Currency: currency.USDT, CurrencyPair: currency.NewBTCUSDT(), Side: "lend", AutoRenew: true, LoanID: "1234", }); err != nil { t.Errorf("%s ModifyALoanRecord() error %v", e.Name, err) } } func TestUpdateUsersAutoRepaymentSetting(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.UpdateUsersAutoRepaymentSetting(t.Context(), true); err != nil { t.Errorf("%s UpdateUsersAutoRepaymentSetting() error %v", e.Name, err) } } func TestGetUserAutoRepaymentSetting(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetUserAutoRepaymentSetting(t.Context()); err != nil { t.Errorf("%s GetUserAutoRepaymentSetting() error %v", e.Name, err) } } func TestGetMaxTransferableAmountForSpecificMarginCurrency(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetMaxTransferableAmountForSpecificMarginCurrency(t.Context(), currency.BTC, currency.EMPTYPAIR); err != nil { t.Errorf("%s GetMaxTransferableAmountForSpecificMarginCurrency() error %v", e.Name, err) } } func TestGetMaxBorrowableAmountForSpecificMarginCurrency(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetMaxBorrowableAmountForSpecificMarginCurrency(t.Context(), currency.BTC, currency.EMPTYPAIR); err != nil { t.Errorf("%s GetMaxBorrowableAmountForSpecificMarginCurrency() error %v", e.Name, err) } } func TestCurrencySupportedByCrossMargin(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.CurrencySupportedByCrossMargin(t.Context()); err != nil { t.Errorf("%s CurrencySupportedByCrossMargin() error %v", e.Name, err) } } func TestGetCrossMarginSupportedCurrencyDetail(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetCrossMarginSupportedCurrencyDetail(t.Context(), currency.BTC); err != nil { t.Errorf("%s GetCrossMarginSupportedCurrencyDetail() error %v", e.Name, err) } } func TestGetCrossMarginAccounts(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetCrossMarginAccounts(t.Context()); err != nil { t.Errorf("%s GetCrossMarginAccounts() error %v", e.Name, err) } } func TestGetCrossMarginAccountChangeHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetCrossMarginAccountChangeHistory(t.Context(), currency.BTC, time.Time{}, time.Time{}, 0, 6, "in"); err != nil { t.Errorf("%s GetCrossMarginAccountChangeHistory() error %v", e.Name, err) } } var createCrossMarginBorrowLoanJSON = `{"id": "17", "create_time": 1620381696159, "update_time": 1620381696159, "currency": "EOS", "amount": "110.553635", "text": "web", "status": 2, "repaid": "110.506649705159", "repaid_interest": "0.046985294841", "unpaid_interest": "0.0000074393366667"}` func TestCreateCrossMarginBorrowLoan(t *testing.T) { t.Parallel() var response CrossMarginLoanResponse if err := json.Unmarshal([]byte(createCrossMarginBorrowLoanJSON), &response); err != nil { t.Errorf("%s error while deserializing to CrossMarginBorrowLoanResponse %v", e.Name, err) } sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CreateCrossMarginBorrowLoan(t.Context(), CrossMarginBorrowLoanParams{ Currency: currency.BTC, Amount: 3, }); err != nil { t.Errorf("%s CreateCrossMarginBorrowLoan() error %v", e.Name, err) } } func TestGetCrossMarginBorrowHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetCrossMarginBorrowHistory(t.Context(), 1, currency.BTC, 0, 0, false); err != nil { t.Errorf("%s GetCrossMarginBorrowHistory() error %v", e.Name, err) } } func TestGetSingleBorrowLoanDetail(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetSingleBorrowLoanDetail(t.Context(), "1234"); err != nil { t.Errorf("%s GetSingleBorrowLoanDetail() error %v", e.Name, err) } } func TestExecuteRepayment(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.ExecuteRepayment(t.Context(), CurrencyAndAmount{ Currency: currency.USD, Amount: 1234.55, }); err != nil { t.Errorf("%s ExecuteRepayment() error %v", e.Name, err) } } func TestGetCrossMarginRepayments(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetCrossMarginRepayments(t.Context(), currency.BTC, "123", 0, 0, false); err != nil { t.Errorf("%s GetCrossMarginRepayments() error %v", e.Name, err) } } func TestGetMaxTransferableAmountForSpecificCrossMarginCurrency(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetMaxTransferableAmountForSpecificCrossMarginCurrency(t.Context(), currency.BTC); err != nil { t.Errorf("%s GetMaxTransferableAmountForSpecificCrossMarginCurrency() error %v", e.Name, err) } } func TestGetMaxBorrowableAmountForSpecificCrossMarginCurrency(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetMaxBorrowableAmountForSpecificCrossMarginCurrency(t.Context(), currency.BTC); err != nil { t.Errorf("%s GetMaxBorrowableAmountForSpecificCrossMarginCurrency() error %v", e.Name, err) } } func TestListCurrencyChain(t *testing.T) { t.Parallel() if _, err := e.ListCurrencyChain(t.Context(), currency.BTC); err != nil { t.Errorf("%s ListCurrencyChain() error %v", e.Name, err) } } func TestGenerateCurrencyDepositAddress(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GenerateCurrencyDepositAddress(t.Context(), currency.BTC); err != nil { t.Errorf("%s GenerateCurrencyDepositAddress() error %v", e.Name, err) } } func TestGetWithdrawalRecords(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetWithdrawalRecords(t.Context(), currency.BTC, time.Time{}, time.Time{}, 0, 0); err != nil { t.Errorf("%s GetWithdrawalRecords() error %v", e.Name, err) } } func TestGetDepositRecords(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetDepositRecords(t.Context(), currency.BTC, time.Time{}, time.Time{}, 0, 0); err != nil { t.Errorf("%s GetDepositRecords() error %v", e.Name, err) } } func TestTransferCurrency(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.TransferCurrency(t.Context(), &TransferCurrencyParam{ Currency: currency.BTC, From: e.assetTypeToString(asset.Spot), To: e.assetTypeToString(asset.Margin), Amount: 1202.000, CurrencyPair: getPair(t, asset.Spot), }); err != nil { t.Errorf("%s TransferCurrency() error %v", e.Name, err) } } func TestSubAccountTransfer(t *testing.T) { t.Parallel() ctx := t.Context() req := SubAccountTransferParam{SubAccountType: "index"} require.ErrorIs(t, e.SubAccountTransfer(ctx, req), currency.ErrCurrencyCodeEmpty) req.Currency = currency.BTC require.ErrorIs(t, e.SubAccountTransfer(ctx, req), errInvalidSubAccount) req.SubAccount = "1337" require.ErrorIs(t, e.SubAccountTransfer(ctx, req), errInvalidTransferDirection) req.Direction = "to" require.ErrorIs(t, e.SubAccountTransfer(ctx, req), errInvalidAmount) req.Amount = 1.337 require.ErrorIs(t, e.SubAccountTransfer(ctx, req), asset.ErrNotSupported) sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) req.SubAccountType = "spot" require.NoError(t, e.SubAccountTransfer(ctx, req)) } func TestGetSubAccountTransferHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.GetSubAccountTransferHistory(t.Context(), "", time.Time{}, time.Time{}, 0, 0); err != nil { t.Errorf("%s GetSubAccountTransferHistory() error %v", e.Name, err) } } func TestSubAccountTransferToSubAccount(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if err := e.SubAccountTransferToSubAccount(t.Context(), &InterSubAccountTransferParams{ Currency: currency.BTC, SubAccountFromUserID: "1234", SubAccountFromAssetType: asset.Spot, SubAccountToUserID: "4567", SubAccountToAssetType: asset.Spot, Amount: 1234, }); err != nil { t.Error(err) } } func TestGetWithdrawalStatus(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetWithdrawalStatus(t.Context(), currency.NewCode("")); err != nil { t.Errorf("%s GetWithdrawalStatus() error %v", e.Name, err) } } func TestGetSubAccountBalances(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetSubAccountBalances(t.Context(), ""); err != nil { t.Errorf("%s GetSubAccountBalances() error %v", e.Name, err) } } func TestGetSubAccountMarginBalances(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetSubAccountMarginBalances(t.Context(), ""); err != nil { t.Errorf("%s GetSubAccountMarginBalances() error %v", e.Name, err) } } func TestGetSubAccountFuturesBalances(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetSubAccountFuturesBalances(t.Context(), "", currency.EMPTYCODE) assert.Error(t, err, "GetSubAccountFuturesBalances should not error") } func TestGetSubAccountCrossMarginBalances(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetSubAccountCrossMarginBalances(t.Context(), ""); err != nil { t.Errorf("%s GetSubAccountCrossMarginBalances() error %v", e.Name, err) } } func TestGetSavedAddresses(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetSavedAddresses(t.Context(), currency.BTC, "", 0); err != nil { t.Errorf("%s GetSavedAddresses() error %v", e.Name, err) } } func TestGetPersonalTradingFee(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetPersonalTradingFee(t.Context(), currency.Pair{Base: currency.BTC, Quote: currency.USDT, Delimiter: currency.UnderscoreDelimiter}, currency.EMPTYCODE) assert.NoError(t, err, "GetPersonalTradingFee should not error") } func TestGetUsersTotalBalance(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetUsersTotalBalance(t.Context(), currency.BTC); err != nil { t.Errorf("%s GetUsersTotalBalance() error %v", e.Name, err) } } func TestGetMarginSupportedCurrencyPairs(t *testing.T) { t.Parallel() if _, err := e.GetMarginSupportedCurrencyPairs(t.Context()); err != nil { t.Errorf("%s GetMarginSupportedCurrencyPair() error %v", e.Name, err) } } func TestGetMarginSupportedCurrencyPair(t *testing.T) { t.Parallel() if _, err := e.GetSingleMarginSupportedCurrencyPair(t.Context(), getPair(t, asset.Margin)); err != nil { t.Errorf("%s GetMarginSupportedCurrencyPair() error %v", e.Name, err) } } func TestGetOrderbookOfLendingLoans(t *testing.T) { t.Parallel() if _, err := e.GetOrderbookOfLendingLoans(t.Context(), currency.BTC); err != nil { t.Errorf("%s GetOrderbookOfLendingLoans() error %v", e.Name, err) } } func TestGetAllFutureContracts(t *testing.T) { t.Parallel() for _, c := range []currency.Code{currency.BTC, currency.USDT} { _, err := e.GetAllFutureContracts(t.Context(), c) assert.NoErrorf(t, err, "GetAllFutureContracts %s should not error", c) } } func TestGetFuturesContract(t *testing.T) { t.Parallel() _, err := e.GetFuturesContract(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures).String()) assert.NoError(t, err, "GetFuturesContract should not error") _, err = e.GetFuturesContract(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures).String()) assert.NoError(t, err, "GetFuturesContract should not error") } func TestGetFuturesOrderbook(t *testing.T) { t.Parallel() _, err := e.GetFuturesOrderbook(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures).String(), "", 10, false) assert.NoError(t, err, "GetFuturesOrderbook should not error for CoinMarginedFutures") _, err = e.GetFuturesOrderbook(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures).String(), "", 10, false) assert.NoError(t, err, "GetFuturesOrderbook should not error for USDTMarginedFutures") } func TestGetFuturesTradingHistory(t *testing.T) { t.Parallel() _, err := e.GetFuturesTradingHistory(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures), 0, 0, "", time.Time{}, time.Time{}) assert.NoError(t, err, "GetFuturesTradingHistory should not error for CoinMarginedFutures") _, err = e.GetFuturesTradingHistory(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures), 0, 0, "", time.Time{}, time.Time{}) assert.NoError(t, err, "GetFuturesTradingHistory should not error for USDTMarginedFutures") } func TestGetFuturesCandlesticks(t *testing.T) { t.Parallel() _, err := e.GetFuturesCandlesticks(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures).String(), time.Time{}, time.Time{}, 0, kline.OneWeek) assert.NoError(t, err, "GetFuturesCandlesticks should not error for CoinMarginedFutures") _, err = e.GetFuturesCandlesticks(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures).String(), time.Time{}, time.Time{}, 0, kline.OneWeek) assert.NoError(t, err, "GetFuturesCandlesticks should not error for USDTMarginedFutures") } func TestPremiumIndexKLine(t *testing.T) { t.Parallel() _, err := e.PremiumIndexKLine(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures), time.Time{}, time.Time{}, 0, kline.OneWeek) assert.NoError(t, err, "PremiumIndexKLine should not error for CoinMarginedFutures") _, err = e.PremiumIndexKLine(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures), time.Time{}, time.Time{}, 0, kline.OneWeek) assert.NoError(t, err, "PremiumIndexKLine should not error for USDTMarginedFutures") } func TestGetFutureTickers(t *testing.T) { t.Parallel() _, err := e.GetFuturesTickers(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures)) assert.NoError(t, err, "GetFutureTickers should not error for CoinMarginedFutures") _, err = e.GetFuturesTickers(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures)) assert.NoError(t, err, "GetFutureTickers should not error for USDTMarginedFutures") } func TestGetFutureFundingRates(t *testing.T) { t.Parallel() _, err := e.GetFutureFundingRates(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures), 0) assert.NoError(t, err, "GetFutureFundingRates should not error for CoinMarginedFutures") _, err = e.GetFutureFundingRates(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures), 0) assert.NoError(t, err, "GetFutureFundingRates should not error for USDTMarginedFutures") } func TestGetFuturesInsuranceBalanceHistory(t *testing.T) { t.Parallel() _, err := e.GetFuturesInsuranceBalanceHistory(t.Context(), currency.USDT, 0) assert.NoError(t, err, "GetFuturesInsuranceBalanceHistory should not error") } func TestGetFutureStats(t *testing.T) { t.Parallel() _, err := e.GetFutureStats(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures), time.Time{}, 0, 0) assert.NoError(t, err, "GetFutureStats should not error for CoinMarginedFutures") _, err = e.GetFutureStats(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures), time.Time{}, 0, 0) assert.NoError(t, err, "GetFutureStats should not error for USDTMarginedFutures") } func TestGetIndexConstituent(t *testing.T) { t.Parallel() _, err := e.GetIndexConstituent(t.Context(), currency.USDT, currency.Pair{Base: currency.BTC, Quote: currency.USDT, Delimiter: currency.UnderscoreDelimiter}.String()) assert.NoError(t, err, "GetIndexConstituent should not error") } func TestGetLiquidationHistory(t *testing.T) { t.Parallel() _, err := e.GetLiquidationHistory(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures), time.Time{}, time.Time{}, 0) assert.NoError(t, err, "GetLiquidationHistory should not error for CoinMarginedFutures") _, err = e.GetLiquidationHistory(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures), time.Time{}, time.Time{}, 0) assert.NoError(t, err, "GetLiquidationHistory should not error for USDTMarginedFutures") } func TestQueryFuturesAccount(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.QueryFuturesAccount(t.Context(), currency.USDT) assert.NoError(t, err, "QueryFuturesAccount should not error") } func TestGetFuturesAccountBooks(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetFuturesAccountBooks(t.Context(), currency.USDT, 0, time.Time{}, time.Time{}, "dnw") assert.NoError(t, err, "GetFuturesAccountBooks should not error") } func TestGetAllFuturesPositionsOfUsers(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetAllFuturesPositionsOfUsers(t.Context(), currency.USDT, true) assert.NoError(t, err, "GetAllPositionsOfUsers should not error") } func TestGetSinglePosition(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetSinglePosition(t.Context(), currency.USDT, currency.Pair{Quote: currency.BTC, Base: currency.USDT}) assert.NoError(t, err, "GetSinglePosition should not error") } func TestUpdateFuturesPositionMargin(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.UpdateFuturesPositionMargin(t.Context(), currency.BTC, 0.01, getPair(t, asset.CoinMarginedFutures)) assert.NoError(t, err, "UpdateFuturesPositionMargin should not error for CoinMarginedFutures") _, err = e.UpdateFuturesPositionMargin(t.Context(), currency.USDT, 0.01, getPair(t, asset.USDTMarginedFutures)) assert.NoError(t, err, "UpdateFuturesPositionMargin should not error for USDTMarginedFutures") } func TestUpdateFuturesPositionLeverage(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.UpdateFuturesPositionLeverage(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures), 1, 0) assert.NoError(t, err, "UpdateFuturesPositionLeverage should not error for CoinMarginedFutures") _, err = e.UpdateFuturesPositionLeverage(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures), 1, 0) assert.NoError(t, err, "UpdateFuturesPositionLeverage should not error for USDTMarginedFutures") } func TestUpdateFuturesPositionRiskLimit(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.UpdateFuturesPositionRiskLimit(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures), 10) assert.NoError(t, err, "UpdateFuturesPositionRiskLimit should not error for CoinMarginedFutures") _, err = e.UpdateFuturesPositionRiskLimit(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures), 10) assert.NoError(t, err, "UpdateFuturesPositionRiskLimit should not error for USDTMarginedFutures") } func TestPlaceDeliveryOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.PlaceDeliveryOrder(t.Context(), &ContractOrderCreateParams{ Contract: getPair(t, asset.DeliveryFutures), Size: 6024, Iceberg: 0, Price: "3765", Text: "t-my-custom-id", Settle: currency.USDT, TimeInForce: gtcTIF, }) assert.NoError(t, err, "CreateDeliveryOrder should not error") } func TestGetDeliveryOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetDeliveryOrders(t.Context(), getPair(t, asset.DeliveryFutures), statusOpen, currency.USDT, "", 0, 0, 1) assert.NoError(t, err, "GetDeliveryOrders should not error") } func TestCancelMultipleDeliveryOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.CancelMultipleDeliveryOrders(t.Context(), getPair(t, asset.DeliveryFutures), "ask", currency.USDT) assert.NoError(t, err, "CancelMultipleDeliveryOrders should not error") } func TestGetSingleDeliveryOrder(t *testing.T) { t.Parallel() _, err := e.GetSingleDeliveryOrder(t.Context(), currency.EMPTYCODE, "123456") assert.ErrorIs(t, err, errEmptyOrInvalidSettlementCurrency, "GetSingleDeliveryOrder should return errEmptyOrInvalidSettlementCurrency") sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err = e.GetSingleDeliveryOrder(t.Context(), currency.USDT, "123456") assert.NoError(t, err, "GetSingleDeliveryOrder should not error") } func TestCancelSingleDeliveryOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.CancelSingleDeliveryOrder(t.Context(), currency.USDT, "123456") assert.NoError(t, err, "CancelSingleDeliveryOrder should not error") } func TestGetMyDeliveryTradingHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetMyDeliveryTradingHistory(t.Context(), currency.USDT, "", getPair(t, asset.DeliveryFutures), 0, 0, 1, "") assert.NoError(t, err, "GetMyDeliveryTradingHistory should not error") } func TestGetDeliveryPositionCloseHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetDeliveryPositionCloseHistory(t.Context(), currency.USDT, getPair(t, asset.DeliveryFutures), 0, 0, time.Time{}, time.Time{}) assert.NoError(t, err, "GetDeliveryPositionCloseHistory should not error") } func TestGetDeliveryLiquidationHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetDeliveryLiquidationHistory(t.Context(), currency.USDT, getPair(t, asset.DeliveryFutures), 0, time.Now()) assert.NoError(t, err, "GetDeliveryLiquidationHistory should not error") } func TestGetDeliverySettlementHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetDeliverySettlementHistory(t.Context(), currency.USDT, getPair(t, asset.DeliveryFutures), 0, time.Now()) assert.NoError(t, err, "GetDeliverySettlementHistory should not error") } func TestGetDeliveryPriceTriggeredOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetDeliveryPriceTriggeredOrder(t.Context(), currency.USDT, &FuturesPriceTriggeredOrderParam{ Initial: FuturesInitial{ Price: 1234., Size: 12, Contract: getPair(t, asset.DeliveryFutures), }, Trigger: FuturesTrigger{ Rule: 1, OrderType: "close-short-position", Price: 123400, }, }) assert.NoError(t, err, "GetDeliveryPriceTriggeredOrder should not error") } func TestGetDeliveryAllAutoOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetDeliveryAllAutoOrder(t.Context(), statusOpen, currency.USDT, getPair(t, asset.DeliveryFutures), 0, 1) assert.NoError(t, err, "GetDeliveryAllAutoOrder should not error") } func TestCancelAllDeliveryPriceTriggeredOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.CancelAllDeliveryPriceTriggeredOrder(t.Context(), currency.USDT, getPair(t, asset.DeliveryFutures)) assert.NoError(t, err, "CancelAllDeliveryPriceTriggeredOrder should not error") } func TestGetSingleDeliveryPriceTriggeredOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetSingleDeliveryPriceTriggeredOrder(t.Context(), currency.USDT, "12345") assert.NoError(t, err, "GetSingleDeliveryPriceTriggeredOrder should not error") } func TestCancelDeliveryPriceTriggeredOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.CancelDeliveryPriceTriggeredOrder(t.Context(), currency.USDT, "12345") assert.NoError(t, err, "CancelDeliveryPriceTriggeredOrder should not error") } func TestEnableOrDisableDualMode(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.EnableOrDisableDualMode(t.Context(), currency.BTC, true) assert.NoError(t, err, "EnableOrDisableDualMode should not error") } func TestRetrivePositionDetailInDualMode(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.RetrivePositionDetailInDualMode(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures)) assert.NoError(t, err, "RetrivePositionDetailInDualMode should not error for CoinMarginedFutures") _, err = e.RetrivePositionDetailInDualMode(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures)) assert.NoError(t, err, "RetrivePositionDetailInDualMode should not error for USDTMarginedFutures") } func TestUpdatePositionMarginInDualMode(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.UpdatePositionMarginInDualMode(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures), 0.001, "dual_long") assert.NoError(t, err, "UpdatePositionMarginInDualMode should not error for CoinMarginedFutures") _, err = e.UpdatePositionMarginInDualMode(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures), 0.001, "dual_long") assert.NoError(t, err, "UpdatePositionMarginInDualMode should not error for USDTMarginedFutures") } func TestUpdatePositionLeverageInDualMode(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.UpdatePositionLeverageInDualMode(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures), 0.001, 0.001) assert.NoError(t, err, "UpdatePositionLeverageInDualMode should not error for CoinMarginedFutures") _, err = e.UpdatePositionLeverageInDualMode(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures), 0.001, 0.001) assert.NoError(t, err, "UpdatePositionLeverageInDualMode should not error for USDTMarginedFutures") } func TestUpdatePositionRiskLimitInDualMode(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.UpdatePositionRiskLimitInDualMode(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures), 10) assert.NoError(t, err, "UpdatePositionRiskLimitInDualMode should not error for CoinMarginedFutures") _, err = e.UpdatePositionRiskLimitInDualMode(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures), 10) assert.NoError(t, err, "UpdatePositionRiskLimitInDualMode should not error for USDTMarginedFutures") } func TestPlaceFuturesOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.PlaceFuturesOrder(t.Context(), &ContractOrderCreateParams{ Contract: getPair(t, asset.CoinMarginedFutures), Size: 6024, Iceberg: 0, Price: "3765", TimeInForce: "gtc", Text: "t-my-custom-id", Settle: currency.BTC, }) assert.NoError(t, err, "PlaceFuturesOrder should not error") } func TestGetFuturesOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetFuturesOrders(t.Context(), currency.NewBTCUSD(), statusOpen, "", currency.BTC, 0, 0, 1) assert.NoError(t, err, "GetFuturesOrders should not error") } func TestCancelMultipleFuturesOpenOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.CancelMultipleFuturesOpenOrders(t.Context(), getPair(t, asset.USDTMarginedFutures), "ask", currency.USDT) assert.NoError(t, err, "CancelMultipleFuturesOpenOrders should not error") } func TestGetSingleFuturesPriceTriggeredOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetSingleFuturesPriceTriggeredOrder(t.Context(), currency.BTC, "12345") assert.NoError(t, err, "GetSingleFuturesPriceTriggeredOrder should not error") } func TestCancelFuturesPriceTriggeredOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.CancelFuturesPriceTriggeredOrder(t.Context(), currency.USDT, "12345") assert.NoError(t, err, "CancelFuturesPriceTriggeredOrder should not error") } func TestPlaceBatchFuturesOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.PlaceBatchFuturesOrders(t.Context(), currency.BTC, []ContractOrderCreateParams{ { Contract: getPair(t, asset.CoinMarginedFutures), Size: 6024, Iceberg: 0, Price: "3765", TimeInForce: "gtc", Text: "t-my-custom-id", Settle: currency.BTC, }, { Contract: getPair(t, asset.CoinMarginedFutures), Size: 232, Iceberg: 0, Price: "376225", TimeInForce: "gtc", Text: "t-my-custom-id", Settle: currency.BTC, }, }) assert.NoError(t, err, "PlaceBatchFuturesOrders should not error") } func TestGetSingleFuturesOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetSingleFuturesOrder(t.Context(), currency.BTC, "12345") assert.NoError(t, err, "GetSingleFuturesOrder should not error") } func TestCancelSingleFuturesOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.CancelSingleFuturesOrder(t.Context(), currency.BTC, "12345") assert.NoError(t, err, "CancelSingleFuturesOrder should not error") } func TestAmendFuturesOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.AmendFuturesOrder(t.Context(), currency.BTC, "1234", AmendFuturesOrderParam{ Price: 12345.990, }) assert.NoError(t, err, "AmendFuturesOrder should not error") } func TestGetMyFuturesTradingHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetMyFuturesTradingHistory(t.Context(), currency.BTC, "", "", getPair(t, asset.CoinMarginedFutures), 0, 0, 0) assert.NoError(t, err, "GetMyFuturesTradingHistory should not error") } func TestGetFuturesPositionCloseHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetFuturesPositionCloseHistory(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures), 0, 0, time.Time{}, time.Time{}) assert.NoError(t, err, "GetFuturesPositionCloseHistory should not error") } func TestGetFuturesLiquidationHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetFuturesLiquidationHistory(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures), 0, time.Time{}) assert.NoError(t, err, "GetFuturesLiquidationHistory should not error") } func TestCountdownCancelOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.CountdownCancelOrders(t.Context(), currency.BTC, CountdownParams{ Timeout: 8, }) assert.NoError(t, err, "CountdownCancelOrders should not error") } func TestCreatePriceTriggeredFuturesOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) for _, tc := range []struct { c currency.Code a asset.Item }{ {currency.BTC, asset.CoinMarginedFutures}, {currency.USDT, asset.USDTMarginedFutures}, } { _, err := e.CreatePriceTriggeredFuturesOrder(t.Context(), tc.c, &FuturesPriceTriggeredOrderParam{ Initial: FuturesInitial{ Price: 1234., Size: 2, Contract: getPair(t, tc.a), }, Trigger: FuturesTrigger{ Rule: 1, OrderType: "close-short-position", }, }) assert.NoErrorf(t, err, "CreatePriceTriggeredFuturesOrder should not error for settlement currency: %s, asset: %s", tc.c, tc.a) } } func TestListAllFuturesAutoOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.ListAllFuturesAutoOrders(t.Context(), statusOpen, currency.BTC, currency.EMPTYPAIR, 0, 0) assert.NoError(t, err, "ListAllFuturesAutoOrders should not error") } func TestCancelAllFuturesOpenOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.CancelAllFuturesOpenOrders(t.Context(), currency.BTC, getPair(t, asset.CoinMarginedFutures)) assert.NoError(t, err, "CancelAllFuturesOpenOrders should not error for CoinMarginedFutures") _, err = e.CancelAllFuturesOpenOrders(t.Context(), currency.USDT, getPair(t, asset.USDTMarginedFutures)) assert.NoError(t, err, "CancelAllFuturesOpenOrders should not error for USDTMarginedFutures") } func TestGetAllDeliveryContracts(t *testing.T) { t.Parallel() r, err := e.GetAllDeliveryContracts(t.Context(), currency.USDT) require.NoError(t, err, "GetAllDeliveryContracts must not error") assert.NotEmpty(t, r, "GetAllDeliveryContracts should return data") r, err = e.GetAllDeliveryContracts(t.Context(), currency.BTC) require.NoError(t, err, "GetAllDeliveryContracts must not error") // The test below will fail if support for BTC settlement is added. This is intentional, as it ensures we are alerted when it's time to reintroduce support if !assert.Empty(t, r, "GetAllDeliveryContracts should not return any data with unsupported settlement currency BTC") { t.Error("BTC settlement for delivery futures appears to be supported again by the API. Please raise an issue to reintroduce BTC support for this exchange") } } func TestGetDeliveryContract(t *testing.T) { t.Parallel() _, err := e.GetDeliveryContract(t.Context(), currency.USDT, getPair(t, asset.DeliveryFutures)) assert.NoError(t, err, "GetDeliveryContract should not error") } func TestGetDeliveryOrderbook(t *testing.T) { t.Parallel() _, err := e.GetDeliveryOrderbook(t.Context(), currency.USDT, "0", getPair(t, asset.DeliveryFutures), 0, false) assert.NoError(t, err, "GetDeliveryOrderbook should not error") } func TestGetDeliveryTradingHistory(t *testing.T) { t.Parallel() _, err := e.GetDeliveryTradingHistory(t.Context(), currency.USDT, "", getPair(t, asset.DeliveryFutures), 0, time.Time{}, time.Time{}) assert.NoError(t, err, "GetDeliveryTradingHistory should not error") } func TestGetDeliveryFuturesCandlesticks(t *testing.T) { t.Parallel() _, err := e.GetDeliveryFuturesCandlesticks(t.Context(), currency.USDT, getPair(t, asset.DeliveryFutures), time.Time{}, time.Time{}, 0, kline.OneWeek) assert.NoError(t, err, "GetDeliveryFuturesCandlesticks should not error") } func TestGetDeliveryFutureTickers(t *testing.T) { t.Parallel() _, err := e.GetDeliveryFutureTickers(t.Context(), currency.USDT, getPair(t, asset.DeliveryFutures)) assert.NoError(t, err, "GetDeliveryFutureTickers should not error") } func TestGetDeliveryInsuranceBalanceHistory(t *testing.T) { t.Parallel() _, err := e.GetDeliveryInsuranceBalanceHistory(t.Context(), currency.BTC, 0) assert.NoError(t, err, "GetDeliveryInsuranceBalanceHistory should not error") } func TestQueryDeliveryFuturesAccounts(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetDeliveryFuturesAccounts(t.Context(), currency.USDT) assert.NoError(t, err, "GetDeliveryFuturesAccounts should not error") } func TestGetDeliveryAccountBooks(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetDeliveryAccountBooks(t.Context(), currency.USDT, 0, time.Time{}, time.Now(), "dnw") assert.NoError(t, err, "GetDeliveryAccountBooks should not error") } func TestGetAllDeliveryPositionsOfUser(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetAllDeliveryPositionsOfUser(t.Context(), currency.USDT) assert.NoError(t, err, "GetAllDeliveryPositionsOfUser should not error") } func TestGetSingleDeliveryPosition(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetSingleDeliveryPosition(t.Context(), currency.USDT, getPair(t, asset.DeliveryFutures)) assert.NoError(t, err, "GetSingleDeliveryPosition should not error") } func TestUpdateDeliveryPositionMargin(t *testing.T) { t.Parallel() _, err := e.UpdateDeliveryPositionMargin(t.Context(), currency.EMPTYCODE, 0.001, currency.Pair{}) assert.ErrorIs(t, err, errEmptyOrInvalidSettlementCurrency) sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err = e.UpdateDeliveryPositionMargin(t.Context(), currency.USDT, 0.001, getPair(t, asset.DeliveryFutures)) assert.NoError(t, err, "UpdateDeliveryPositionMargin should not error") } func TestUpdateDeliveryPositionLeverage(t *testing.T) { t.Parallel() _, err := e.UpdateDeliveryPositionLeverage(t.Context(), currency.EMPTYCODE, currency.Pair{}, 0.001) assert.ErrorIs(t, err, errEmptyOrInvalidSettlementCurrency) sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err = e.UpdateDeliveryPositionLeverage(t.Context(), currency.USDT, getPair(t, asset.DeliveryFutures), 0.001) assert.NoError(t, err, "UpdateDeliveryPositionLeverage should not error") } func TestUpdateDeliveryPositionRiskLimit(t *testing.T) { t.Parallel() _, err := e.UpdateDeliveryPositionRiskLimit(t.Context(), currency.EMPTYCODE, currency.Pair{}, 0) assert.ErrorIs(t, err, errEmptyOrInvalidSettlementCurrency) sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err = e.UpdateDeliveryPositionRiskLimit(t.Context(), currency.USDT, getPair(t, asset.DeliveryFutures), 30) assert.NoError(t, err, "UpdateDeliveryPositionRiskLimit should not error") } func TestGetAllOptionsUnderlyings(t *testing.T) { t.Parallel() if _, err := e.GetAllOptionsUnderlyings(t.Context()); err != nil { t.Errorf("%s GetAllOptionsUnderlyings() error %v", e.Name, err) } } func TestGetExpirationTime(t *testing.T) { t.Parallel() _, err := e.GetExpirationTime(t.Context(), "") assert.ErrorIs(t, err, errInvalidUnderlying) _, err = e.GetExpirationTime(t.Context(), "BTC_USDT") assert.NoError(t, err, "GetExpirationTime should not error") } func TestGetAllContractOfUnderlyingWithinExpiryDate(t *testing.T) { t.Parallel() if _, err := e.GetAllContractOfUnderlyingWithinExpiryDate(t.Context(), "BTC_USDT", time.Time{}); err != nil { t.Errorf("%s GetAllContractOfUnderlyingWithinExpiryDate() error %v", e.Name, err) } } func TestGetOptionsSpecifiedContractDetail(t *testing.T) { t.Parallel() if _, err := e.GetOptionsSpecifiedContractDetail(t.Context(), getPair(t, asset.Options)); err != nil { t.Errorf("%s GetOptionsSpecifiedContractDetail() error %v", e.Name, err) } } func TestGetSettlementHistory(t *testing.T) { t.Parallel() if _, err := e.GetSettlementHistory(t.Context(), "BTC_USDT", 0, 0, time.Time{}, time.Time{}); err != nil { t.Errorf("%s GetSettlementHistory() error %v", e.Name, err) } } func TestGetOptionsSpecifiedSettlementHistory(t *testing.T) { t.Parallel() underlying := "BTC_USDT" optionsSettlement, err := e.GetSettlementHistory(t.Context(), underlying, 0, 1, time.Time{}, time.Time{}) if err != nil { t.Fatal(err) } cp, err := currency.NewPairFromString(optionsSettlement[0].Contract) if err != nil { t.Fatal(err) } if _, err := e.GetOptionsSpecifiedContractsSettlement(t.Context(), cp, underlying, optionsSettlement[0].Timestamp.Time().Unix()); err != nil { t.Errorf("%s GetOptionsSpecifiedContractsSettlement() error %s", e.Name, err) } } func TestGetSupportedFlashSwapCurrencies(t *testing.T) { t.Parallel() if _, err := e.GetSupportedFlashSwapCurrencies(t.Context()); err != nil { t.Errorf("%s GetSupportedFlashSwapCurrencies() error %v", e.Name, err) } } const flashSwapOrderResponseJSON = `{"id": 54646, "create_time": 1651116876378, "update_time": 1651116876378, "user_id": 11135567, "sell_currency": "BTC", "sell_amount": "0.01", "buy_currency": "USDT", "buy_amount": "10", "price": "100", "status": 1}` func TestCreateFlashSwapOrder(t *testing.T) { t.Parallel() var response FlashSwapOrderResponse if err := json.Unmarshal([]byte(flashSwapOrderResponseJSON), &response); err != nil { t.Errorf("%s error while deserializing to FlashSwapOrderResponse %v", e.Name, err) } sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CreateFlashSwapOrder(t.Context(), FlashSwapOrderParams{ PreviewID: "1234", SellCurrency: currency.USDT, BuyCurrency: currency.BTC, BuyAmount: 34234, SellAmount: 34234, }); err != nil { t.Errorf("%s CreateFlashSwapOrder() error %v", e.Name, err) } } func TestGetAllFlashSwapOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetAllFlashSwapOrders(t.Context(), 1, currency.EMPTYCODE, currency.EMPTYCODE, true, 0, 0); err != nil { t.Errorf("%s GetAllFlashSwapOrders() error %v", e.Name, err) } } func TestGetSingleFlashSwapOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetSingleFlashSwapOrder(t.Context(), "1234"); err != nil { t.Errorf("%s GetSingleFlashSwapOrder() error %v", e.Name, err) } } func TestInitiateFlashSwapOrderReview(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.InitiateFlashSwapOrderReview(t.Context(), FlashSwapOrderParams{ PreviewID: "1234", SellCurrency: currency.USDT, BuyCurrency: currency.BTC, SellAmount: 100, }); err != nil { t.Errorf("%s InitiateFlashSwapOrderReview() error %v", e.Name, err) } } func TestGetMyOptionsSettlements(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetMyOptionsSettlements(t.Context(), "BTC_USDT", currency.EMPTYPAIR, 0, 0, time.Time{}); err != nil { t.Errorf("%s GetMyOptionsSettlements() error %v", e.Name, err) } } func TestGetOptionAccounts(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetOptionAccounts(t.Context()); err != nil { t.Errorf("%s GetOptionAccounts() error %v", e.Name, err) } } func TestGetAccountChangingHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetAccountChangingHistory(t.Context(), 0, 0, time.Time{}, time.Time{}, ""); err != nil { t.Errorf("%s GetAccountChangingHistory() error %v", e.Name, err) } } func TestGetUsersPositionSpecifiedUnderlying(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetUsersPositionSpecifiedUnderlying(t.Context(), ""); err != nil { t.Errorf("%s GetUsersPositionSpecifiedUnderlying() error %v", e.Name, err) } } func TestGetSpecifiedContractPosition(t *testing.T) { t.Parallel() _, err := e.GetSpecifiedContractPosition(t.Context(), currency.EMPTYPAIR) assert.ErrorIs(t, err, errInvalidOrMissingContractParam) sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err = e.GetSpecifiedContractPosition(t.Context(), getPair(t, asset.Options)) assert.NoError(t, err, "GetSpecifiedContractPosition should not error") } func TestGetUsersLiquidationHistoryForSpecifiedUnderlying(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetUsersLiquidationHistoryForSpecifiedUnderlying(t.Context(), "BTC_USDT", currency.EMPTYPAIR); err != nil { t.Errorf("%s GetUsersLiquidationHistoryForSpecifiedUnderlying() error %v", e.Name, err) } } func TestPlaceOptionOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err := e.PlaceOptionOrder(t.Context(), &OptionOrderParam{ Contract: getPair(t, asset.Options).String(), OrderSize: -1, Iceberg: 0, Text: "-", TimeInForce: "gtc", Price: 100, }) if err != nil { t.Errorf("%s PlaceOptionOrder() error %v", e.Name, err) } } func TestGetOptionFuturesOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetOptionFuturesOrders(t.Context(), currency.EMPTYPAIR, "", "", 0, 0, time.Time{}, time.Time{}); err != nil { t.Errorf("%s GetOptionFuturesOrders() error %v", e.Name, err) } } func TestCancelOptionOpenOrders(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CancelMultipleOptionOpenOrders(t.Context(), getPair(t, asset.Options), "", ""); err != nil { t.Errorf("%s CancelOptionOpenOrders() error %v", e.Name, err) } } func TestGetSingleOptionOrder(t *testing.T) { t.Parallel() _, err := e.GetSingleOptionOrder(t.Context(), "") assert.ErrorIs(t, err, errInvalidOrderID) sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err = e.GetSingleOptionOrder(t.Context(), "1234") assert.NoError(t, err, "GetSingleOptionOrder should not error") } func TestCancelSingleOrder(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CancelOptionSingleOrder(t.Context(), "1234"); err != nil { t.Errorf("%s CancelSingleOrder() error %v", e.Name, err) } } func TestGetMyOptionsTradingHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetMyOptionsTradingHistory(t.Context(), "BTC_USDT", currency.EMPTYPAIR, 0, 0, time.Time{}, time.Time{}) require.NoError(t, err) } func TestWithdrawCurrency(t *testing.T) { t.Parallel() _, err := e.WithdrawCurrency(t.Context(), WithdrawalRequestParam{}) assert.ErrorIs(t, err, errInvalidAmount) sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) _, err = e.WithdrawCurrency(t.Context(), WithdrawalRequestParam{ Currency: currency.BTC, Amount: 0.00000001, Chain: "BTC", Address: core.BitcoinDonationAddress, }) if err != nil { t.Errorf("%s WithdrawCurrency() expecting error %v, but found %v", e.Name, errInvalidAmount, err) } } func TestCancelWithdrawalWithSpecifiedID(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CancelWithdrawalWithSpecifiedID(t.Context(), "1234567"); err != nil { t.Errorf("%s CancelWithdrawalWithSpecifiedID() error %v", e.Name, err) } } func TestGetOptionsOrderbook(t *testing.T) { t.Parallel() _, err := e.GetOptionsOrderbook(t.Context(), getPair(t, asset.Options), "0.1", 9, true) assert.NoError(t, err, "GetOptionsOrderbook should not error") } func TestGetOptionsTickers(t *testing.T) { t.Parallel() if _, err := e.GetOptionsTickers(t.Context(), "BTC_USDT"); err != nil { t.Errorf("%s GetOptionsTickers() error %v", e.Name, err) } } func TestGetOptionUnderlyingTickers(t *testing.T) { t.Parallel() if _, err := e.GetOptionUnderlyingTickers(t.Context(), "BTC_USDT"); err != nil { t.Errorf("%s GetOptionUnderlyingTickers() error %v", e.Name, err) } } func TestGetOptionFuturesCandlesticks(t *testing.T) { t.Parallel() if _, err := e.GetOptionFuturesCandlesticks(t.Context(), getPair(t, asset.Options), 0, time.Now().Add(-time.Hour*10), time.Time{}, kline.ThirtyMin); err != nil { t.Error(err) } } func TestGetOptionFuturesMarkPriceCandlesticks(t *testing.T) { t.Parallel() if _, err := e.GetOptionFuturesMarkPriceCandlesticks(t.Context(), "BTC_USDT", 0, time.Time{}, time.Time{}, kline.OneMonth); err != nil { t.Errorf("%s GetOptionFuturesMarkPriceCandlesticks() error %v", e.Name, err) } } func TestGetOptionsTradeHistory(t *testing.T) { t.Parallel() if _, err := e.GetOptionsTradeHistory(t.Context(), getPair(t, asset.Options), "C", 0, 0, time.Time{}, time.Time{}); err != nil { t.Errorf("%s GetOptionsTradeHistory() error %v", e.Name, err) } } // Sub-account endpoints func TestCreateNewSubAccount(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CreateNewSubAccount(t.Context(), SubAccountParams{ LoginName: "Sub_Account_for_testing", }); err != nil { t.Errorf("%s CreateNewSubAccount() error %v", e.Name, err) } } func TestGetSubAccounts(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetSubAccounts(t.Context()); err != nil { t.Errorf("%s GetSubAccounts() error %v", e.Name, err) } } func TestGetSingleSubAccount(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetSingleSubAccount(t.Context(), "123423"); err != nil { t.Errorf("%s GetSingleSubAccount() error %v", e.Name, err) } } // Wrapper test functions func TestFetchTradablePairs(t *testing.T) { t.Parallel() for _, a := range e.GetAssetTypes(false) { pairs, err := e.FetchTradablePairs(t.Context(), a) require.NoErrorf(t, err, "FetchTradablePairs must not error for %s", a) require.NotEmptyf(t, pairs, "FetchTradablePairs must return some pairs for %s", a) if a == asset.USDTMarginedFutures || a == asset.CoinMarginedFutures { for _, p := range pairs { _, err := getSettlementCurrency(p, a) require.NoErrorf(t, err, "Fetched pair %s %s must not error on getSettlementCurrency", a, p) } } } } func TestUpdateTickers(t *testing.T) { t.Parallel() for _, a := range e.GetAssetTypes(false) { err := e.UpdateTickers(t.Context(), a) assert.NoErrorf(t, err, "UpdateTickers should not error for %s", a) } } func TestUpdateOrderbook(t *testing.T) { t.Parallel() for _, a := range e.GetAssetTypes(false) { pair := getPair(t, a) t.Run(a.String()+" "+pair.String(), func(t *testing.T) { t.Parallel() o, err := e.UpdateOrderbook(t.Context(), pair, a) require.NoError(t, err) if a != asset.Options { // Options orderbooks can be empty assert.NotEmpty(t, o.Bids) assert.NotEmpty(t, o.Asks) } }) } } func TestGetWithdrawalsHistory(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if _, err := e.GetWithdrawalsHistory(t.Context(), currency.BTC, asset.Empty); err != nil { t.Errorf("%s GetWithdrawalsHistory() error %v", e.Name, err) } } func TestGetRecentTrades(t *testing.T) { t.Parallel() for _, a := range e.GetAssetTypes(false) { if a != asset.CoinMarginedFutures { _, err := e.GetRecentTrades(t.Context(), getPair(t, a), a) assert.NoErrorf(t, err, "GetRecentTrades should not error for %s", a) } } } func TestSubmitOrder(t *testing.T) { sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) for _, a := range e.GetAssetTypes(false) { _, err := e.SubmitOrder(t.Context(), &order.Submit{ Exchange: e.Name, Pair: getPair(t, a), Side: order.Buy, Type: order.Limit, Price: 1, Amount: 1, AssetType: a, TimeInForce: order.GoodTillCancel, }) assert.NoErrorf(t, err, "SubmitOrder should not error for %s", a) } } func TestCancelExchangeOrder(t *testing.T) { sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) for _, a := range e.GetAssetTypes(false) { orderCancellation := &order.Cancel{ OrderID: "1", AccountID: "1", Pair: getPair(t, a), AssetType: a, } err := e.CancelOrder(t.Context(), orderCancellation) assert.NoErrorf(t, err, "CancelOrder should not error for %s", a) } } func TestCancelBatchOrders(t *testing.T) { sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) for _, a := range e.GetAssetTypes(false) { _, err := e.CancelBatchOrders(t.Context(), []order.Cancel{ { OrderID: "1", AccountID: "1", Pair: getPair(t, a), AssetType: a, }, { OrderID: "2", AccountID: "1", Pair: getPair(t, a), AssetType: a, }, }) assert.NoErrorf(t, err, "CancelBatchOrders should not error for %s", a) } } func TestGetDepositAddress(t *testing.T) { sharedtestvalues.SkipTestIfCredentialsUnset(t, e) chains, err := e.GetAvailableTransferChains(t.Context(), currency.BTC) if err != nil { t.Fatal(err) } for i := range chains { _, err = e.GetDepositAddress(t.Context(), currency.BTC, "", chains[i]) if err != nil { t.Error("Test Fail - GetDepositAddress error", err) } } } func TestGetActiveOrders(t *testing.T) { sharedtestvalues.SkipTestIfCredentialsUnset(t, e) for _, a := range e.GetAssetTypes(false) { enabledPairs := getPairs(t, a) if len(enabledPairs) > 2 { enabledPairs = enabledPairs[:2] } _, err := e.GetActiveOrders(t.Context(), &order.MultiOrderRequest{ Pairs: enabledPairs, Type: order.AnyType, Side: order.AnySide, AssetType: a, }) assert.NoErrorf(t, err, "GetActiveOrders should not error for %s", a) } } func TestGetOrderHistory(t *testing.T) { sharedtestvalues.SkipTestIfCredentialsUnset(t, e) for _, a := range e.GetAssetTypes(false) { enabledPairs := getPairs(t, a) if len(enabledPairs) > 4 { enabledPairs = enabledPairs[:4] } multiOrderRequest := order.MultiOrderRequest{ Type: order.AnyType, Side: order.Buy, Pairs: enabledPairs, AssetType: a, } _, err := e.GetOrderHistory(t.Context(), &multiOrderRequest) assert.NoErrorf(t, err, "GetOrderHistory should not error for %s", a) } } func TestGetHistoricCandles(t *testing.T) { t.Parallel() startTime := time.Now().Add(-time.Hour * 10) for _, a := range e.GetAssetTypes(false) { _, err := e.GetHistoricCandles(t.Context(), getPair(t, a), a, kline.OneDay, startTime, time.Now()) if a == asset.Options { assert.ErrorIs(t, err, asset.ErrNotSupported, "GetHistoricCandles should error correctly for options") } else { assert.NoErrorf(t, err, "GetHistoricCandles should not error for %s", a) } } } func TestGetHistoricCandlesExtended(t *testing.T) { t.Parallel() startTime := time.Now().Add(-time.Hour * 5) for _, a := range e.GetAssetTypes(false) { _, err := e.GetHistoricCandlesExtended(t.Context(), getPair(t, a), a, kline.OneMin, startTime, time.Now()) if a == asset.Options { assert.ErrorIs(t, err, asset.ErrNotSupported, "GetHistoricCandlesExtended should error correctly for options") } else { assert.NoErrorf(t, err, "GetHistoricCandlesExtended should not error for %s", a) } } } func TestGetAvailableTransferTrains(t *testing.T) { t.Parallel() _, err := e.GetAvailableTransferChains(t.Context(), currency.USDT) if err != nil { t.Error(err) } } func TestGetUnderlyingFromCurrencyPair(t *testing.T) { t.Parallel() if uly, err := e.GetUnderlyingFromCurrencyPair(currency.Pair{Delimiter: currency.UnderscoreDelimiter, Base: currency.BTC, Quote: currency.NewCode("USDT_LLK")}); err != nil { t.Error(err) } else if !uly.Equal(currency.NewBTCUSDT()) { t.Error("unexpected underlying") } } const wsTickerPushDataJSON = `{"time": 1606291803, "channel": "spot.tickers", "event": "update", "result": { "currency_pair": "BTC_USDT", "last": "19106.55", "lowest_ask": "19108.71", "highest_bid": "19106.55", "change_percentage": "3.66", "base_volume": "2811.3042155865", "quote_volume": "53441606.52411221454674732293", "high_24h": "19417.74", "low_24h": "18434.21" }}` func TestWsTickerPushData(t *testing.T) { t.Parallel() if err := e.WsHandleSpotData(t.Context(), nil, []byte(wsTickerPushDataJSON)); err != nil { t.Errorf("%s websocket ticker push data error: %v", e.Name, err) } } const wsTradePushDataJSON = `{ "time": 1606292218, "channel": "spot.trades", "event": "update", "result": { "id": 309143071, "create_time": 1606292218, "create_time_ms": "1606292218213.4578", "side": "sell", "currency_pair": "BTC_USDT", "amount": "16.4700000000", "price": "0.4705000000"}}` func TestWsTradePushData(t *testing.T) { t.Parallel() if err := e.WsHandleSpotData(t.Context(), nil, []byte(wsTradePushDataJSON)); err != nil { t.Errorf("%s websocket trade push data error: %v", e.Name, err) } } const wsCandlestickPushDataJSON = `{"time": 1606292600, "channel": "spot.candlesticks", "event": "update", "result": { "t": "1606292580", "v": "2362.32035", "c": "19128.1", "h": "19128.1", "l": "19128.1", "o": "19128.1","n": "1m_BTC_USDT"}}` func TestWsCandlestickPushData(t *testing.T) { t.Parallel() if err := e.WsHandleSpotData(t.Context(), nil, []byte(wsCandlestickPushDataJSON)); err != nil { t.Errorf("%s websocket candlestick push data error: %v", e.Name, err) } } const wsOrderbookTickerJSON = `{"time": 1606293275, "channel": "spot.book_ticker", "event": "update", "result": { "t": 1606293275123, "u": 48733182, "s": "BTC_USDT", "b": "19177.79", "B": "0.0003341504", "a": "19179.38", "A": "0.09" }}` func TestWsOrderbookTickerPushData(t *testing.T) { t.Parallel() if err := e.WsHandleSpotData(t.Context(), nil, []byte(wsOrderbookTickerJSON)); err != nil { t.Errorf("%s websocket orderbook push data error: %v", e.Name, err) } } const ( wsOrderbookUpdatePushDataJSON = `{"time": 1606294781, "channel": "spot.order_book_update", "event": "update", "result": { "t": 1606294781123, "e": "depthUpdate", "E": 1606294781,"s": "BTC_USDT","U": 48776301,"u": 48776306,"b": [["19137.74","0.0001"],["19088.37","0"]],"a": [["19137.75","0.6135"]] }}` wsOrderbookSnapshotPushDataJSON = `{"time":1606295412,"channel": "spot.order_book", "event": "update", "result": { "t": 1606295412123, "lastUpdateId": 48791820, "s": "BTC_USDT", "bids": [ [ "19079.55", "0.0195" ], [ "19079.07", "0.7341"],["19076.23", "0.00011808" ], [ "19073.9", "0.105" ], [ "19068.83", "0.1009" ] ], "asks": [ [ "19080.24", "0.1638" ], [ "19080.91","0.1366"],["19080.92","0.01"],["19081.29","0.01"],["19083.8","0.097"]]}}` ) func TestWsOrderbookSnapshotPushData(t *testing.T) { t.Parallel() err := e.WsHandleSpotData(t.Context(), nil, []byte(wsOrderbookSnapshotPushDataJSON)) if err != nil { t.Errorf("%s websocket orderbook snapshot push data error: %v", e.Name, err) } if err = e.WsHandleSpotData(t.Context(), nil, []byte(wsOrderbookUpdatePushDataJSON)); err != nil { t.Errorf("%s websocket orderbook update push data error: %v", e.Name, err) } } const wsSpotOrderPushDataJSON = `{"time": 1605175506, "channel": "spot.orders", "event": "update", "result": [ { "id": "30784435", "user": 123456, "text": "t-abc", "create_time": "1605175506", "create_time_ms": "1605175506123", "update_time": "1605175506", "update_time_ms": "1605175506123", "event": "put", "currency_pair": "BTC_USDT", "type": "limit", "account": "spot", "side": "sell", "amount": "1", "price": "10001", "time_in_force": "gtc", "left": "1", "filled_total": "0", "fee": "0", "fee_currency": "USDT", "point_fee": "0", "gt_fee": "0", "gt_discount": true, "rebated_fee": "0", "rebated_fee_currency": "USDT"} ]}` func TestWsPushOrders(t *testing.T) { t.Parallel() if err := e.WsHandleSpotData(t.Context(), nil, []byte(wsSpotOrderPushDataJSON)); err != nil { t.Errorf("%s websocket orders push data error: %v", e.Name, err) } } const wsUserTradePushDataJSON = `{"time": 1605176741, "channel": "spot.usertrades", "event": "update", "result": [ { "id": 5736713, "user_id": 1000001, "order_id": "30784428", "currency_pair": "BTC_USDT", "create_time": 1605176741, "create_time_ms": "1605176741123.456", "side": "sell", "amount": "1.00000000", "role": "taker", "price": "10000.00000000", "fee": "0.00200000000000", "point_fee": "0", "gt_fee": "0", "text": "apiv4" } ]}` func TestWsUserTradesPushDataJSON(t *testing.T) { t.Parallel() if err := e.WsHandleSpotData(t.Context(), nil, []byte(wsUserTradePushDataJSON)); err != nil { t.Errorf("%s websocket users trade push data error: %v", e.Name, err) } } const wsBalancesPushDataJSON = `{"time": 1605248616, "channel": "spot.balances", "event": "update", "result": [ { "timestamp": "1605248616", "timestamp_ms": "1605248616123", "user": "1000001", "currency": "USDT", "change": "100", "total": "1032951.325075926", "available": "1022943.325075926"} ]}` func TestBalancesPushData(t *testing.T) { t.Parallel() ctx := account.DeployCredentialsToContext(t.Context(), &account.Credentials{Key: "test", Secret: "test"}) if err := e.WsHandleSpotData(ctx, nil, []byte(wsBalancesPushDataJSON)); err != nil { t.Errorf("%s websocket balances push data error: %v", e.Name, err) } } const wsMarginBalancePushDataJSON = `{"time": 1605248616, "channel": "spot.funding_balances", "event": "update", "result": [ {"timestamp": "1605248616","timestamp_ms": "1605248616123","user": "1000001","currency": "USDT","change": "100","freeze": "100","lent": "0"} ]}` func TestMarginBalancePushData(t *testing.T) { t.Parallel() if err := e.WsHandleSpotData(t.Context(), nil, []byte(wsMarginBalancePushDataJSON)); err != nil { t.Errorf("%s websocket margin balance push data error: %v", e.Name, err) } } const wsCrossMarginBalancePushDataJSON = `{"time": 1605248616,"channel": "spot.cross_balances","event": "update", "result": [{"timestamp": "1605248616","timestamp_ms": "1605248616123","user": "1000001","currency": "USDT", "change": "100","total": "1032951.325075926","available": "1022943.325075926"}]}` func TestCrossMarginBalancePushData(t *testing.T) { t.Parallel() ctx := account.DeployCredentialsToContext(t.Context(), &account.Credentials{Key: "test", Secret: "test"}) if err := e.WsHandleSpotData(ctx, nil, []byte(wsCrossMarginBalancePushDataJSON)); err != nil { t.Errorf("%s websocket cross margin balance push data error: %v", e.Name, err) } } const wsCrossMarginBalanceLoan = `{ "time":1658289372, "channel":"spot.cross_loan", "event":"update", "result":{ "timestamp":1658289372338, "user":"1000001", "currency":"BTC", "change":"0.01", "total":"4.992341029566", "available":"0.078054772536", "borrowed":"0.01", "interest":"0.00001375" }}` func TestCrossMarginBalanceLoan(t *testing.T) { t.Parallel() if err := e.WsHandleSpotData(t.Context(), nil, []byte(wsCrossMarginBalanceLoan)); err != nil { t.Errorf("%s websocket cross margin loan push data error: %v", e.Name, err) } } // TestFuturesDataHandler ensures that messages from various futures channels do not error func TestFuturesDataHandler(t *testing.T) { t.Parallel() e := new(Exchange) //nolint:govet // Intentional shadow require.NoError(t, testexch.Setup(e), "Test instance Setup must not error") testexch.FixtureToDataHandler(t, "testdata/wsFutures.json", func(ctx context.Context, m []byte) error { if strings.Contains(string(m), "futures.balances") { ctx = account.DeployCredentialsToContext(ctx, &account.Credentials{Key: "test", Secret: "test"}) } return e.WsHandleFuturesData(ctx, nil, m, asset.CoinMarginedFutures) }) close(e.Websocket.DataHandler) assert.Len(t, e.Websocket.DataHandler, 14, "Should see the correct number of messages") for resp := range e.Websocket.DataHandler { if err, isErr := resp.(error); isErr { assert.NoError(t, err, "Should not get any errors down the data handler") } } } // ******************************************** Options web-socket unit test funcs ******************** const optionsContractTickerPushDataJSON = `{"time": 1630576352, "channel": "options.contract_tickers", "event": "update", "result": { "name": "BTC_USDT-20211231-59800-P", "last_price": "11349.5", "mark_price": "11170.19", "index_price": "", "position_size": 993, "bid1_price": "10611.7", "bid1_size": 100, "ask1_price": "11728.7", "ask1_size": 100, "vega": "34.8731", "theta": "-72.80588", "rho": "-28.53331", "gamma": "0.00003", "delta": "-0.78311", "mark_iv": "0.86695", "bid_iv": "0.65481", "ask_iv": "0.88145", "leverage": "3.5541112718136" }}` func TestOptionsContractTickerPushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsContractTickerPushDataJSON)); err != nil { t.Errorf("%s websocket options contract ticker push data failed with error %v", e.Name, err) } } const optionsUnderlyingTickerPushDataJSON = `{"time": 1630576352, "channel": "options.ul_tickers", "event": "update", "result": { "trade_put": 800, "trade_call": 41700, "index_price": "50695.43", "name": "BTC_USDT" }}` func TestOptionsUnderlyingTickerPushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsUnderlyingTickerPushDataJSON)); err != nil { t.Errorf("%s websocket options underlying ticker push data error: %v", e.Name, err) } } const optionsContractTradesPushDataJSON = `{"time": 1630576356, "channel": "options.trades", "event": "update", "result": [ { "contract": "BTC_USDT-20211231-59800-C", "create_time": 1639144526, "id": 12279, "price": 997.8, "size": -100, "create_time_ms": 1639144526597, "underlying": "BTC_USDT" } ]}` func TestOptionsContractTradesPushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsContractTradesPushDataJSON)); err != nil { t.Errorf("%s websocket contract trades push data error: %v", e.Name, err) } } const optionsUnderlyingTradesPushDataJSON = `{"time": 1630576356, "channel": "options.ul_trades", "event": "update", "result": [{"contract": "BTC_USDT-20211231-59800-C","create_time": 1639144526,"id": 12279,"price": 997.8,"size": -100,"create_time_ms": 1639144526597,"underlying": "BTC_USDT","is_call": true} ]}` func TestOptionsUnderlyingTradesPushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsUnderlyingTradesPushDataJSON)); err != nil { t.Errorf("%s websocket underlying trades push data error: %v", e.Name, err) } } const optionsUnderlyingPricePushDataJSON = `{ "time": 1630576356, "channel": "options.ul_price", "event": "update", "result": { "underlying": "BTC_USDT", "price": 49653.24,"time": 1639143988,"time_ms": 1639143988931}}` func TestOptionsUnderlyingPricePushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsUnderlyingPricePushDataJSON)); err != nil { t.Errorf("%s websocket underlying price push data error: %v", e.Name, err) } } const optionsMarkPricePushDataJSON = `{ "time": 1630576356, "channel": "options.mark_price", "event": "update", "result": { "contract": "BTC_USDT-20211231-59800-P", "price": 11021.27, "time": 1639143401, "time_ms": 1639143401676}}` func TestOptionsMarkPricePushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsMarkPricePushDataJSON)); err != nil { t.Errorf("%s websocket mark price push data error: %v", e.Name, err) } } const optionsSettlementsPushDataJSON = `{ "time": 1630576356, "channel": "options.settlements", "event": "update", "result": { "contract": "BTC_USDT-20211130-55000-P", "orderbook_id": 2, "position_size": 1, "profit": 0.5, "settle_price": 70000, "strike_price": 65000, "tag": "WEEK", "trade_id": 1, "trade_size": 1, "underlying": "BTC_USDT", "time": 1639051907, "time_ms": 1639051907000}}` func TestSettlementsPushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsSettlementsPushDataJSON)); err != nil { t.Errorf("%s websocket options settlements push data error: %v", e.Name, err) } } const optionsContractPushDataJSON = `{"time": 1630576356, "channel": "options.contracts", "event": "update", "result": { "contract": "BTC_USDT-20211130-50000-P", "create_time": 1637917026, "expiration_time": 1638230400, "init_margin_high": 0.15, "init_margin_low": 0.1, "is_call": false, "maint_margin_base": 0.075, "maker_fee_rate": 0.0004, "mark_price_round": 0.1, "min_balance_short": 0.5, "min_order_margin": 0.1, "multiplier": 0.0001, "order_price_deviate": 0, "order_price_round": 0.1, "order_size_max": 1, "order_size_min": 10, "orders_limit": 100000, "ref_discount_rate": 0.1, "ref_rebate_rate": 0, "strike_price": 50000, "tag": "WEEK", "taker_fee_rate": 0.0004, "underlying": "BTC_USDT", "time": 1639051907, "time_ms": 1639051907000}}` func TestOptionsContractPushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsContractPushDataJSON)); err != nil { t.Errorf("%s websocket options contracts push data error: %v", e.Name, err) } } const ( optionsContractCandlesticksPushDataJSON = `{ "time": 1630650451, "channel": "options.contract_candlesticks", "event": "update", "result": [ { "t": 1639039260, "v": 100, "c": "1041.4", "h": "1041.4", "l": "1041.4", "o": "1041.4", "a": "0", "n": "10s_BTC_USDT-20211231-59800-C" } ]}` optionsUnderlyingCandlesticksPushDataJSON = `{ "time": 1630650451, "channel": "options.ul_candlesticks", "event": "update", "result": [ { "t": 1639039260, "v": 100, "c": "1041.4", "h": "1041.4", "l": "1041.4", "o": "1041.4", "a": "0", "n": "10s_BTC_USDT" } ]}` ) func TestOptionsCandlesticksPushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsContractCandlesticksPushDataJSON)); err != nil { t.Errorf("%s websocket options contracts candlestick push data error: %v", e.Name, err) } if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsUnderlyingCandlesticksPushDataJSON)); err != nil { t.Errorf("%s websocket options underlying candlestick push data error: %v", e.Name, err) } } const ( optionsOrderbookTickerPushDataJSON = `{ "time": 1630650452, "channel": "options.book_ticker", "event": "update", "result": { "t": 1615366379123, "u": 2517661076, "s": "BTC_USDT-20211130-50000-C", "b": "54696.6", "B": 37000, "a": "54696.7", "A": 47061 }}` optionsOrderbookUpdatePushDataJSON = `{ "time": 1630650445, "channel": "options.order_book_update", "event": "update", "result": { "t": 1615366381417, "s": "%s", "U": 2517661101, "u": 2517661113, "b": [ { "p": "54672.1", "s": 95 }, { "p": "54664.5", "s": 58794 } ], "a": [ { "p": "54743.6", "s": 95 }, { "p": "54742", "s": 95 } ] }}` optionsOrderbookSnapshotPushDataJSON = `{ "time": 1630650445, "channel": "options.order_book", "event": "all", "result": { "t": 1541500161123, "contract": "BTC_USDT-20211130-50000-C", "id": 93973511, "asks": [ { "p": "97.1", "s": 2245 }, { "p": "97.2", "s": 2245 } ], "bids": [ { "p": "97.2", "s": 2245 }, { "p": "97.1", "s": 2245 } ] }}` optionsOrderbookSnapshotUpdateEventPushDataJSON = `{"channel": "options.order_book", "event": "update", "time": 1630650445, "result": [ { "p": "49525.6", "s": 7726, "c": "BTC_USDT-20211130-50000-C", "id": 93973511 } ]}` ) func TestOptionsOrderbookPushData(t *testing.T) { t.Parallel() p := getPair(t, asset.Options) assert.NoError(t, e.WsHandleOptionsData(t.Context(), nil, []byte(optionsOrderbookTickerPushDataJSON))) assert.NoError(t, e.WsHandleOptionsData(t.Context(), nil, fmt.Appendf(nil, optionsOrderbookUpdatePushDataJSON, p.Upper().String()))) assert.NoError(t, e.WsHandleOptionsData(t.Context(), nil, []byte(optionsOrderbookSnapshotPushDataJSON))) assert.NoError(t, e.WsHandleOptionsData(t.Context(), nil, []byte(optionsOrderbookSnapshotUpdateEventPushDataJSON))) } const optionsOrderPushDataJSON = `{"time": 1630654851,"channel": "options.orders", "event": "update", "result": [ { "contract": "BTC_USDT-20211130-65000-C", "create_time": 1637897000, "fill_price": 0, "finish_as": "cancelled", "iceberg": 0, "id": 106, "is_close": false, "is_liq": false, "is_reduce_only": false, "left": -10, "mkfr": 0.0004, "price": 15000, "refr": 0, "refu": 0, "size": -10, "status": "finished", "text": "web", "tif": "gtc", "tkfr": 0.0004, "underlying": "BTC_USDT", "user": "9xxx", "time": 1639051907,"time_ms": 1639051907000}]}` func TestOptionsOrderPushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsOrderPushDataJSON)); err != nil { t.Errorf("%s websocket options orders push data error: %v", e.Name, err) } } const optionsUsersTradesPushDataJSON = `{ "time": 1639144214, "channel": "options.usertrades", "event": "update", "result": [{"id": "1","underlying": "BTC_USDT","order": "557940","contract": "BTC_USDT-20211216-44800-C","create_time": 1639144214,"create_time_ms": 1639144214583,"price": "4999","role": "taker","size": -1}]}` func TestOptionUserTradesPushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsUsersTradesPushDataJSON)); err != nil { t.Errorf("%s websocket options orders push data error: %v", e.Name, err) } } const optionsLiquidatesPushDataJSON = `{ "channel": "options.liquidates", "event": "update", "time": 1630654851, "result": [ { "user": "1xxxx", "init_margin": 1190, "maint_margin": 1042.5, "order_margin": 0, "time": 1639051907, "time_ms": 1639051907000} ]}` func TestOptionsLiquidatesPushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsLiquidatesPushDataJSON)); err != nil { t.Errorf("%s websocket options liquidates push data error: %v", e.Name, err) } } const optionsSettlementPushDataJSON = `{ "channel": "options.user_settlements", "event": "update", "time": 1639051907, "result": [{"contract": "BTC_USDT-20211130-65000-C","realised_pnl": -13.028,"settle_price": 70000,"settle_profit": 5,"size": 10,"strike_price": 65000,"underlying": "BTC_USDT","user": "9xxx","time": 1639051907,"time_ms": 1639051907000}]}` func TestOptionsSettlementPushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsSettlementPushDataJSON)); err != nil { t.Errorf("%s websocket options settlement push data error: %v", e.Name, err) } } const optionsPositionClosePushDataJSON = `{"channel": "options.position_closes", "event": "update", "time": 1630654851, "result": [{"contract": "BTC_USDT-20211130-50000-C","pnl": -0.0056,"settle_size": 0,"side": "long","text": "web","underlying": "BTC_USDT","user": "11xxxxx","time": 1639051907,"time_ms": 1639051907000}]}` func TestOptionsPositionClosePushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsPositionClosePushDataJSON)); err != nil { t.Errorf("%s websocket options position close push data error: %v", e.Name, err) } } const optionsBalancePushDataJSON = `{ "channel": "options.balances", "event": "update", "time": 1630654851, "result": [ { "balance": 60.79009,"change": -0.5,"text": "BTC_USDT-20211130-55000-P","type": "set","user": "11xxxx","time": 1639051907,"time_ms": 1639051907000}]}` func TestOptionsBalancePushData(t *testing.T) { t.Parallel() ctx := account.DeployCredentialsToContext(t.Context(), &account.Credentials{Key: "test", Secret: "test"}) if err := e.WsHandleOptionsData(ctx, nil, []byte(optionsBalancePushDataJSON)); err != nil { t.Errorf("%s websocket options balance push data error: %v", e.Name, err) } } const optionsPositionPushDataJSON = `{"time": 1630654851, "channel": "options.positions", "event": "update", "error": null, "result": [ { "entry_price": 0, "realised_pnl": -13.028, "size": 0, "contract": "BTC_USDT-20211130-65000-C", "user": "9010", "time": 1639051907, "time_ms": 1639051907000} ]}` func TestOptionsPositionPushData(t *testing.T) { t.Parallel() if err := e.WsHandleOptionsData(t.Context(), nil, []byte(optionsPositionPushDataJSON)); err != nil { t.Errorf("%s websocket options position push data error: %v", e.Name, err) } } func TestGenerateSubscriptionsSpot(t *testing.T) { t.Parallel() e := new(Exchange) //nolint:govet // Intentional shadow require.NoError(t, testexch.Setup(e), "Test instance Setup must not error") e.Websocket.SetCanUseAuthenticatedEndpoints(true) subs, err := e.generateSubscriptionsSpot() require.NoError(t, err, "generateSubscriptions must not error") exp := subscription.List{} assets := slices.DeleteFunc(e.GetAssetTypes(true), func(a asset.Item) bool { return !e.IsAssetWebsocketSupported(a) }) for _, s := range e.Features.Subscriptions { for _, a := range assets { if s.Asset != asset.All && s.Asset != a { continue } pairs, err := e.GetEnabledPairs(a) require.NoErrorf(t, err, "GetEnabledPairs %s must not error", a) pairs = common.SortStrings(pairs).Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}) s := s.Clone() //nolint:govet // Intentional lexical scope shadow s.Asset = a if singleSymbolChannel(channelName(s)) { for i := range pairs { s := s.Clone() //nolint:govet // Intentional lexical scope shadow switch s.Channel { case subscription.CandlesChannel: s.QualifiedChannel = "5m," + pairs[i].String() case subscription.OrderbookChannel: s.QualifiedChannel = pairs[i].String() + ",100ms" case spotOrderbookChannel: s.QualifiedChannel = pairs[i].String() + ",5,1000ms" } s.Pairs = pairs[i : i+1] exp = append(exp, s) } } else { s.Pairs = pairs s.QualifiedChannel = pairs.Join() exp = append(exp, s) } } } testsubs.EqualLists(t, exp, subs) } func TestSubscribe(t *testing.T) { t.Parallel() subs, err := e.Features.Subscriptions.ExpandTemplates(e) require.NoError(t, err, "ExpandTemplates must not error") e.Features.Subscriptions = subscription.List{} err = e.Subscribe(t.Context(), &DummyConnection{}, subs) require.NoError(t, err, "Subscribe must not error") } func TestGenerateDeliveryFuturesDefaultSubscriptions(t *testing.T) { t.Parallel() if _, err := e.GenerateDeliveryFuturesDefaultSubscriptions(); err != nil { t.Error(err) } } func TestGenerateFuturesDefaultSubscriptions(t *testing.T) { t.Parallel() e := new(Exchange) //nolint:govet // Intentional shadow require.NoError(t, testexch.Setup(e), "Test instance Setup must not error") subs, err := e.GenerateFuturesDefaultSubscriptions(asset.USDTMarginedFutures) require.NoError(t, err) require.NotEmpty(t, subs) subs, err = e.GenerateFuturesDefaultSubscriptions(asset.CoinMarginedFutures) require.NoError(t, err) require.NotEmpty(t, subs) require.NoError(t, e.CurrencyPairs.SetAssetEnabled(asset.USDTMarginedFutures, false), "SetAssetEnabled must not error") subs, err = e.GenerateFuturesDefaultSubscriptions(asset.USDTMarginedFutures) require.NoError(t, err, "Disabled asset must not error") require.Empty(t, subs, "Disabled asset must return no pairs") } func TestGenerateOptionsDefaultSubscriptions(t *testing.T) { t.Parallel() if _, err := e.GenerateOptionsDefaultSubscriptions(); err != nil { t.Error(err) } } func TestCreateAPIKeysOfSubAccount(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if _, err := e.CreateAPIKeysOfSubAccount(t.Context(), CreateAPIKeySubAccountParams{ SubAccountUserID: 12345, Body: &SubAccountKey{ APIKeyName: "12312mnfsndfsfjsdklfjsdlkfj", Permissions: []APIV4KeyPerm{ { PermissionName: "wallet", ReadOnly: false, }, { PermissionName: "spot", ReadOnly: false, }, { PermissionName: "futures", ReadOnly: false, }, { PermissionName: "delivery", ReadOnly: false, }, { PermissionName: "earn", ReadOnly: false, }, { PermissionName: "options", ReadOnly: false, }, }, }, }); err != nil { t.Error(err) } } func TestListAllAPIKeyOfSubAccount(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetAllAPIKeyOfSubAccount(t.Context(), 1234) if err != nil { t.Error(err) } } func TestUpdateAPIKeyOfSubAccount(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) if err := e.UpdateAPIKeyOfSubAccount(t.Context(), apiKey, CreateAPIKeySubAccountParams{ SubAccountUserID: 12345, Body: &SubAccountKey{ APIKeyName: "12312mnfsndfsfjsdklfjsdlkfj", Permissions: []APIV4KeyPerm{ { PermissionName: "wallet", ReadOnly: false, }, { PermissionName: "spot", ReadOnly: false, }, { PermissionName: "futures", ReadOnly: false, }, { PermissionName: "delivery", ReadOnly: false, }, { PermissionName: "earn", ReadOnly: false, }, { PermissionName: "options", ReadOnly: false, }, }, }, }); err != nil { t.Error(err) } } func TestGetAPIKeyOfSubAccount(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) _, err := e.GetAPIKeyOfSubAccount(t.Context(), 1234, "target_api_key") if err != nil { t.Error(err) } } func TestLockSubAccount(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if err := e.LockSubAccount(t.Context(), 1234); err != nil { t.Error(err) } } func TestUnlockSubAccount(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) if err := e.UnlockSubAccount(t.Context(), 1234); err != nil { t.Error(err) } } func TestParseGateioMilliSecTimeUnmarshal(t *testing.T) { t.Parallel() var timeWhenTesting int64 = 1684981731098 timeWhenTestingString := `"1684981731098"` // Normal string integerJSON := `{"number": 1684981731098}` float64JSON := `{"number": 1684981731.098}` time := time.UnixMilli(timeWhenTesting) var in types.Time err := json.Unmarshal([]byte(timeWhenTestingString), &in) if err != nil { t.Fatal(err) } if !in.Time().Equal(time) { t.Fatalf("found %v, but expected %v", in.Time(), time) } inInteger := struct { Number types.Time `json:"number"` }{} err = json.Unmarshal([]byte(integerJSON), &inInteger) if err != nil { t.Fatal(err) } if !inInteger.Number.Time().Equal(time) { t.Fatalf("found %v, but expected %v", inInteger.Number.Time(), time) } inFloat64 := struct { Number types.Time `json:"number"` }{} err = json.Unmarshal([]byte(float64JSON), &inFloat64) if err != nil { t.Fatal(err) } if !inFloat64.Number.Time().Equal(time) { t.Fatalf("found %v, but expected %v", inFloat64.Number.Time(), time) } } func TestParseTimeUnmarshal(t *testing.T) { t.Parallel() var timeWhenTesting int64 = 1684981731 timeWhenTestingString := `"1684981731"` integerJSON := `{"number": 1684981731}` float64JSON := `{"number": 1684981731.234}` timeWhenTestingStringMicroSecond := `"1691122380942.173000"` whenTime := time.Unix(timeWhenTesting, 0) var in types.Time err := json.Unmarshal([]byte(timeWhenTestingString), &in) if err != nil { t.Fatal(err) } if !in.Time().Equal(whenTime) { t.Fatalf("found %v, but expected %v", in.Time(), whenTime) } inInteger := struct { Number types.Time `json:"number"` }{} err = json.Unmarshal([]byte(integerJSON), &inInteger) if err != nil { t.Fatal(err) } if !inInteger.Number.Time().Equal(whenTime) { t.Fatalf("found %v, but expected %v", inInteger.Number.Time(), whenTime) } inFloat64 := struct { Number types.Time `json:"number"` }{} err = json.Unmarshal([]byte(float64JSON), &inFloat64) if err != nil { t.Fatal(err) } msTime := time.UnixMilli(1684981731234) if !inFloat64.Number.Time().Equal(time.UnixMilli(1684981731234)) { t.Fatalf("found %v, but expected %v", inFloat64.Number.Time(), msTime) } var microSeconds types.Time err = json.Unmarshal([]byte(timeWhenTestingStringMicroSecond), µSeconds) if err != nil { t.Fatal(err) } if !microSeconds.Time().Equal(time.UnixMicro(1691122380942173)) { t.Fatalf("found %v, but expected %v", microSeconds.Time(), time.UnixMicro(1691122380942173)) } } func TestUpdateOrderExecutionLimits(t *testing.T) { t.Parallel() testexch.UpdatePairsOnce(t, e) err := e.UpdateOrderExecutionLimits(t.Context(), 1336) require.ErrorIs(t, err, asset.ErrNotSupported) err = e.UpdateOrderExecutionLimits(t.Context(), asset.Options) require.ErrorIs(t, err, common.ErrNotYetImplemented) err = e.UpdateOrderExecutionLimits(t.Context(), asset.Spot) if err != nil { t.Fatal(err) } avail, err := e.GetAvailablePairs(asset.Spot) if err != nil { t.Fatal(err) } for i := range avail { mm, err := e.GetOrderExecutionLimits(asset.Spot, avail[i]) if err != nil { t.Fatal(err) } if mm == (order.MinMaxLevel{}) { t.Fatal("expected a value") } if mm.MinimumBaseAmount <= 0 { t.Fatalf("MinimumBaseAmount expected 0 but received %v for %v", mm.MinimumBaseAmount, avail[i]) } // 1INCH_TRY no minimum quote or base values are returned. if mm.QuoteStepIncrementSize <= 0 { t.Fatalf("QuoteStepIncrementSize expected 0 but received %v for %v", mm.QuoteStepIncrementSize, avail[i]) } if mm.AmountStepIncrementSize <= 0 { t.Fatalf("AmountStepIncrementSize expected 0 but received %v for %v", mm.AmountStepIncrementSize, avail[i]) } } } func TestForceFileStandard(t *testing.T) { t.Parallel() err := sharedtestvalues.ForceFileStandard(t, sharedtestvalues.EmptyStringPotentialPattern) if err != nil { t.Error(err) } if t.Failed() { t.Fatal("Please use types.Number type instead of `float64` and remove `,string` as strings can be empty in unmarshal process. Then call the Float64() method.") } } func TestGetFuturesContractDetails(t *testing.T) { t.Parallel() _, err := e.GetFuturesContractDetails(t.Context(), asset.Spot) require.ErrorIs(t, err, futures.ErrNotFuturesAsset) _, err = e.GetFuturesContractDetails(t.Context(), asset.PerpetualContract) require.ErrorIs(t, err, asset.ErrNotSupported) exp, err := e.GetAllDeliveryContracts(t.Context(), currency.USDT) require.NoError(t, err, "GetAllDeliveryContracts must not error") c, err := e.GetFuturesContractDetails(t.Context(), asset.DeliveryFutures) require.NoError(t, err, "GetFuturesContractDetails must not error for DeliveryFutures") assert.Equal(t, len(exp), len(c), "GetFuturesContractDetails should return same number of Delivery contracts as exist") for _, a := range []asset.Item{asset.CoinMarginedFutures, asset.USDTMarginedFutures} { c, err = e.GetFuturesContractDetails(t.Context(), a) require.NoErrorf(t, err, "GetFuturesContractDetails must not error for %s", a) assert.NotEmptyf(t, c, "GetFuturesContractDetails should return some contracts for %s", a) } } func TestGetLatestFundingRates(t *testing.T) { t.Parallel() _, err := e.GetLatestFundingRates(t.Context(), &fundingrate.LatestRateRequest{ Asset: asset.USDTMarginedFutures, Pair: currency.NewBTCUSDT(), IncludePredictedRate: true, }) assert.NoError(t, err) _, err = e.GetLatestFundingRates(t.Context(), &fundingrate.LatestRateRequest{ Asset: asset.CoinMarginedFutures, Pair: currency.NewBTCUSD(), }) assert.NoError(t, err) _, err = e.GetLatestFundingRates(t.Context(), &fundingrate.LatestRateRequest{Asset: asset.CoinMarginedFutures}) assert.NoError(t, err) _, err = e.GetLatestFundingRates(t.Context(), &fundingrate.LatestRateRequest{Asset: asset.USDTMarginedFutures}) assert.NoError(t, err) } func TestGetHistoricalFundingRates(t *testing.T) { t.Parallel() _, err := e.GetHistoricalFundingRates(t.Context(), nil) assert.ErrorIs(t, err, common.ErrNilPointer) _, err = e.GetHistoricalFundingRates(t.Context(), &fundingrate.HistoricalRatesRequest{}) assert.ErrorIs(t, err, asset.ErrNotSupported) _, err = e.GetHistoricalFundingRates(t.Context(), &fundingrate.HistoricalRatesRequest{Asset: asset.CoinMarginedFutures}) assert.ErrorIs(t, err, currency.ErrCurrencyPairEmpty) _, err = e.GetHistoricalFundingRates(t.Context(), &fundingrate.HistoricalRatesRequest{Asset: asset.Futures}) assert.ErrorIs(t, err, asset.ErrNotSupported) _, err = e.GetHistoricalFundingRates(t.Context(), &fundingrate.HistoricalRatesRequest{ Asset: asset.USDTMarginedFutures, Pair: currency.NewPair(currency.ENJ, currency.USDT), }) assert.ErrorIs(t, err, fundingrate.ErrPaymentCurrencyCannotBeEmpty) _, err = e.GetHistoricalFundingRates(t.Context(), &fundingrate.HistoricalRatesRequest{ Asset: asset.USDTMarginedFutures, Pair: currency.NewPair(currency.ENJ, currency.USDT), PaymentCurrency: currency.USDT, IncludePayments: true, }) assert.ErrorIs(t, err, common.ErrNotYetImplemented) _, err = e.GetHistoricalFundingRates(t.Context(), &fundingrate.HistoricalRatesRequest{ Asset: asset.USDTMarginedFutures, Pair: currency.NewPair(currency.ENJ, currency.USDT), PaymentCurrency: currency.USDT, IncludePredictedRate: true, }) assert.ErrorIs(t, err, common.ErrNotYetImplemented) _, err = e.GetHistoricalFundingRates(t.Context(), &fundingrate.HistoricalRatesRequest{ Asset: asset.USDTMarginedFutures, Pair: currency.NewPair(currency.ENJ, currency.USDT), PaymentCurrency: currency.USDT, StartDate: time.Now().Add(time.Hour * 16), EndDate: time.Now(), }) assert.ErrorIs(t, err, common.ErrStartAfterEnd) _, err = e.GetHistoricalFundingRates(t.Context(), &fundingrate.HistoricalRatesRequest{ Asset: asset.USDTMarginedFutures, Pair: currency.NewPair(currency.ENJ, currency.USDT), PaymentCurrency: currency.USDT, StartDate: time.Now().Add(-time.Hour * 8008), EndDate: time.Now(), }) assert.ErrorIs(t, err, fundingrate.ErrFundingRateOutsideLimits) history, err := e.GetHistoricalFundingRates(t.Context(), &fundingrate.HistoricalRatesRequest{ Asset: asset.USDTMarginedFutures, Pair: currency.NewPair(currency.ENJ, currency.USDT), PaymentCurrency: currency.USDT, }) require.NoError(t, err) assert.NotEmpty(t, history) } func TestGetOpenInterest(t *testing.T) { t.Parallel() _, err := e.GetOpenInterest(t.Context(), key.PairAsset{ Base: currency.NewCode("GOLDFISH").Item, Quote: currency.USDT.Item, Asset: asset.USDTMarginedFutures, }) assert.ErrorIs(t, err, currency.ErrPairNotFound, "GetOpenInterest should error correctly") var resp []futures.OpenInterest for _, a := range []asset.Item{asset.CoinMarginedFutures, asset.USDTMarginedFutures, asset.DeliveryFutures} { p := getPair(t, a) resp, err = e.GetOpenInterest(t.Context(), key.PairAsset{ Base: p.Base.Item, Quote: p.Quote.Item, Asset: a, }) assert.NoErrorf(t, err, "GetOpenInterest should not error for %s asset", a) assert.Lenf(t, resp, 1, "GetOpenInterest should return 1 item for %s asset", a) } resp, err = e.GetOpenInterest(t.Context()) assert.NoError(t, err, "GetOpenInterest should not error") assert.NotEmpty(t, resp, "GetOpenInterest should return some items") } func TestGetClientOrderIDFromText(t *testing.T) { t.Parallel() assert.Empty(t, getClientOrderIDFromText("api"), "should not return anything") assert.Equal(t, "t-123", getClientOrderIDFromText("t-123"), "should return t-123") } func TestGetSideAndAmountFromSize(t *testing.T) { t.Parallel() side, amount, remaining := getSideAndAmountFromSize(1, 1) assert.Equal(t, order.Long, side, "should be a buy order") assert.Equal(t, 1.0, amount, "should be 1.0") assert.Equal(t, 1.0, remaining, "should be 1.0") side, amount, remaining = getSideAndAmountFromSize(-1, -1) assert.Equal(t, order.Short, side, "should be a sell order") assert.Equal(t, 1.0, amount, "should be 1.0") assert.Equal(t, 1.0, remaining, "should be 1.0") } func TestGetFutureOrderSize(t *testing.T) { t.Parallel() _, err := getFutureOrderSize(&order.Submit{Side: order.CouldNotCloseShort, Amount: 1}) assert.ErrorIs(t, err, order.ErrSideIsInvalid) ret, err := getFutureOrderSize(&order.Submit{Side: order.Buy, Amount: 1}) require.NoError(t, err) assert.Equal(t, 1.0, ret) ret, err = getFutureOrderSize(&order.Submit{Side: order.Sell, Amount: 1}) require.NoError(t, err) assert.Equal(t, -1.0, ret) } func TestProcessFuturesOrdersPushData(t *testing.T) { t.Parallel() testCases := []struct { incoming string status order.Status }{ {`{"channel":"futures.orders","event":"update","time":1541505434,"time_ms":1541505434123,"result":[{"contract":"BTC_USD","create_time":1628736847,"create_time_ms":1628736847325,"fill_price":40000.4,"finish_as":"","finish_time":1628736848,"finish_time_ms":1628736848321,"iceberg":0,"id":4872460,"is_close":false,"is_liq":false,"is_reduce_only":false,"left":0,"mkfr":-0.00025,"price":40000.4,"refr":0,"refu":0,"size":1,"status":"open","text":"-","tif":"gtc","tkfr":0.0005,"user":"110xxxxx"}]}`, order.Open}, {`{"channel":"futures.orders","event":"update","time":1541505434,"time_ms":1541505434123,"result":[{"contract":"BTC_USD","create_time":1628736847,"create_time_ms":1628736847325,"fill_price":40000.4,"finish_as":"filled","finish_time":1628736848,"finish_time_ms":1628736848321,"iceberg":0,"id":4872460,"is_close":false,"is_liq":false,"is_reduce_only":false,"left":0,"mkfr":-0.00025,"price":40000.4,"refr":0,"refu":0,"size":1,"status":"finished","text":"-","tif":"gtc","tkfr":0.0005,"user":"110xxxxx"}]}`, order.Filled}, {`{"channel":"futures.orders","event":"update","time":1541505434,"time_ms":1541505434123,"result":[{"contract":"BTC_USD","create_time":1628736847,"create_time_ms":1628736847325,"fill_price":40000.4,"finish_as":"cancelled","finish_time":1628736848,"finish_time_ms":1628736848321,"iceberg":0,"id":4872460,"is_close":false,"is_liq":false,"is_reduce_only":false,"left":0,"mkfr":-0.00025,"price":40000.4,"refr":0,"refu":0,"size":1,"status":"finished","text":"-","tif":"gtc","tkfr":0.0005,"user":"110xxxxx"}]}`, order.Cancelled}, {`{"channel":"futures.orders","event":"update","time":1541505434,"time_ms":1541505434123,"result":[{"contract":"BTC_USD","create_time":1628736847,"create_time_ms":1628736847325,"fill_price":40000.4,"finish_as":"liquidated","finish_time":1628736848,"finish_time_ms":1628736848321,"iceberg":0,"id":4872460,"is_close":false,"is_liq":false,"is_reduce_only":false,"left":0,"mkfr":-0.00025,"price":40000.4,"refr":0,"refu":0,"size":1,"status":"finished","text":"-","tif":"gtc","tkfr":0.0005,"user":"110xxxxx"}]}`, order.Liquidated}, {`{"channel":"futures.orders","event":"update","time":1541505434,"time_ms":1541505434123,"result":[{"contract":"BTC_USD","create_time":1628736847,"create_time_ms":1628736847325,"fill_price":40000.4,"finish_as":"ioc","finish_time":1628736848,"finish_time_ms":1628736848321,"iceberg":0,"id":4872460,"is_close":false,"is_liq":false,"is_reduce_only":false,"left":0,"mkfr":-0.00025,"price":40000.4,"refr":0,"refu":0,"size":1,"status":"finished","text":"-","tif":"gtc","tkfr":0.0005,"user":"110xxxxx"}]}`, order.Cancelled}, {`{"channel":"futures.orders","event":"update","time":1541505434,"time_ms":1541505434123,"result":[{"contract":"BTC_USD","create_time":1628736847,"create_time_ms":1628736847325,"fill_price":40000.4,"finish_as":"auto_deleveraged","finish_time":1628736848,"finish_time_ms":1628736848321,"iceberg":0,"id":4872460,"is_close":false,"is_liq":false,"is_reduce_only":false,"left":0,"mkfr":-0.00025,"price":40000.4,"refr":0,"refu":0,"size":1,"status":"finished","text":"-","tif":"gtc","tkfr":0.0005,"user":"110xxxxx"}]}`, order.AutoDeleverage}, {`{"channel":"futures.orders","event":"update","time":1541505434,"time_ms":1541505434123,"result":[{"contract":"BTC_USD","create_time":1628736847,"create_time_ms":1628736847325,"fill_price":40000.4,"finish_as":"reduce_only","finish_time":1628736848,"finish_time_ms":1628736848321,"iceberg":0,"id":4872460,"is_close":false,"is_liq":false,"is_reduce_only":false,"left":0,"mkfr":-0.00025,"price":40000.4,"refr":0,"refu":0,"size":1,"status":"finished","text":"-","tif":"gtc","tkfr":0.0005,"user":"110xxxxx"}]}`, order.Cancelled}, {`{"channel":"futures.orders","event":"update","time":1541505434,"time_ms":1541505434123,"result":[{"contract":"BTC_USD","create_time":1628736847,"create_time_ms":1628736847325,"fill_price":40000.4,"finish_as":"position_closed","finish_time":1628736848,"finish_time_ms":1628736848321,"iceberg":0,"id":4872460,"is_close":false,"is_liq":false,"is_reduce_only":false,"left":0,"mkfr":-0.00025,"price":40000.4,"refr":0,"refu":0,"size":1,"status":"finished","text":"-","tif":"gtc","tkfr":0.0005,"user":"110xxxxx"}]}`, order.Closed}, {`{"channel":"futures.orders","event":"update","time":1541505434,"time_ms":1541505434123,"result":[{"contract":"BTC_USD","create_time":1628736847,"create_time_ms":1628736847325,"fill_price":40000.4,"finish_as":"stp","finish_time":1628736848,"finish_time_ms":1628736848321,"iceberg":0,"id":4872460,"is_close":false,"is_liq":false,"is_reduce_only":false,"left":0,"mkfr":-0.00025,"price":40000.4,"refr":0,"refu":0,"size":1,"status":"finished","text":"-","tif":"gtc","tkfr":0.0005,"user":"110xxxxx"}]}`, order.STP}, } for _, tc := range testCases { t.Run("", func(t *testing.T) { t.Parallel() processed, err := e.processFuturesOrdersPushData([]byte(tc.incoming), asset.CoinMarginedFutures) require.NoError(t, err) require.NotNil(t, processed) for i := range processed { assert.Equal(t, tc.status.String(), processed[i].Status.String()) } }) } } func TestGetCurrencyTradeURL(t *testing.T) { t.Parallel() testexch.UpdatePairsOnce(t, e) for _, a := range e.GetAssetTypes(false) { pairs, err := e.CurrencyPairs.GetPairs(a, false) require.NoErrorf(t, err, "cannot get pairs for %s", a) require.NotEmptyf(t, pairs, "no pairs for %s", a) resp, err := e.GetCurrencyTradeURL(t.Context(), a, pairs[0]) if a == asset.Options { require.ErrorIs(t, err, asset.ErrNotSupported) } else { require.NoError(t, err) assert.NotEmpty(t, resp) } } } func TestGetUnifiedAccount(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) // Requires unified account to be enabled for this to function. payload, err := e.GetUnifiedAccount(t.Context(), currency.EMPTYCODE) require.NoError(t, err) require.NotEmpty(t, payload) } func TestGetSettlementCurrency(t *testing.T) { t.Parallel() for _, tt := range []struct { a asset.Item p currency.Pair exp currency.Code err error }{ {asset.Futures, currency.EMPTYPAIR, currency.EMPTYCODE, asset.ErrNotSupported}, {asset.DeliveryFutures, currency.EMPTYPAIR, currency.USDT, nil}, {asset.DeliveryFutures, getPair(t, asset.DeliveryFutures), currency.USDT, nil}, {asset.USDTMarginedFutures, currency.EMPTYPAIR, currency.USDT, nil}, {asset.USDTMarginedFutures, getPair(t, asset.USDTMarginedFutures), currency.USDT, nil}, {asset.USDTMarginedFutures, getPair(t, asset.CoinMarginedFutures), currency.EMPTYCODE, errInvalidSettlementQuote}, {asset.CoinMarginedFutures, currency.EMPTYPAIR, currency.BTC, nil}, {asset.CoinMarginedFutures, getPair(t, asset.CoinMarginedFutures), currency.BTC, nil}, {asset.CoinMarginedFutures, getPair(t, asset.USDTMarginedFutures), currency.EMPTYCODE, errInvalidSettlementBase}, {asset.CoinMarginedFutures, currency.Pair{Base: currency.ETH, Quote: currency.USD}, currency.EMPTYCODE, errInvalidSettlementBase}, {asset.CoinMarginedFutures, currency.NewBTCUSDT(), currency.EMPTYCODE, errInvalidSettlementQuote}, } { c, err := getSettlementCurrency(tt.p, tt.a) if tt.err == nil { require.NoErrorf(t, err, "getSettlementCurrency must not error for %s %s", tt.a, tt.p) } else { assert.ErrorIsf(t, err, tt.err, "getSettlementCurrency should return correct error for %s %s", tt.a, tt.p) } assert.Equalf(t, tt.exp, c, "getSettlementCurrency should return correct settlement currency for %s %s", tt.a, tt.p) } } func TestGenerateWebsocketMessageID(t *testing.T) { t.Parallel() require.NotEmpty(t, e.GenerateWebsocketMessageID(false)) } type DummyConnection struct{ websocket.Connection } func (d *DummyConnection) GenerateMessageID(bool) int64 { return 1337 } func (d *DummyConnection) SendMessageReturnResponse(context.Context, request.EndpointLimit, any, any) ([]byte, error) { return []byte(`{"time":1726121320,"time_ms":1726121320745,"id":1,"conn_id":"f903779a148987ca","trace_id":"d8ee37cd14347e4ed298d44e69aedaa7","channel":"spot.tickers","event":"subscribe","payload":["BRETT_USDT"],"result":{"status":"success"},"requestId":"d8ee37cd14347e4ed298d44e69aedaa7"}`), nil } func TestHandleSubscriptions(t *testing.T) { t.Parallel() subs := subscription.List{{Channel: subscription.OrderbookChannel}} err := e.handleSubscription(t.Context(), &DummyConnection{}, subscribeEvent, subs, func(context.Context, websocket.Connection, string, subscription.List) ([]WsInput, error) { return []WsInput{{}}, nil }) require.NoError(t, err) err = e.handleSubscription(t.Context(), &DummyConnection{}, unsubscribeEvent, subs, func(context.Context, websocket.Connection, string, subscription.List) ([]WsInput, error) { return []WsInput{{}}, nil }) require.NoError(t, err) } func TestParseWSHeader(t *testing.T) { in := []string{ `{"time":1726121320,"time_ms":1726121320745,"id":1,"channel":"spot.tickers","event":"subscribe","result":{"status":"success"},"request_id":"a4"}`, `{"time_ms":1726121320746,"id":2,"channel":"spot.tickers","event":"subscribe","result":{"status":"success"},"request_id":"a4"}`, `{"time":1726121321,"id":3,"channel":"spot.tickers","event":"subscribe","result":{"status":"success"},"request_id":"a4"}`, } for _, i := range in { h, err := parseWSHeader([]byte(i)) require.NoError(t, err) require.NotEmpty(t, h.ID) assert.Equal(t, "a4", h.RequestID) assert.Equal(t, "spot.tickers", h.Channel) assert.Equal(t, "subscribe", h.Event) assert.NotEmpty(t, h.Result) switch h.ID { case 1: assert.Equal(t, int64(1726121320745), h.Time.UnixMilli()) case 2: assert.Equal(t, int64(1726121320746), h.Time.UnixMilli()) case 3: assert.Equal(t, int64(1726121321), h.Time.Unix()) } } } func TestDeriveSpotWebsocketOrderResponse(t *testing.T) { t.Parallel() var resp *WebsocketOrderResponse require.NoError(t, json.Unmarshal([]byte(`{"left":"0","update_time":"1735720637","amount":"0.0001","create_time":"1735720637","price":"0","finish_as":"filled","time_in_force":"ioc","currency_pair":"BTC_USDT","type":"market","account":"spot","side":"sell","amend_text":"-","text":"t-1735720637181634009","status":"closed","iceberg":"0","avg_deal_price":"93503.3","filled_total":"9.35033","id":"766075454481","fill_price":"9.35033","update_time_ms":1735720637188,"create_time_ms":1735720637188}`), &resp), "unmarshal must not error") got, err := e.deriveSpotWebsocketOrderResponse(resp) require.NoError(t, err) assert.Equal(t, &order.SubmitResponse{ Exchange: e.Name, OrderID: "766075454481", AssetType: asset.Spot, Pair: currency.NewBTCUSDT().Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), ClientOrderID: "t-1735720637181634009", Date: time.UnixMilli(1735720637188), LastUpdated: time.UnixMilli(1735720637188), Amount: 0.0001, AverageExecutedPrice: 93503.3, Type: order.Market, Side: order.Sell, Status: order.Filled, TimeInForce: order.ImmediateOrCancel, Cost: 0.0001, Purchased: 9.35033, }, got) } func TestDeriveSpotWebsocketOrderResponses(t *testing.T) { t.Parallel() testCases := []struct { name string orders [][]byte error error expected []*order.SubmitResponse }{ { name: "no response", orders: [][]byte{}, error: common.ErrNoResponse, }, { name: "assortment of spot orders", orders: [][]byte{ []byte(`{"left":"0","update_time":"1735720637","amount":"0.0001","create_time":"1735720637","price":"0","finish_as":"filled","time_in_force":"ioc","currency_pair":"BTC_USDT","type":"market","account":"spot","side":"sell","amend_text":"-","text":"t-1735720637181634009","status":"closed","iceberg":"0","avg_deal_price":"93503.3","filled_total":"9.35033","id":"766075454481","fill_price":"9.35033","update_time_ms":1735720637188,"create_time_ms":1735720637188}`), []byte(`{"left":"0.000008","update_time":"1735720637","amount":"9.99152","create_time":"1735720637","price":"0","finish_as":"filled","time_in_force":"ioc","currency_pair":"HNS_USDT","type":"market","account":"spot","side":"buy","amend_text":"-","text":"t-1735720637126962151","status":"closed","iceberg":"0","avg_deal_price":"0.01224","filled_total":"9.991512","id":"766075454188","fill_price":"9.991512","update_time_ms":1735720637142,"create_time_ms":1735720637142}`), []byte(`{"left":"0","update_time":"1735778597","amount":"200","create_time":"1735778597","price":"0.03673","finish_as":"filled","time_in_force":"fok","currency_pair":"REX_USDT","type":"limit","account":"spot","side":"buy","amend_text":"-","text":"t-1364","status":"closed","iceberg":"0","avg_deal_price":"0.03673","filled_total":"7.346","id":"766488882062","fill_price":"7.346","update_time_ms":1735778597363,"create_time_ms":1735778597363}`), []byte(`{"left":"0.0003","update_time":"1735780321","amount":"0.0003","create_time":"1735780321","price":"20000","finish_as":"open","time_in_force":"poc","currency_pair":"BTC_USDT","type":"limit","account":"spot","side":"buy","amend_text":"-","text":"t-1735780321603944400","status":"open","iceberg":"0","filled_total":"0","id":"766504537761","fill_price":"0","update_time_ms":1735780321729,"create_time_ms":1735780321729}`), []byte(`{"left":"1","update_time":"1735784755","amount":"1","create_time":"1735784755","price":"100","finish_as":"open","time_in_force":"gtc","currency_pair":"GT_USDT","type":"limit","account":"spot","side":"sell","amend_text":"-","text":"t-1735784754905434100","status":"open","iceberg":"0","filled_total":"0","id":"766536556747","fill_price":"0","update_time_ms":1735784755068,"create_time_ms":1735784755068}`), }, expected: []*order.SubmitResponse{ { Exchange: e.Name, OrderID: "766075454481", AssetType: asset.Spot, Pair: currency.NewBTCUSDT().Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), ClientOrderID: "t-1735720637181634009", Date: time.UnixMilli(1735720637188), LastUpdated: time.UnixMilli(1735720637188), Amount: 0.0001, AverageExecutedPrice: 93503.3, Type: order.Market, Side: order.Sell, Status: order.Filled, TimeInForce: order.ImmediateOrCancel, Cost: 0.0001, Purchased: 9.35033, }, { Exchange: e.Name, OrderID: "766075454188", AssetType: asset.Spot, Pair: currency.NewPair(currency.HNS, currency.USDT).Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), ClientOrderID: "t-1735720637126962151", Date: time.UnixMilli(1735720637142), LastUpdated: time.UnixMilli(1735720637142), RemainingAmount: 0.000008, Amount: 9.99152, AverageExecutedPrice: 0.01224, Type: order.Market, Side: order.Buy, Status: order.Filled, TimeInForce: order.ImmediateOrCancel, Cost: 9.991512, Purchased: 816.3, }, { Exchange: e.Name, OrderID: "766488882062", AssetType: asset.Spot, Pair: currency.NewPair(currency.NewCode("REX"), currency.USDT).Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), ClientOrderID: "t-1364", Date: time.UnixMilli(1735778597363), LastUpdated: time.UnixMilli(1735778597363), Amount: 200, Price: 0.03673, AverageExecutedPrice: 0.03673, Type: order.Limit, Side: order.Buy, Status: order.Filled, TimeInForce: order.FillOrKill, Cost: 7.346, Purchased: 200, }, { Exchange: e.Name, OrderID: "766504537761", AssetType: asset.Spot, Pair: currency.NewBTCUSDT().Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), ClientOrderID: "t-1735780321603944400", Date: time.UnixMilli(1735780321729), LastUpdated: time.UnixMilli(1735780321729), RemainingAmount: 0.0003, Amount: 0.0003, Price: 20000, Type: order.Limit, Side: order.Buy, Status: order.Open, TimeInForce: order.PostOnly, }, { Exchange: e.Name, OrderID: "766536556747", AssetType: asset.Spot, Pair: currency.NewPair(currency.NewCode("GT"), currency.USDT).Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), ClientOrderID: "t-1735784754905434100", Date: time.UnixMilli(1735784755068), LastUpdated: time.UnixMilli(1735784755068), RemainingAmount: 1, Amount: 1, Price: 100, Type: order.Limit, Side: order.Sell, Status: order.Open, TimeInForce: order.GoodTillCancel, }, }, }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { t.Parallel() orders := bytes.Join(tc.orders, []byte(",")) orders = append([]byte("["), append(orders, []byte("]")...)...) var resp []*WebsocketOrderResponse require.NoError(t, json.Unmarshal(orders, &resp), "unmarshal must not error") got, err := e.deriveSpotWebsocketOrderResponses(resp) require.ErrorIs(t, err, tc.error) require.Len(t, got, len(tc.expected)) for i := range got { assert.Equal(t, tc.expected[i], got[i]) } }) } } func TestDeriveFuturesWebsocketOrderResponse(t *testing.T) { t.Parallel() var resp *WebsocketFuturesOrderResponse require.NoError(t, json.Unmarshal([]byte(`{"text":"t-1337","price":"0","biz_info":"-","tif":"ioc","amend_text":"-","status":"finished","contract":"CWIF_USDT","stp_act":"-","finish_as":"filled","fill_price":"0.0000002625","id":596729318437,"create_time":1735787107.449,"size":2,"finish_time":1735787107.45,"update_time":1735787107.45,"left":0,"user":12870774,"is_reduce_only":true}`), &resp), "unmarshal must not error") got, err := e.deriveFuturesWebsocketOrderResponse(resp) require.NoError(t, err) assert.Equal(t, &order.SubmitResponse{ Exchange: e.Name, OrderID: "596729318437", AssetType: asset.Futures, Pair: currency.NewPair(currency.NewCode("CWIF"), currency.USDT).Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), ClientOrderID: "t-1337", Date: time.UnixMilli(1735787107449), LastUpdated: time.UnixMilli(1735787107450), Amount: 2, AverageExecutedPrice: 0.0000002625, Type: order.Market, Side: order.Long, Status: order.Filled, TimeInForce: order.ImmediateOrCancel, ReduceOnly: true, }, got) } func TestDeriveFuturesWebsocketOrderResponses(t *testing.T) { t.Parallel() testCases := []struct { name string orders [][]byte error error expected []*order.SubmitResponse }{ { name: "no response", orders: [][]byte{}, error: common.ErrNoResponse, }, { name: "assortment of futures orders", orders: [][]byte{ []byte(`{"text":"t-1337","price":"0","biz_info":"-","tif":"ioc","amend_text":"-","status":"finished","contract":"CWIF_USDT","stp_act":"-","finish_as":"filled","fill_price":"0.0000002625","id":596729318437,"create_time":1735787107.449,"size":2,"finish_time":1735787107.45,"update_time":1735787107.45,"left":0,"user":12870774,"is_reduce_only":true}`), []byte(`{"text":"t-1336","price":"0","biz_info":"-","tif":"ioc","amend_text":"-","status":"finished","contract":"REX_USDT","stp_act":"-","finish_as":"filled","fill_price":"0.03654","id":596662040388,"create_time":1735778597.374,"size":-2,"finish_time":1735778597.374,"update_time":1735778597.374,"left":0,"user":12870774}`), []byte(`{"text":"apiv4-ws","price":"40000","biz_info":"-","tif":"gtc","amend_text":"-","status":"open","contract":"BTC_USDT","stp_act":"-","fill_price":"0","id":596746193678,"create_time":1735789790.476,"size":1,"update_time":1735789790.476,"left":1,"user":2365748}`), []byte(`{"text":"apiv4-ws","price":"200000","biz_info":"-","tif":"gtc","amend_text":"-","status":"open","contract":"BTC_USDT","stp_act":"-","fill_price":"0","id":596748780649,"create_time":1735790222.185,"size":-1,"update_time":1735790222.185,"left":-1,"user":2365748}`), []byte(`{"text":"apiv4-ws","price":"0","biz_info":"-","tif":"ioc","amend_text":"-","status":"finished","contract":"BTC_USDT","stp_act":"-","finish_as":"filled","fill_price":"98172.9","id":36028797827161124,"create_time":1740108860.761,"size":1,"finish_time":1740108860.761,"update_time":1740108860.761,"left":0,"user":2365748}`), []byte(`{"text":"apiv4-ws","price":"0","biz_info":"-","tif":"ioc","amend_text":"-","status":"finished","contract":"BTC_USDT","stp_act":"-","finish_as":"filled","fill_price":"98113.1","id":36028797827225781,"create_time":1740109172.06,"size":-1,"finish_time":1740109172.06,"update_time":1740109172.06,"left":0,"user":2365748,"is_reduce_only":true}`), }, expected: []*order.SubmitResponse{ { Exchange: e.Name, OrderID: "596729318437", AssetType: asset.Futures, Pair: currency.NewPair(currency.NewCode("CWIF"), currency.USDT).Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), ClientOrderID: "t-1337", Date: time.UnixMilli(1735787107449), LastUpdated: time.UnixMilli(1735787107450), Amount: 2, AverageExecutedPrice: 0.0000002625, Type: order.Market, Side: order.Long, Status: order.Filled, TimeInForce: order.ImmediateOrCancel, ReduceOnly: true, }, { Exchange: e.Name, OrderID: "596662040388", AssetType: asset.Futures, Pair: currency.NewPair(currency.NewCode("REX"), currency.USDT).Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), ClientOrderID: "t-1336", Date: time.UnixMilli(1735778597374), LastUpdated: time.UnixMilli(1735778597374), Amount: 2, AverageExecutedPrice: 0.03654, Type: order.Market, Side: order.Short, Status: order.Filled, TimeInForce: order.ImmediateOrCancel, }, { Exchange: e.Name, OrderID: "596746193678", AssetType: asset.Futures, Pair: currency.NewBTCUSDT().Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), Date: time.UnixMilli(1735789790476), LastUpdated: time.UnixMilli(1735789790476), RemainingAmount: 1, Amount: 1, Price: 40000, Type: order.Limit, Side: order.Long, Status: order.Open, TimeInForce: order.GoodTillCancel, }, { Exchange: e.Name, OrderID: "596748780649", AssetType: asset.Futures, Pair: currency.NewBTCUSDT().Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), Date: time.UnixMilli(1735790222185), LastUpdated: time.UnixMilli(1735790222185), RemainingAmount: 1, Amount: 1, Price: 200000, Type: order.Limit, Side: order.Short, Status: order.Open, TimeInForce: order.GoodTillCancel, }, { Exchange: e.Name, OrderID: "36028797827161124", AssetType: asset.Futures, Pair: currency.NewBTCUSDT().Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), Date: time.UnixMilli(1740108860761), LastUpdated: time.UnixMilli(1740108860761), Amount: 1, AverageExecutedPrice: 98172.9, Type: order.Market, Side: order.Long, Status: order.Filled, TimeInForce: order.ImmediateOrCancel, }, { Exchange: e.Name, OrderID: "36028797827225781", AssetType: asset.Futures, Pair: currency.NewBTCUSDT().Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}), Date: time.UnixMilli(1740109172060), LastUpdated: time.UnixMilli(1740109172060), Amount: 1, AverageExecutedPrice: 98113.1, Type: order.Market, Side: order.Short, Status: order.Filled, TimeInForce: order.ImmediateOrCancel, ReduceOnly: true, }, }, }, } for _, tc := range testCases { t.Run(tc.name, func(t *testing.T) { t.Parallel() orders := bytes.Join(tc.orders, []byte(",")) orders = append([]byte("["), append(orders, []byte("]")...)...) var resp []*WebsocketFuturesOrderResponse require.NoError(t, json.Unmarshal(orders, &resp), "unmarshal must not error") got, err := e.deriveFuturesWebsocketOrderResponses(resp) require.ErrorIs(t, err, tc.error) require.Len(t, got, len(tc.expected)) for i := range got { assert.Equal(t, tc.expected[i], got[i]) } }) } } func TestConvertSmallBalances(t *testing.T) { t.Parallel() err := e.ConvertSmallBalances(t.Context(), currency.EMPTYCODE) require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty) sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders) err = e.ConvertSmallBalances(t.Context(), currency.F16) require.NoError(t, err) } func TestGetAccountDetails(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) got, err := e.GetAccountDetails(t.Context()) require.NoError(t, err) require.NotEmpty(t, got) } func TestGetUserTransactionRateLimitInfo(t *testing.T) { t.Parallel() sharedtestvalues.SkipTestIfCredentialsUnset(t, e) got, err := e.GetUserTransactionRateLimitInfo(t.Context()) require.NoError(t, err) require.NotEmpty(t, got) } var pairMap = map[asset.Item]currency.Pairs{} var pairsGuard sync.RWMutex func getPair(tb testing.TB, a asset.Item) currency.Pair { tb.Helper() if p := getPairs(tb, a); len(p) != 0 { return p[0] } return currency.EMPTYPAIR } func getPairs(tb testing.TB, a asset.Item) currency.Pairs { tb.Helper() pairsGuard.RLock() p, ok := pairMap[a] pairsGuard.RUnlock() if ok { return p } pairsGuard.Lock() defer pairsGuard.Unlock() p, ok = pairMap[a] // Protect Race if we blocked on Lock and another RW populated if ok { return p } testexch.UpdatePairsOnce(tb, e) enabledPairs, err := e.GetEnabledPairs(a) assert.NoErrorf(tb, err, "%s GetEnabledPairs should not error", a) if !assert.NotEmptyf(tb, enabledPairs, "%s GetEnabledPairs should not be empty", a) { tb.Fatalf("No pair available for asset %s", a) return nil } pairMap[a] = enabledPairs return enabledPairs } func BenchmarkTimeInForceFromString(b *testing.B) { for b.Loop() { for _, tifString := range []string{gtcTIF, iocTIF, pocTIF, fokTIF} { if _, err := timeInForceFromString(tifString); err != nil { b.Fatal(tifString) } } } } func TestTimeInForceFromString(t *testing.T) { t.Parallel() _, err := timeInForceFromString("abcdef") assert.ErrorIs(t, err, order.ErrUnsupportedTimeInForce) for k, v := range map[string]order.TimeInForce{gtcTIF: order.GoodTillCancel, iocTIF: order.ImmediateOrCancel, pocTIF: order.PostOnly, fokTIF: order.FillOrKill} { t.Run(k, func(t *testing.T) { t.Parallel() tif, err := timeInForceFromString(k) require.NoError(t, err) assert.Equal(t, v, tif) }) } } func TestGetTypeFromTimeInForce(t *testing.T) { t.Parallel() typeResp := getTypeFromTimeInForce("gtc", 0) assert.Equal(t, order.Limit, typeResp) typeResp = getTypeFromTimeInForce("ioc", 0) assert.Equal(t, order.Market, typeResp, "should be market order") typeResp = getTypeFromTimeInForce("poc", 123) assert.Equal(t, order.Limit, typeResp, "should be limit order") typeResp = getTypeFromTimeInForce("fok", 0) assert.Equal(t, order.Market, typeResp, "should be market order") } func TestTimeInForceString(t *testing.T) { t.Parallel() assert.Empty(t, timeInForceString(order.UnknownTIF)) for _, valid := range validTimesInForce { assert.Equal(t, valid.String, timeInForceString(valid.TimeInForce)) } } func TestIsSingleOrderbookChannel(t *testing.T) { t.Parallel() for _, tc := range []struct { channel string expected bool }{ {channel: spotOrderbookUpdateChannel, expected: true}, {channel: spotOrderbookChannel, expected: true}, {channel: spotOrderbookTickerChannel, expected: true}, {channel: futuresOrderbookChannel, expected: true}, {channel: futuresOrderbookTickerChannel, expected: true}, {channel: futuresOrderbookUpdateChannel, expected: true}, {channel: optionsOrderbookChannel, expected: true}, {channel: optionsOrderbookTickerChannel, expected: true}, {channel: optionsOrderbookUpdateChannel, expected: true}, {channel: spotTickerChannel, expected: false}, {channel: "sad", expected: false}, } { assert.Equal(t, tc.expected, isSingleOrderbookChannel(tc.channel)) } } func TestValidateSubscriptions(t *testing.T) { t.Parallel() require.NoError(t, e.ValidateSubscriptions(nil)) require.NoError(t, e.ValidateSubscriptions([]*subscription.Subscription{{Channel: spotTickerChannel, Pairs: []currency.Pair{currency.NewBTCUSDT()}}})) require.NoError(t, e.ValidateSubscriptions([]*subscription.Subscription{ {Channel: spotTickerChannel, Pairs: []currency.Pair{currency.NewBTCUSD()}}, {Channel: spotOrderbookUpdateChannel, Pairs: []currency.Pair{currency.NewBTCUSD()}}, })) require.NoError(t, e.ValidateSubscriptions([]*subscription.Subscription{ {Channel: spotTickerChannel, Pairs: []currency.Pair{currency.NewBTCUSD()}}, {Channel: spotOrderbookUpdateChannel, Pairs: []currency.Pair{currency.NewBTCUSD(), currency.NewBTCUSDT()}}, })) require.NoError(t, e.ValidateSubscriptions([]*subscription.Subscription{ {Channel: spotTickerChannel, Pairs: []currency.Pair{currency.NewBTCUSD()}}, {Channel: spotOrderbookUpdateChannel, Pairs: []currency.Pair{currency.NewBTCUSD()}}, {Channel: spotOrderbookUpdateChannel, Pairs: []currency.Pair{currency.NewBTCUSDT()}}, })) require.ErrorIs(t, e.ValidateSubscriptions([]*subscription.Subscription{ {Channel: spotTickerChannel, Pairs: []currency.Pair{currency.NewBTCUSD()}}, {Channel: spotOrderbookUpdateChannel, Pairs: []currency.Pair{currency.NewBTCUSD()}}, {Channel: spotOrderbookChannel, Pairs: []currency.Pair{currency.NewBTCUSD()}}, }), subscription.ErrExclusiveSubscription) } func TestCandlesChannelIntervals(t *testing.T) { t.Parallel() s := &subscription.Subscription{Channel: subscription.CandlesChannel, Asset: asset.Spot, Interval: 0} _, err := candlesChannelInterval(s) require.ErrorIs(t, err, kline.ErrUnsupportedInterval, "candlestickChannelInterval must error correctly with a 0 interval") s.Interval = kline.ThousandMilliseconds i, err := candlesChannelInterval(s) require.NoError(t, err) assert.Equal(t, "1000ms", i) } func TestOrderbookChannelIntervals(t *testing.T) { t.Parallel() s := &subscription.Subscription{Channel: futuresOrderbookUpdateChannel, Interval: kline.TwentyMilliseconds, Levels: 100} _, err := orderbookChannelInterval(s, asset.Futures) require.ErrorIs(t, err, subscription.ErrInvalidInterval) require.ErrorContains(t, err, "20ms only valid with Levels 20") s.Levels = 20 i, err := orderbookChannelInterval(s, asset.Futures) require.NoError(t, err) assert.Equal(t, "20ms", i) for s, exp := range map[*subscription.Subscription]error{ {Asset: asset.Binary, Channel: "unknown_channel", Interval: kline.OneYear}: nil, {Asset: asset.Spot, Channel: spotOrderbookTickerChannel, Interval: kline.OneDay}: subscription.ErrInvalidInterval, {Asset: asset.Spot, Channel: spotOrderbookTickerChannel, Interval: 0}: nil, {Asset: asset.Spot, Channel: spotOrderbookChannel, Interval: kline.OneDay}: subscription.ErrInvalidInterval, {Asset: asset.Spot, Channel: spotOrderbookChannel, Interval: kline.HundredMilliseconds}: nil, {Asset: asset.Spot, Channel: spotOrderbookChannel, Interval: kline.ThousandMilliseconds}: nil, {Asset: asset.Spot, Channel: spotOrderbookUpdateChannel, Interval: kline.OneDay}: subscription.ErrInvalidInterval, {Asset: asset.Spot, Channel: spotOrderbookUpdateChannel, Interval: kline.HundredMilliseconds}: nil, {Asset: asset.Futures, Channel: futuresOrderbookTickerChannel, Interval: kline.TenMilliseconds}: subscription.ErrInvalidInterval, {Asset: asset.Futures, Channel: futuresOrderbookTickerChannel, Interval: 0}: nil, {Asset: asset.Futures, Channel: futuresOrderbookChannel, Interval: kline.TenMilliseconds}: subscription.ErrInvalidInterval, {Asset: asset.Futures, Channel: futuresOrderbookChannel, Interval: 0}: nil, {Asset: asset.Futures, Channel: futuresOrderbookUpdateChannel, Interval: kline.OneDay}: subscription.ErrInvalidInterval, {Asset: asset.Futures, Channel: futuresOrderbookUpdateChannel, Interval: kline.HundredMilliseconds}: nil, {Asset: asset.DeliveryFutures, Channel: futuresOrderbookTickerChannel, Interval: kline.TenMilliseconds}: subscription.ErrInvalidInterval, {Asset: asset.DeliveryFutures, Channel: futuresOrderbookTickerChannel, Interval: 0}: nil, {Asset: asset.DeliveryFutures, Channel: futuresOrderbookChannel, Interval: kline.TenMilliseconds}: subscription.ErrInvalidInterval, {Asset: asset.DeliveryFutures, Channel: futuresOrderbookChannel, Interval: 0}: nil, {Asset: asset.DeliveryFutures, Channel: futuresOrderbookUpdateChannel, Interval: kline.OneDay}: subscription.ErrInvalidInterval, {Asset: asset.DeliveryFutures, Channel: futuresOrderbookUpdateChannel, Interval: kline.HundredMilliseconds}: nil, {Asset: asset.DeliveryFutures, Channel: futuresOrderbookUpdateChannel, Interval: kline.ThousandMilliseconds}: nil, {Asset: asset.Options, Channel: optionsOrderbookTickerChannel, Interval: kline.TenMilliseconds}: subscription.ErrInvalidInterval, {Asset: asset.Options, Channel: optionsOrderbookTickerChannel, Interval: 0}: nil, {Asset: asset.Options, Channel: optionsOrderbookChannel, Interval: kline.TwoHundredAndFiftyMilliseconds}: subscription.ErrInvalidInterval, {Asset: asset.Options, Channel: optionsOrderbookChannel, Interval: 0}: nil, {Asset: asset.Options, Channel: optionsOrderbookUpdateChannel, Interval: kline.OneDay}: subscription.ErrInvalidInterval, {Asset: asset.Options, Channel: optionsOrderbookUpdateChannel, Interval: kline.HundredMilliseconds}: nil, {Asset: asset.Options, Channel: optionsOrderbookUpdateChannel, Interval: kline.ThousandMilliseconds}: nil, } { t.Run(s.Asset.String()+"/"+s.Channel+"/"+s.Interval.Short(), func(t *testing.T) { t.Parallel() i, err := orderbookChannelInterval(s, s.Asset) if exp != nil { require.ErrorIs(t, err, exp) } else { switch { case s.Channel == "unknown_channel": assert.Empty(t, i, "orderbookChannelInterval should return empty for unknown channels") case strings.HasSuffix(s.Channel, "_ticker"): assert.Empty(t, i) case s.Interval == 0: assert.Equal(t, "0", i) default: exp, err2 := getIntervalString(s.Interval) require.NoError(t, err2, "getIntervalString must not error for validating expected value") require.Equal(t, exp, i) } } }) } } func TestChannelLevels(t *testing.T) { t.Parallel() for s, exp := range map[*subscription.Subscription]error{ {Channel: "unknown_channel", Asset: asset.Binary}: nil, {Channel: spotOrderbookTickerChannel, Asset: asset.Spot}: nil, {Channel: spotOrderbookTickerChannel, Asset: asset.Spot, Levels: 1}: subscription.ErrInvalidLevel, {Channel: spotOrderbookUpdateChannel, Asset: asset.Spot}: nil, {Channel: spotOrderbookUpdateChannel, Asset: asset.Spot, Levels: 100}: subscription.ErrInvalidLevel, {Channel: spotOrderbookChannel, Asset: asset.Spot}: subscription.ErrInvalidLevel, {Channel: spotOrderbookChannel, Asset: asset.Spot, Levels: 5}: nil, {Channel: spotOrderbookChannel, Asset: asset.Spot, Levels: 10}: nil, {Channel: spotOrderbookChannel, Asset: asset.Spot, Levels: 20}: nil, {Channel: spotOrderbookChannel, Asset: asset.Spot, Levels: 50}: nil, {Channel: spotOrderbookChannel, Asset: asset.Spot, Levels: 100}: nil, {Channel: futuresOrderbookChannel, Asset: asset.Futures}: subscription.ErrInvalidLevel, {Channel: futuresOrderbookChannel, Asset: asset.Futures, Levels: 1}: nil, {Channel: futuresOrderbookChannel, Asset: asset.Futures, Levels: 5}: nil, {Channel: futuresOrderbookChannel, Asset: asset.Futures, Levels: 10}: nil, {Channel: futuresOrderbookChannel, Asset: asset.Futures, Levels: 20}: nil, {Channel: futuresOrderbookChannel, Asset: asset.Futures, Levels: 50}: nil, {Channel: futuresOrderbookChannel, Asset: asset.Futures, Levels: 100}: nil, {Channel: futuresOrderbookTickerChannel, Asset: asset.Futures}: nil, {Channel: futuresOrderbookTickerChannel, Asset: asset.Futures, Levels: 1}: subscription.ErrInvalidLevel, {Channel: futuresOrderbookUpdateChannel, Asset: asset.Futures}: subscription.ErrInvalidLevel, {Channel: futuresOrderbookUpdateChannel, Asset: asset.Futures, Levels: 20}: nil, {Channel: futuresOrderbookUpdateChannel, Asset: asset.Futures, Levels: 50}: nil, {Channel: futuresOrderbookUpdateChannel, Asset: asset.DeliveryFutures}: subscription.ErrInvalidLevel, {Channel: futuresOrderbookUpdateChannel, Asset: asset.DeliveryFutures, Levels: 5}: nil, {Channel: futuresOrderbookUpdateChannel, Asset: asset.DeliveryFutures, Levels: 10}: nil, {Channel: futuresOrderbookUpdateChannel, Asset: asset.DeliveryFutures, Levels: 20}: nil, {Channel: futuresOrderbookUpdateChannel, Asset: asset.DeliveryFutures, Levels: 50}: nil, {Channel: futuresOrderbookUpdateChannel, Asset: asset.DeliveryFutures, Levels: 100}: nil, {Channel: optionsOrderbookTickerChannel, Asset: asset.Options}: nil, {Channel: optionsOrderbookTickerChannel, Asset: asset.Options, Levels: 1}: subscription.ErrInvalidLevel, {Channel: optionsOrderbookUpdateChannel, Asset: asset.Options}: subscription.ErrInvalidLevel, {Channel: optionsOrderbookUpdateChannel, Asset: asset.Options, Levels: 5}: nil, {Channel: optionsOrderbookUpdateChannel, Asset: asset.Options, Levels: 10}: nil, {Channel: optionsOrderbookUpdateChannel, Asset: asset.Options, Levels: 20}: nil, {Channel: optionsOrderbookUpdateChannel, Asset: asset.Options, Levels: 50}: nil, {Channel: optionsOrderbookChannel, Asset: asset.Options}: subscription.ErrInvalidLevel, {Channel: optionsOrderbookChannel, Asset: asset.Options, Levels: 5}: nil, {Channel: optionsOrderbookChannel, Asset: asset.Options, Levels: 10}: nil, {Channel: optionsOrderbookChannel, Asset: asset.Options, Levels: 20}: nil, {Channel: optionsOrderbookChannel, Asset: asset.Options, Levels: 50}: nil, } { t.Run(s.Asset.String()+"/"+s.Channel+"/"+strconv.Itoa(s.Levels), func(t *testing.T) { t.Parallel() l, err := channelLevels(s, s.Asset) switch { case exp != nil: require.ErrorIs(t, err, exp) case s.Levels == 0: assert.Empty(t, l) default: require.NoError(t, err) require.NotEmpty(t, l) } }) } } func TestGetIntervalString(t *testing.T) { t.Parallel() for k, exp := range map[kline.Interval]string{ kline.TenMilliseconds: "10ms", kline.TwentyMilliseconds: "20ms", kline.HundredMilliseconds: "100ms", kline.TwoHundredAndFiftyMilliseconds: "250ms", kline.ThousandMilliseconds: "1000ms", kline.TenSecond: "10s", kline.ThirtySecond: "30s", kline.OneMin: "1m", kline.FiveMin: "5m", kline.FifteenMin: "15m", kline.ThirtyMin: "30m", kline.OneHour: "1h", kline.TwoHour: "2h", kline.FourHour: "4h", kline.EightHour: "8h", kline.TwelveHour: "12h", kline.OneDay: "1d", kline.SevenDay: "7d", kline.OneMonth: "30d", } { t.Run(exp, func(t *testing.T) { t.Parallel() s, err := getIntervalString(k) require.NoError(t, err) assert.Equal(t, exp, s) }) } _, err := getIntervalString(0) assert.ErrorIs(t, err, kline.ErrUnsupportedInterval, "0 should be an invalid interval") _, err = getIntervalString(kline.FiveDay) assert.ErrorIs(t, err, kline.ErrUnsupportedInterval, "Any other random interval should also be invalid") }