Files
gocryptotrader/exchanges/gateio/gateio_test.go
Ryan O'Hara-Reid cf54764cb7 GateIO: Update websocket orderbook manager (#1989)
* gateio: websocket ob manager fix (cherry-pick me)

* fix(gateio): update websocket orderbook manager to support delay and deadline parameters
    feat(subscriptions): mv ChannelKey type for subscription management from gateio to subscriptions
    test(gateio): enhance tests for orderbook update manager and subscription keys

* ai: nits

* linter: fix

* Fix asset typo and add in case test

* cranktakular: nits

* cranktakular: nits after merge master

* bump time delay for cache

* fix bug where on error it never initiates another orderbook fetch

* lint: fix

* Update exchanges/gateio/ws_ob_update_manager.go

Co-authored-by: Scott <gloriousCode@users.noreply.github.com>

* glorious: nits

* linter: fix

* Update exchanges/gateio/gateio_wrapper_test.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* Update exchanges/gateio/gateio_wrapper.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* Update exchanges/gateio/gateio_wrapper_test.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* Update exchanges/gateio/ws_ob_update_manager.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* Update exchanges/gateio/ws_ob_update_manager.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* Update exchanges/gateio/ws_ob_update_manager.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* Update exchanges/gateio/ws_ob_update_manager_test.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* gk: nits

* Update exchanges/gateio/gateio_wrapper.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* Update exchanges/gateio/ws_ob_update_manager.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* gk: nits

* bossking: nits

* Update exchanges/gateio/ws_ob_update_manager_test.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* gk: nits

* apply patch

* rm error state as this was on the same thread as cacheStateQueuing and had the potential to drop messages, add tests.

* linter: fix

* mock live request

* misc: fix

* Update exchanges/gateio/ws_ob_update_manager.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* Update exchanges/gateio/ws_ob_update_manager.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* Update exchanges/gateio/ws_ob_update_manager.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* Update exchanges/gateio/ws_ob_update_manager.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* Update exchanges/gateio/ws_ob_update_manager.go

Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>

* gk: nits

* field name from mtx -> m

* lint: fix

* race: check fix

* thrasher-: patch adams

---------

Co-authored-by: Ryan O'Hara-Reid <ryan.oharareid@thrasher.io>
Co-authored-by: Scott <gloriousCode@users.noreply.github.com>
Co-authored-by: Gareth Kirwan <gbjkirwan@gmail.com>
2025-11-27 12:21:17 +11:00

3717 lines
156 KiB
Go

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/accounts"
"github.com/thrasher-corp/gocryptotrader/exchange/websocket"
"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 TestGetAccountBalances(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
for _, a := range e.GetAssetTypes(false) {
_, err := e.UpdateAccountBalances(t.Context(), a)
assert.NoErrorf(t, err, "UpdateAccountBalances 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 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, true)
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 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, true)
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 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()
_, err := e.UpdateOrderbook(t.Context(), currency.EMPTYPAIR, 1336)
require.ErrorIs(t, err, currency.ErrCurrencyPairEmpty)
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 := accounts.DeployCredentialsToContext(t.Context(), &accounts.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 := accounts.DeployCredentialsToContext(t.Context(), &accounts.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)
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 = accounts.DeployCredentialsToContext(ctx, &accounts.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 := accounts.DeployCredentialsToContext(t.Context(), &accounts.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 TestOptionsPongPushData(t *testing.T) {
t.Parallel()
err := e.WsHandleOptionsData(t.Context(), nil, []byte(`{"time":1756700469,"channel":"options.pong","event":"","result":null}`))
require.NoError(t, err)
}
func TestGenerateSubscriptionsSpot(t *testing.T) {
t.Parallel()
e := new(Exchange)
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(), &FixtureConnection{}, 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)
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 TestUpdateOrderExecutionLimits(t *testing.T) {
t.Parallel()
testexch.UpdatePairsOnce(t, e)
for _, a := range e.GetAssetTypes(false) {
t.Run(a.String(), func(t *testing.T) {
t.Parallel()
switch a {
case asset.Options:
return // Options not supported
case asset.CrossMargin, asset.Margin:
require.ErrorIs(t, e.UpdateOrderExecutionLimits(t.Context(), a), asset.ErrNotSupported)
default:
require.NoError(t, e.UpdateOrderExecutionLimits(t.Context(), a), "UpdateOrderExecutionLimits must not error")
avail, err := e.GetAvailablePairs(a)
require.NoError(t, err, "GetAvailablePairs must not error")
for _, pair := range avail {
l, err := e.GetOrderExecutionLimits(a, pair)
require.NoErrorf(t, err, "GetOrderExecutionLimits must not error for %s", pair)
require.NotNilf(t, l, "GetOrderExecutionLimits %s result cannot be nil", pair)
assert.Equalf(t, a, l.Key.Asset, "asset should equal for %s", pair)
assert.Truef(t, pair.Equal(l.Key.Pair()), "pair should equal for %s", pair)
assert.Positivef(t, l.MinimumBaseAmount, "MinimumBaseAmount should be positive for %s", pair)
assert.Positivef(t, l.AmountStepIncrementSize, "AmountStepIncrementSize should be positive for %s", pair)
switch a {
case asset.USDTMarginedFutures:
assert.Positivef(t, l.MultiplierDecimal, "MultiplierDecimal should be positive for %s", pair)
assert.NotZerof(t, l.Listed, "Listed should be populated for %s", pair)
fallthrough
case asset.CoinMarginedFutures:
if !l.Delisted.IsZero() {
assert.Truef(t, l.Delisted.After(l.Delisting), "Delisted should be after Delisting for %s", pair)
}
case asset.Spot:
assert.Positivef(t, l.MinimumQuoteAmount, "MinimumQuoteAmount should be positive for %s", pair)
assert.Positivef(t, l.QuoteStepIncrementSize, "QuoteStepIncrementSize should be positive for %s", pair)
case asset.DeliveryFutures:
assert.NotZerof(t, l.Expiry, "Expiry should be populated for %s", pair)
}
}
}
})
}
}
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 TestFormatClientOrderID(t *testing.T) {
t.Parallel()
assert.Empty(t, formatClientOrderID(""), "should not return anything")
assert.Equal(t, "t-123", formatClientOrderID("t-123"), "should return t-123")
assert.Equal(t, "t-456", formatClientOrderID("456"), "should return t-456")
}
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)
}
}
type FixtureConnection struct{ websocket.Connection }
func (d *FixtureConnection) 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(), &FixtureConnection{}, subscribeEvent, subs, func(context.Context, string, subscription.List) ([]WsInput, error) {
return []WsInput{{}}, nil
})
require.NoError(t, err)
err = e.handleSubscription(t.Context(), &FixtureConnection{}, unsubscribeEvent, subs, func(context.Context, 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,
},
},
},
{
name: "batch of spot orders with error at end",
// This is specifically testing the return responses of WebsocketSpotSubmitOrders
// AverageDealPrice is not returned when using this endpoint so purchased and cost fields cannot be set.
orders: [][]byte{
[]byte(`{"account":"spot","status":"closed","side":"buy","amount":"9.98","id":"775453816782","create_time":"1736980695","update_time":"1736980695","text":"t-740","left":"0.047239","currency_pair":"ETH_USDT","type":"market","finish_as":"filled","price":"0","time_in_force":"fok","iceberg":"0","filled_total":"9.932761","fill_price":"9.932761","create_time_ms":1736980695949,"update_time_ms":1736980695949,"succeeded":true}`),
[]byte(`{"account":"spot","status":"closed","side":"buy","amount":"0.00289718","id":"775453816824","create_time":"1736980695","update_time":"1736980695","text":"t-741","left":"0.00000000962","currency_pair":"LIKE_ETH","type":"market","finish_as":"filled","price":"0","time_in_force":"fok","iceberg":"0","filled_total":"0.00289717038","fill_price":"0.00289717038","create_time_ms":1736980695956,"update_time_ms":1736980695956,"succeeded":true}`),
[]byte(`{"text":"t-742","label":"BALANCE_NOT_ENOUGH","message":"Not enough balance"}`),
},
expected: []*order.SubmitResponse{
{
Exchange: e.Name,
OrderID: "775453816782",
AssetType: asset.Spot,
Pair: currency.NewPair(currency.ETH, currency.USDT).Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}),
ClientOrderID: "t-740",
Date: time.UnixMilli(1736980695949),
LastUpdated: time.UnixMilli(1736980695949),
Amount: 9.98,
RemainingAmount: 0.047239,
Type: order.Market,
Side: order.Buy,
Status: order.Filled,
TimeInForce: order.FillOrKill,
},
{
Exchange: e.Name,
OrderID: "775453816824",
AssetType: asset.Spot,
Pair: currency.NewPair(currency.LIKE, currency.ETH).Format(currency.PairFormat{Uppercase: true, Delimiter: "_"}),
ClientOrderID: "t-741",
Date: time.UnixMilli(1736980695956),
LastUpdated: time.UnixMilli(1736980695956),
RemainingAmount: 0.00000000962,
Amount: 0.00289718,
Type: order.Market,
Side: order.Buy,
Status: order.Filled,
TimeInForce: order.FillOrKill,
},
{
Exchange: e.Name,
ClientOrderID: "t-742",
SubmissionError: order.ErrUnableToPlaceOrder,
},
},
},
}
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 {
if tc.expected[i].SubmissionError != nil {
assert.ErrorIs(t, got[i].SubmissionError, tc.expected[i].SubmissionError)
assert.Equal(t, tc.expected[i].Exchange, got[i].Exchange)
assert.Equal(t, tc.expected[i].ClientOrderID, got[i].ClientOrderID)
continue
}
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 TestToExchangeTIF(t *testing.T) {
t.Parallel()
for _, tc := range []struct {
tif order.TimeInForce
price float64
expected string
err error
}{
{price: 0, expected: iocTIF}, // market orders default to IOC
{price: 0, tif: order.FillOrKill, expected: fokTIF},
{price: 420, expected: gtcTIF}, // limit orders default to GTC
{price: 420, tif: order.GoodTillCancel, expected: gtcTIF},
{price: 420, tif: order.ImmediateOrCancel, expected: iocTIF},
{price: 420, tif: order.PostOnly, expected: pocTIF},
{price: 420, tif: order.FillOrKill, expected: fokTIF},
{tif: order.GoodTillTime, err: order.ErrUnsupportedTimeInForce},
} {
t.Run(fmt.Sprintf("TIF:%q Price:'%v'", tc.tif, tc.price), func(t *testing.T) {
t.Parallel()
got, err := toExchangeTIF(tc.tif, tc.price)
if tc.err != nil {
require.ErrorIs(t, err, tc.err)
} else {
require.NoError(t, err)
}
require.Equal(t, tc.expected, got)
})
}
}
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")
}
func TestWebsocketSubmitOrders(t *testing.T) {
t.Parallel()
_, err := e.WebsocketSubmitOrders(t.Context(), nil)
require.ErrorIs(t, err, asset.ErrNotSupported)
sub := &order.Submit{
Exchange: e.Name,
AssetType: asset.Spot,
Side: order.Buy,
Type: order.Market,
QuoteAmount: 10,
}
_, err = e.WebsocketSubmitOrders(t.Context(), []*order.Submit{sub})
require.ErrorIs(t, err, order.ErrPairIsEmpty)
sub.Pair = currency.NewBTCUSD()
cpy := *sub
cpy.AssetType = asset.Futures
_, err = e.WebsocketSubmitOrders(t.Context(), []*order.Submit{sub, &cpy})
require.ErrorIs(t, err, errSingleAssetRequired)
cpy.AssetType = asset.Spread
sub.AssetType = asset.Spread
_, err = e.WebsocketSubmitOrders(t.Context(), []*order.Submit{sub, &cpy})
require.ErrorIs(t, err, asset.ErrNotSupported)
sub.AssetType = asset.USDTMarginedFutures
cpy.AssetType = asset.USDTMarginedFutures
_, err = e.WebsocketSubmitOrders(t.Context(), []*order.Submit{sub, &cpy})
require.ErrorIs(t, err, common.ErrNotYetImplemented)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
e := newExchangeWithWebsocket(t, asset.Spot)
sub.AssetType = asset.Spot
cpy.AssetType = asset.Spot
_, err = e.WebsocketSubmitOrders(request.WithVerbose(t.Context()), []*order.Submit{sub, &cpy})
require.NoError(t, err)
}
func TestValidateContractOrderCreateParams(t *testing.T) {
t.Parallel()
for _, tc := range []struct {
params *ContractOrderCreateParams
isRest bool
err error
}{
{
err: common.ErrNilPointer,
},
{
params: &ContractOrderCreateParams{}, err: currency.ErrCurrencyPairEmpty,
},
{
params: &ContractOrderCreateParams{Contract: BTCUSDT},
err: errInvalidOrderSize,
},
{
params: &ContractOrderCreateParams{Contract: BTCUSDT, Size: 1, TimeInForce: "bad"},
err: order.ErrUnsupportedTimeInForce,
},
{
params: &ContractOrderCreateParams{Contract: BTCUSDT, Size: 1, TimeInForce: pocTIF},
err: order.ErrUnsupportedTimeInForce,
},
{
params: &ContractOrderCreateParams{Contract: BTCUSDT, Size: 1, TimeInForce: iocTIF, Text: "test"},
err: errInvalidTextPrefix,
},
{
params: &ContractOrderCreateParams{
Contract: BTCUSDT, Size: 1, TimeInForce: iocTIF, Text: "t-test", AutoSize: "silly_billy",
},
err: errInvalidAutoSize,
},
{
params: &ContractOrderCreateParams{
Contract: BTCUSDT, Size: 1, TimeInForce: iocTIF, Text: "t-test", AutoSize: "close_long",
},
err: errInvalidOrderSize,
},
{
params: &ContractOrderCreateParams{
Contract: BTCUSDT, TimeInForce: iocTIF, Text: "t-test", AutoSize: "close_long",
},
isRest: true,
err: errEmptyOrInvalidSettlementCurrency,
},
{
params: &ContractOrderCreateParams{
Contract: BTCUSDT, TimeInForce: iocTIF, Text: "t-test", AutoSize: "close_long", Settle: currency.NewCode("Silly"),
},
err: errEmptyOrInvalidSettlementCurrency,
},
{
params: &ContractOrderCreateParams{
Contract: BTCUSDT, TimeInForce: iocTIF, Text: "t-test", AutoSize: "close_long", Settle: currency.USDT,
},
},
} {
assert.ErrorIs(t, tc.params.validate(tc.isRest), tc.err)
}
}
func TestMarshalJSONNumber(t *testing.T) {
t.Parallel()
for _, tc := range []struct {
number number
expected string
}{
{number: 0, expected: `"0"`},
{number: 1, expected: `"1"`},
{number: 1.5, expected: `"1.5"`},
} {
payload, err := tc.number.MarshalJSON()
require.NoError(t, err, "MarshalJSON must not error")
assert.Equal(t, tc.expected, string(payload), "MarshalJSON should return expected value")
}
}
func TestUnmarshalJSONOrderbookLevels(t *testing.T) {
t.Parallel()
var ob OrderbookLevels
require.NoError(t, ob.UnmarshalJSON([]byte(`[{"p":"123.45","s":"0.001"}]`)))
assert.Equal(t, 123.45, ob[0].Price, "Price should be correct")
assert.Equal(t, 0.001, ob[0].Amount, "Amount should be correct")
require.Error(t, ob.UnmarshalJSON([]byte(`["p":"123.45","s":"0.001"]`)))
}
func TestGetEstimatedInterestRate(t *testing.T) {
t.Parallel()
_, err := e.GetEstimatedInterestRate(t.Context(), nil)
require.ErrorIs(t, err, currency.ErrCurrencyCodesEmpty)
_, err = e.GetEstimatedInterestRate(t.Context(), currency.Currencies{currency.EMPTYCODE})
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.GetEstimatedInterestRate(t.Context(), currency.Currencies{
currency.USDT,
currency.BTC,
currency.ETH,
currency.XRP,
currency.LTC,
currency.DOGE,
currency.BCH,
currency.SOL,
currency.ADA,
currency.DOT,
currency.MATIC,
})
require.ErrorIs(t, err, errTooManyCurrencyCodes)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
got, err := e.GetEstimatedInterestRate(t.Context(), currency.Currencies{currency.BTC})
require.NoError(t, err)
val, ok := got["BTC"]
require.True(t, ok, "result map must contain BTC key")
require.Positive(t, val.Float64(), "estimated interest rate must not be 0")
}