Files
gocryptotrader/exchanges/bitmex/bitmex_test.go
Gareth Kirwan 5d6755b76e linters: Exclude govet shadow check on exchange instances (#2097)
* Linters: Exclude govet shadow check on e in tests

* Linters: Remove nolint rule for new(Exchange) in tests

Replay with:
```
perl -pi -e 's{(\se\s:=\s.*?)\s*//nolint:govet // Intentional shadow.*}{$1}' **/*_test.go
```
2025-10-29 09:30:23 +11:00

1218 lines
44 KiB
Go

package bitmex
import (
"log"
"net/http"
"net/http/httptest"
"os"
"testing"
"time"
gws "github.com/gorilla/websocket"
"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/exchange/websocket"
exchange "github.com/thrasher-corp/gocryptotrader/exchanges"
"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/order"
"github.com/thrasher-corp/gocryptotrader/exchanges/orderbook"
"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"
)
// Please supply your own keys 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("Bitmex Setup error: %s", err)
}
if apiKey != "" && apiSecret != "" {
e.API.AuthenticatedSupport = true
e.API.AuthenticatedWebsocketSupport = true
e.SetCredentials(apiKey, apiSecret, "", "", "", "")
}
os.Exit(m.Run())
}
func TestGetFullFundingHistory(t *testing.T) {
t.Parallel()
_, err := e.GetFullFundingHistory(t.Context(),
"", "", "", "", "", true, time.Now().Add(-time.Minute), time.Now())
require.NoError(t, err)
_, err = e.GetFullFundingHistory(t.Context(),
"LTCUSD", "1", "", "", "", true, time.Now().Add(-time.Minute), time.Now())
require.NoError(t, err)
}
func TestGetUrgentAnnouncement(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.GetUrgentAnnouncement(t.Context())
require.Error(t, err)
}
func TestGetAPIKeys(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.GetAPIKeys(t.Context())
require.Error(t, err)
}
func TestRemoveAPIKey(t *testing.T) {
t.Parallel()
_, err := e.RemoveAPIKey(t.Context(), APIKeyParams{APIKeyID: "1337"})
require.Error(t, err)
}
func TestDisableAPIKey(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.DisableAPIKey(t.Context(), APIKeyParams{APIKeyID: "1337"})
require.Error(t, err)
}
func TestEnableAPIKey(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.EnableAPIKey(t.Context(), APIKeyParams{APIKeyID: "1337"})
require.Error(t, err)
}
func TestGetTrollboxMessages(t *testing.T) {
t.Parallel()
_, err := e.GetTrollboxMessages(t.Context(), ChatGetParams{Count: 1})
require.NoError(t, err)
}
func TestSendTrollboxMessage(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.SendTrollboxMessage(t.Context(),
ChatSendParams{
ChannelID: 1337,
Message: "Hello,World!",
})
require.Error(t, err)
}
func TestGetTrollboxChannels(t *testing.T) {
t.Parallel()
_, err := e.GetTrollboxChannels(t.Context())
require.NoError(t, err)
}
func TestGetTrollboxConnectedUsers(t *testing.T) {
t.Parallel()
_, err := e.GetTrollboxConnectedUsers(t.Context())
require.NoError(t, err)
}
func TestGetAccountExecutions(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.GetAccountExecutions(t.Context(),
&GenericRequestParams{})
require.Error(t, err)
}
func TestGetAccountExecutionTradeHistory(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.GetAccountExecutionTradeHistory(t.Context(),
&GenericRequestParams{})
require.Error(t, err)
}
func TestGetFundingHistory(t *testing.T) {
t.Parallel()
_, err := e.GetAccountFundingHistory(t.Context())
require.Error(t, err)
}
func TestGetInstruments(t *testing.T) {
t.Parallel()
_, err := e.GetInstruments(t.Context(),
&GenericRequestParams{
Symbol: "XRPUSD",
})
require.NoError(t, err)
}
func TestGetActiveInstruments(t *testing.T) {
t.Parallel()
_, err := e.GetActiveInstruments(t.Context(),
&GenericRequestParams{})
require.NoError(t, err)
}
func TestGetActiveAndIndexInstruments(t *testing.T) {
t.Parallel()
_, err := e.GetActiveAndIndexInstruments(t.Context())
require.NoError(t, err)
}
func TestGetActiveIntervals(t *testing.T) {
t.Parallel()
_, err := e.GetActiveIntervals(t.Context())
require.NoError(t, err)
}
func TestGetCompositeIndex(t *testing.T) {
t.Parallel()
_, err := e.GetCompositeIndex(t.Context(),
".XBT", "", "", "", "", "", time.Time{}, time.Time{})
require.NoError(t, err)
}
func TestGetIndices(t *testing.T) {
t.Parallel()
_, err := e.GetIndices(t.Context())
require.NoError(t, err)
}
func TestGetInsuranceFundHistory(t *testing.T) {
t.Parallel()
_, err := e.GetInsuranceFundHistory(t.Context(),
&GenericRequestParams{})
require.NoError(t, err)
}
func TestGetLeaderboard(t *testing.T) {
t.Parallel()
e := new(Exchange)
err := testexch.Setup(e)
require.NoError(t, err, "Setup must not error")
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
assert.Equal(t, http.MethodGet, r.Method, "Request method should be correct")
assert.Equal(t, "/api/v1/leaderboard", r.URL.Path, "Request path should be correct")
w.WriteHeader(http.StatusOK)
_, err = w.Write([]byte(`[
{
"isRealName": true,
"name": "Trader Joe",
"profit": 12345.67
},
{
"isRealName": false,
"name": "CryptoKing",
"profit": 9876.54
}
]`))
assert.NoError(t, err, "Writing response to handler should not error")
}))
defer server.Close()
err = e.API.Endpoints.SetRunningURL(exchange.RestSpot.String(), server.URL+"/api/v1")
require.NoError(t, err, "SetRunningURL must not error")
expectedLeaderboard := []Leaderboard{
{IsRealName: true, Name: "Trader Joe", Profit: 12345.67},
{Name: "CryptoKing", Profit: 9876.54},
}
result, err := e.GetLeaderboard(t.Context(), LeaderboardGetParams{})
require.NoError(t, err, "GetLeaderboard must not error")
assert.Equal(t, expectedLeaderboard, result, "GetLeaderboard result should be correct")
}
func TestGetAliasOnLeaderboard(t *testing.T) {
t.Parallel()
_, err := e.GetAliasOnLeaderboard(t.Context())
require.Error(t, err)
}
func TestGetLiquidationOrders(t *testing.T) {
t.Parallel()
_, err := e.GetLiquidationOrders(t.Context(),
&GenericRequestParams{})
require.NoError(t, err)
}
func TestGetCurrentNotifications(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.GetCurrentNotifications(t.Context())
require.Error(t, err)
}
func TestAmendOrder(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.AmendOrder(t.Context(), &OrderAmendParams{})
require.Error(t, err)
}
func TestCreateOrder(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.CreateOrder(t.Context(),
&OrderNewParams{
Symbol: "XBTM15",
Price: 219.0,
ClientOrderID: "mm_bitmex_1a/oemUeQ4CAJZgP3fjHsA",
OrderQuantity: 98,
})
require.Error(t, err)
}
func TestCancelOrders(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.CancelOrders(t.Context(), &OrderCancelParams{})
require.Error(t, err)
}
func TestCancelAllOrders(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.CancelAllExistingOrders(t.Context(),
OrderCancelAllParams{})
require.Error(t, err)
}
func TestAmendBulkOrders(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.AmendBulkOrders(t.Context(), OrderAmendBulkParams{})
require.Error(t, err)
}
func TestCreateBulkOrders(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.CreateBulkOrders(t.Context(), OrderNewBulkParams{})
require.Error(t, err)
}
func TestCancelAllOrdersAfterTime(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.CancelAllOrdersAfterTime(t.Context(),
OrderCancelAllAfterParams{})
require.Error(t, err)
}
func TestClosePosition(t *testing.T) {
t.Parallel()
_, err := e.ClosePosition(t.Context(), OrderClosePositionParams{})
require.Error(t, err)
}
func TestGetOrderbook(t *testing.T) {
t.Parallel()
_, err := e.GetOrderbook(t.Context(),
OrderBookGetL2Params{Symbol: "XBT"})
require.NoError(t, err)
}
func TestGetPositions(t *testing.T) {
t.Parallel()
_, err := e.GetPositions(t.Context(), PositionGetParams{})
require.Error(t, err)
}
func TestIsolatePosition(t *testing.T) {
t.Parallel()
_, err := e.IsolatePosition(t.Context(),
PositionIsolateMarginParams{Symbol: "XBT"})
require.Error(t, err)
}
func TestLeveragePosition(t *testing.T) {
t.Parallel()
_, err := e.LeveragePosition(t.Context(),
PositionUpdateLeverageParams{})
require.Error(t, err)
}
func TestUpdateRiskLimit(t *testing.T) {
t.Parallel()
_, err := e.UpdateRiskLimit(t.Context(),
PositionUpdateRiskLimitParams{})
require.Error(t, err)
}
func TestTransferMargin(t *testing.T) {
t.Parallel()
_, err := e.TransferMargin(t.Context(),
PositionTransferIsolatedMarginParams{})
require.Error(t, err)
}
func TestGetQuotesByBuckets(t *testing.T) {
t.Parallel()
_, err := e.GetQuotesByBuckets(t.Context(),
&QuoteGetBucketedParams{})
require.Error(t, err)
}
func TestGetSettlementHistory(t *testing.T) {
t.Parallel()
_, err := e.GetSettlementHistory(t.Context(),
&GenericRequestParams{})
require.NoError(t, err)
}
func TestGetStats(t *testing.T) {
t.Parallel()
_, err := e.GetStats(t.Context())
require.NoError(t, err)
}
func TestGetStatsHistorical(t *testing.T) {
t.Parallel()
e := new(Exchange)
err := testexch.Setup(e)
require.NoError(t, err, "Setup must not error")
server := httptest.NewServer(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
assert.Equal(t, http.MethodGet, r.Method, "Request method should be correct")
assert.Equal(t, "/api/v1/stats/history", r.URL.Path, "Request path should be correct")
w.WriteHeader(http.StatusOK)
_, err = w.Write([]byte(`[
{
"currency": "XBt",
"date": "2023-10-26T00:00:00.000Z",
"rootSymbol": "XBT",
"turnover": 5000000000,
"volume": 100000
},
{
"currency": "XBt",
"date": "2023-10-25T10:35:42.123Z",
"rootSymbol": "XBT",
"turnover": 4500000000,
"volume": 90000
}
]`)) // Bitmex sends XBt as the currency for BTC, so we mimic it exactly
assert.NoError(t, err, "Writing response to handler should not error")
}))
defer server.Close()
err = e.API.Endpoints.SetRunningURL(exchange.RestSpot.String(), server.URL+"/api/v1")
require.NoError(t, err, "SetRunningURL must not error")
expectedStats := []StatsHistory{
{Currency: "XBt", Date: time.Date(2023, 10, 26, 0, 0, 0, 0, time.UTC), RootSymbol: "XBT", Turnover: 5000000000, Volume: 100000},
{Currency: "XBt", Date: time.Date(2023, 10, 25, 10, 35, 42, 123000000, time.UTC), RootSymbol: "XBT", Turnover: 4500000000, Volume: 90000},
}
result, err := e.GetStatsHistorical(t.Context())
require.NoError(t, err, "GetStatsHistorical must not error")
assert.Equal(t, expectedStats, result, "GetStatsHistorical result should be correct")
}
func TestGetStatSummary(t *testing.T) {
t.Parallel()
_, err := e.GetStatSummary(t.Context())
require.NoError(t, err)
}
func TestGetTrade(t *testing.T) {
t.Parallel()
_, err := e.GetTrade(t.Context(),
&GenericRequestParams{
Symbol: "XBT",
Reverse: false,
StartTime: time.Now().Add(-time.Minute).Format(time.RFC3339),
})
require.NoError(t, err)
}
func TestGetPreviousTrades(t *testing.T) {
t.Parallel()
_, err := e.GetPreviousTrades(t.Context(),
&TradeGetBucketedParams{
Symbol: "XBTBTC",
Start: time.Now().Add(-time.Hour).Unix(),
Columns: "open,high,low,close,volume",
})
require.Error(t, err)
}
func setFeeBuilder() *exchange.FeeBuilder {
return &exchange.FeeBuilder{
Amount: 1,
FeeType: exchange.CryptocurrencyTradeFee,
Pair: currency.NewPair(currency.BTC, currency.LTC),
PurchasePrice: 1,
}
}
func TestGetFeeByTypeOfflineTradeFee(t *testing.T) {
t.Parallel()
feeBuilder := setFeeBuilder()
_, err := e.GetFeeByType(t.Context(), feeBuilder)
require.NoError(t, err)
if !sharedtestvalues.AreAPICredentialsSet(e) {
assert.Equal(t, exchange.OfflineTradeFee, feeBuilder.FeeType)
} else {
assert.Equal(t, exchange.CryptocurrencyTradeFee, feeBuilder.FeeType)
}
}
func TestGetFee(t *testing.T) {
t.Parallel()
feeBuilder := setFeeBuilder()
// CryptocurrencyTradeFee Basic
_, err := e.GetFee(feeBuilder)
require.NoError(t, err)
// CryptocurrencyTradeFee High quantity
feeBuilder = setFeeBuilder()
feeBuilder.Amount = 1000
feeBuilder.PurchasePrice = 1000
_, err = e.GetFee(feeBuilder)
require.NoError(t, err)
// CryptocurrencyTradeFee IsMaker
feeBuilder = setFeeBuilder()
feeBuilder.IsMaker = true
_, err = e.GetFee(feeBuilder)
require.NoError(t, err)
// CryptocurrencyTradeFee Negative purchase price
feeBuilder = setFeeBuilder()
feeBuilder.PurchasePrice = -1000
_, err = e.GetFee(feeBuilder)
require.NoError(t, err)
// CryptocurrencyWithdrawalFee Basic
feeBuilder = setFeeBuilder()
feeBuilder.FeeType = exchange.CryptocurrencyWithdrawalFee
_, err = e.GetFee(feeBuilder)
require.NoError(t, err)
// CryptocurrencyDepositFee Basic
feeBuilder = setFeeBuilder()
feeBuilder.FeeType = exchange.CryptocurrencyDepositFee
_, err = e.GetFee(feeBuilder)
require.NoError(t, err)
// InternationalBankDepositFee Basic
feeBuilder = setFeeBuilder()
feeBuilder.FeeType = exchange.InternationalBankDepositFee
feeBuilder.FiatCurrency = currency.HKD
_, err = e.GetFee(feeBuilder)
require.NoError(t, err)
// InternationalBankWithdrawalFee Basic
feeBuilder = setFeeBuilder()
feeBuilder.FeeType = exchange.InternationalBankWithdrawalFee
feeBuilder.FiatCurrency = currency.HKD
_, err = e.GetFee(feeBuilder)
require.NoError(t, err)
}
func TestFormatWithdrawPermissions(t *testing.T) {
t.Parallel()
expectedResult := exchange.AutoWithdrawCryptoWithAPIPermissionText + " & " + exchange.WithdrawCryptoWith2FAText +
" & " + exchange.WithdrawCryptoWithEmailText + " & " + exchange.NoFiatWithdrawalsText
withdrawPermissions := e.FormatWithdrawPermissions()
assert.Equal(t, expectedResult, withdrawPermissions)
}
func TestGetActiveOrders(t *testing.T) {
t.Parallel()
getOrdersRequest := order.MultiOrderRequest{
Type: order.AnyType,
AssetType: asset.Spot,
Side: order.AnySide,
}
_, err := e.GetActiveOrders(t.Context(), &getOrdersRequest)
if sharedtestvalues.AreAPICredentialsSet(e) {
require.NoError(t, err)
} else {
require.ErrorIs(t, err, exchange.ErrAuthenticationSupportNotEnabled)
}
}
func TestGetOrderHistory(t *testing.T) {
t.Parallel()
getOrdersRequest := order.MultiOrderRequest{
Type: order.AnyType,
Pairs: []currency.Pair{currency.NewPair(currency.LTC, currency.BTC)},
AssetType: asset.Spot,
Side: order.AnySide,
}
_, err := e.GetOrderHistory(t.Context(), &getOrdersRequest)
if sharedtestvalues.AreAPICredentialsSet(e) {
require.NoError(t, err)
} else {
require.ErrorIs(t, err, exchange.ErrAuthenticationSupportNotEnabled)
}
}
// Any tests below this line have the ability to impact your orders on the exchange. Enable canManipulateRealOrders to run them
// ----------------------------------------------------------------------------------------------------------------------------
func TestSubmitOrder(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
orderSubmission := &order.Submit{
Exchange: e.Name,
Pair: currency.Pair{
Base: currency.XBT,
Quote: currency.USD,
},
Side: order.Buy,
Type: order.Limit,
Price: 1,
Amount: 1,
ClientID: "meowOrder",
AssetType: asset.Futures,
}
response, err := e.SubmitOrder(t.Context(), orderSubmission)
if sharedtestvalues.AreAPICredentialsSet(e) {
require.NoError(t, err)
assert.Equal(t, order.New, response.Status)
} else {
require.ErrorIs(t, err, exchange.ErrAuthenticationSupportNotEnabled)
}
}
func TestCancelExchangeOrder(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
currencyPair := currency.NewPair(currency.LTC, currency.BTC)
orderCancellation := &order.Cancel{
OrderID: "123456789012345678901234567890123456",
AccountID: "1",
Pair: currencyPair,
AssetType: asset.Futures,
}
err := e.CancelOrder(t.Context(), orderCancellation)
if sharedtestvalues.AreAPICredentialsSet(e) {
require.NoError(t, err)
} else {
require.ErrorIs(t, err, exchange.ErrAuthenticationSupportNotEnabled)
}
}
func TestCancelAllExchangeOrders(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
currencyPair := currency.NewPair(currency.LTC, currency.BTC)
orderCancellation := &order.Cancel{
OrderID: "123456789012345678901234567890123456",
AccountID: "1",
Pair: currencyPair,
AssetType: asset.Futures,
}
resp, err := e.CancelAllOrders(t.Context(), orderCancellation)
if sharedtestvalues.AreAPICredentialsSet(e) {
require.NoError(t, err)
require.Empty(t, resp.Status, "CancelAllOrders must not fail to cancel orders")
} else {
require.ErrorIs(t, err, exchange.ErrAuthenticationSupportNotEnabled)
}
}
func TestUpdateAccountBalances(t *testing.T) {
t.Parallel()
if sharedtestvalues.AreAPICredentialsSet(e) {
_, err := e.UpdateAccountBalances(t.Context(), asset.Spot)
require.NoError(t, err)
_, err = e.UpdateAccountBalances(t.Context(), asset.Futures)
require.NoError(t, err)
} else {
_, err := e.UpdateAccountBalances(t.Context(), asset.Spot)
require.Error(t, err)
_, err = e.UpdateAccountBalances(t.Context(), asset.Futures)
require.Error(t, err)
}
}
func TestModifyOrder(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
_, err := e.ModifyOrder(t.Context(),
&order.Modify{OrderID: "1337", AssetType: asset.Futures})
require.Error(t, err)
}
func TestWithdraw(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
withdrawCryptoRequest := withdraw.Request{
Exchange: e.Name,
Crypto: withdraw.CryptoRequest{
Address: core.BitcoinDonationAddress,
},
Amount: -1,
Currency: currency.BTC,
Description: "WITHDRAW IT ALL",
OneTimePassword: 696969,
}
_, err := e.WithdrawCryptocurrencyFunds(t.Context(), &withdrawCryptoRequest)
if sharedtestvalues.AreAPICredentialsSet(e) {
require.NoError(t, err)
} else {
require.Error(t, err)
}
}
func TestWithdrawFiat(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
withdrawFiatRequest := withdraw.Request{}
_, err := e.WithdrawFiatFunds(t.Context(), &withdrawFiatRequest)
require.ErrorIs(t, err, common.ErrFunctionNotSupported)
}
func TestWithdrawInternationalBank(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCannotManipulateOrders(t, e, canManipulateRealOrders)
withdrawFiatRequest := withdraw.Request{}
_, err := e.WithdrawFiatFundsToInternationalBank(t.Context(),
&withdrawFiatRequest)
require.ErrorIs(t, err, common.ErrFunctionNotSupported)
}
func TestGetDepositAddress(t *testing.T) {
t.Parallel()
if sharedtestvalues.AreAPICredentialsSet(e) {
_, err := e.GetDepositAddress(t.Context(), currency.BTC, "", "")
require.NoError(t, err)
} else {
_, err := e.GetDepositAddress(t.Context(), currency.BTC, "", "")
require.Error(t, err)
}
}
func TestWsAuth(t *testing.T) {
t.Parallel()
if !e.Websocket.IsEnabled() && !e.API.AuthenticatedWebsocketSupport || !sharedtestvalues.AreAPICredentialsSet(e) {
t.Skip(websocket.ErrWebsocketNotEnabled.Error())
}
var dialer gws.Dialer
err := e.Websocket.Conn.Dial(t.Context(), &dialer, http.Header{})
require.NoError(t, err)
go e.wsReadData()
err = e.websocketSendAuth(t.Context())
require.NoError(t, err)
timer := time.NewTimer(sharedtestvalues.WebsocketResponseDefaultTimeout)
select {
case resp := <-e.Websocket.DataHandler:
sub, ok := resp.(WebsocketSubscribeResp)
if !ok {
t.Fatal("unable to type assert WebsocketSubscribeResp")
}
if !sub.Success {
t.Error("Expected successful subscription")
}
case <-timer.C:
t.Error("Have not received a response")
}
timer.Stop()
}
func TestUpdateTradablePairs(t *testing.T) {
t.Parallel()
err := e.UpdateTradablePairs(t.Context())
require.NoError(t, err)
}
func TestWsPositionUpdate(t *testing.T) {
t.Parallel()
pressXToJSON := []byte(`{"table":"position",
"action":"update",
"data":[{
"account":2,"symbol":"ETHUSD","currency":"XBt",
"currentTimestamp":"2017-04-04T22:07:42.442Z", "currentQty":1,"markPrice":1136.88,"markValue":-87960,
"riskValue":87960,"homeNotional":0.0008796,"posState":"Liquidation","maintMargin":263,
"unrealisedGrossPnl":-677,"unrealisedPnl":-677,"unrealisedPnlPcnt":-0.0078,"unrealisedRoePcnt":-0.7756,
"simpleQty":0.001,"liquidationPrice":1140.1, "timestamp":"2017-04-04T22:07:45.442Z"
}]}`)
err := e.wsHandleData(pressXToJSON)
require.NoError(t, err)
}
func TestWsInsertExectuionUpdate(t *testing.T) {
t.Parallel()
pressXToJSON := []byte(`{"table":"execution",
"action":"insert",
"data":[{
"execID":"0193e879-cb6f-2891-d099-2c4eb40fee21",
"orderID":"00000000-0000-0000-0000-000000000000","clOrdID":"","clOrdLinkID":"","account":2,"symbol":"ETHUSD",
"side":"Sell","lastQty":1,"lastPx":1134.37,"underlyingLastPx":null,"lastMkt":"XBME",
"lastLiquidityInd":"RemovedLiquidity", "simpleOrderQty":null,"orderQty":1,"price":1134.37,"displayQty":null,
"stopPx":null,"pegOffsetValue":null,"pegPriceType":"","currency":"USD","settlCurrency":"XBt",
"execType":"Trade","ordType":"Limit","timeInForce":"ImmediateOrCancel","execInst":"",
"contingencyType":"","exDestination":"XBME","ordStatus":"Filled","triggered":"","workingIndicator":false,
"ordRejReason":"","simpleLeavesQty":0,"leavesQty":0,"simpleCumQty":0.001,"cumQty":1,"avgPx":1134.37,
"commission":0.00075,"tradePublishIndicator":"DoNotPublishTrade","multiLegReportingType":"SingleSecurity",
"text":"Liquidation","trdMatchID":"7f4ab7f6-0006-3234-76f4-ae1385aad00f","execCost":88155,"execComm":66,
"homeNotional":-0.00088155,"foreignNotional":1,"transactTime":"2017-04-04T22:07:46.035Z",
"timestamp":"2017-04-04T22:07:46.035Z"
}]}`)
err := e.wsHandleData(pressXToJSON)
require.NoError(t, err)
}
func TestWSPositionUpdateHandling(t *testing.T) {
t.Parallel()
pressXToJSON := []byte(`{"table":"position",
"action":"update",
"data":[{
"account":2,"symbol":"ETHUSD","currency":"XBt","currentQty":1,
"markPrice":1136.88,"posState":"Liquidated","simpleQty":0.001,"liquidationPrice":1140.1,"bankruptPrice":1134.37,
"timestamp":"2017-04-04T22:07:46.019Z"
}]}`)
err := e.wsHandleData(pressXToJSON)
require.NoError(t, err)
pressXToJSON = []byte(`{"table":"position",
"action":"update",
"data":[{
"account":2,"symbol":"ETHUSD","currency":"XBt",
"deleveragePercentile":null,"rebalancedPnl":1003,"prevRealisedPnl":-1003,"execSellQty":1,
"execSellCost":88155,"execQty":0,"execCost":872,"execComm":131,"currentTimestamp":"2017-04-04T22:07:46.140Z",
"currentQty":0,"currentCost":872,"currentComm":131,"realisedCost":872,"unrealisedCost":0,"grossExecCost":0,
"isOpen":false,"markPrice":null,"markValue":0,"riskValue":0,"homeNotional":0,"foreignNotional":0,"posState":"",
"posCost":0,"posCost2":0,"posInit":0,"posComm":0,"posMargin":0,"posMaint":0,"maintMargin":0,
"realisedGrossPnl":-872,"realisedPnl":-1003,"unrealisedGrossPnl":0,"unrealisedPnl":0,
"unrealisedPnlPcnt":0,"unrealisedRoePcnt":0,"simpleQty":0,"simpleCost":0,"simpleValue":0,"avgCostPrice":null,
"avgEntryPrice":null,"breakEvenPrice":null,"marginCallPrice":null,"liquidationPrice":null,"bankruptPrice":null,
"timestamp":"2017-04-04T22:07:46.140Z"
}]}`)
err = e.wsHandleData(pressXToJSON)
require.NoError(t, err)
}
func TestWSOrderbookHandling(t *testing.T) {
t.Parallel()
pressXToJSON := []byte(`{
"table":"orderBookL2_25",
"keys":["symbol","id","side"],
"types":{"id":"long","price":"float","side":"symbol","size":"long","symbol":"symbol"},
"foreignKeys":{"side":"side","symbol":"instrument"},
"attributes":{"id":"sorted","symbol":"grouped"},
"action":"partial",
"data":[
{"symbol":"ETHUSD","id":17999992000,"side":"Sell","size":100,"price":80,"timestamp":"2017-04-04T22:16:38.461Z"},
{"symbol":"ETHUSD","id":17999993000,"side":"Sell","size":20,"price":70},
{"symbol":"ETHUSD","id":17999994000,"side":"Sell","size":10,"price":60},
{"symbol":"ETHUSD","id":17999995000,"side":"Buy","size":10,"price":50},
{"symbol":"ETHUSD","id":17999996000,"side":"Buy","size":20,"price":40},
{"symbol":"ETHUSD","id":17999997000,"side":"Buy","size":100,"price":30}
]}`)
err := e.wsHandleData(pressXToJSON)
require.NoError(t, err)
pressXToJSON = []byte(`{
"table":"orderBookL2_25",
"action":"update",
"data":[
{"symbol":"ETHUSD","id":17999995000,"side":"Buy","size":5,"timestamp":"2017-04-04T22:16:38.461Z"}
]}`)
err = e.wsHandleData(pressXToJSON)
require.NoError(t, err)
pressXToJSON = []byte(`{
"table":"orderBookL2_25",
"action":"update",
"data":[]}`)
err = e.wsHandleData(pressXToJSON)
require.ErrorContains(t, err, "empty orderbook")
pressXToJSON = []byte(`{
"table":"orderBookL2_25",
"action":"delete",
"data":[
{"symbol":"ETHUSD","id":17999995000,"side":"Buy","timestamp":"2017-04-04T22:16:38.461Z"}
]}`)
err = e.wsHandleData(pressXToJSON)
require.NoError(t, err)
pressXToJSON = []byte(`{
"table":"orderBookL2_25",
"action":"delete",
"data":[
{"symbol":"ETHUSD","id":17999995000,"side":"Buy","timestamp":"2017-04-04T22:16:38.461Z"}
]}`)
err = e.wsHandleData(pressXToJSON)
assert.ErrorIs(t, err, orderbook.ErrOrderbookInvalid)
}
func TestWSDeleveragePositionUpdateHandling(t *testing.T) {
t.Parallel()
pressXToJSON := []byte(`{"table":"position",
"action":"update",
"data":[{
"account":2,"symbol":"ETHUSD","currency":"XBt","currentQty":2000,
"markPrice":1160.72,"posState":"Deleverage","simpleQty":1.746,"liquidationPrice":1140.1,
"timestamp":"2017-04-04T22:16:38.460Z"
}]}`)
err := e.wsHandleData(pressXToJSON)
require.NoError(t, err)
pressXToJSON = []byte(`{"table":"position",
"action":"update",
"data":[{
"account":2,"symbol":"ETHUSD","currency":"XBt",
"deleveragePercentile":null,"rebalancedPnl":-2171150,"prevRealisedPnl":2172153,"execSellQty":2001,
"execSellCost":172394155,"execQty":0,"execCost":-2259128,"execComm":87978,
"currentTimestamp":"2017-04-04T22:16:38.547Z","currentQty":0,"currentCost":-2259128,
"currentComm":87978,"realisedCost":-2259128,"unrealisedCost":0,"grossExecCost":0,"isOpen":false,
"markPrice":null,"markValue":0,"riskValue":0,"homeNotional":0,"foreignNotional":0,"posState":"","posCost":0,
"posCost2":0,"posInit":0,"posComm":0,"posMargin":0,"posMaint":0,"maintMargin":0,"realisedGrossPnl":2259128,
"realisedPnl":2171150,"unrealisedGrossPnl":0,"unrealisedPnl":0,"unrealisedPnlPcnt":0,"unrealisedRoePcnt":0,
"simpleQty":0,"simpleCost":0,"simpleValue":0,"simplePnl":0,"simplePnlPcnt":0,"avgCostPrice":null,
"avgEntryPrice":null,"breakEvenPrice":null,"marginCallPrice":null,"liquidationPrice":null,"bankruptPrice":null,
"timestamp":"2017-04-04T22:16:38.547Z"
}]}`)
err = e.wsHandleData(pressXToJSON)
require.NoError(t, err)
}
func TestWSDeleverageExecutionInsertHandling(t *testing.T) {
t.Parallel()
pressXToJSON := []byte(`{"table":"execution",
"action":"insert",
"data":[{
"execID":"20ad1ff4-c110-a4f2-dd31-f94eaa0701fd",
"orderID":"00000000-0000-0000-0000-000000000000","clOrdID":"","clOrdLinkID":"","account":2,"symbol":"ETHUSD",
"side":"Sell","lastQty":2000,"lastPx":1160.72,"underlyingLastPx":null,"lastMkt":"XBME",
"lastLiquidityInd":"AddedLiquidity","simpleOrderQty":null,"orderQty":2000,"price":1160.72,"displayQty":null,
"stopPx":null,"pegOffsetValue":null,"pegPriceType":"","currency":"USD","settlCurrency":"XBt","execType":"Trade",
"ordType":"Limit","timeInForce":"GoodTillCancel","execInst":"","contingencyType":"","exDestination":"XBME",
"ordStatus":"Filled","triggered":"","workingIndicator":false,"ordRejReason":"",
"simpleLeavesQty":0,"leavesQty":0,"simpleCumQty":1.746,"cumQty":2000,"avgPx":1160.72,"commission":-0.00025,
"tradePublishIndicator":"PublishTrade","multiLegReportingType":"SingleSecurity","text":"Deleverage",
"trdMatchID":"1e849b8a-7e88-3c67-a93f-cc654d40e8ba","execCost":172306000,"execComm":-43077,
"homeNotional":-1.72306,"foreignNotional":2000,"transactTime":"2017-04-04T22:16:38.472Z",
"timestamp":"2017-04-04T22:16:38.472Z"
}]}`)
err := e.wsHandleData(pressXToJSON)
require.NoError(t, err)
}
func TestWsTrades(t *testing.T) {
t.Parallel()
e := new(Exchange)
require.NoError(t, testexch.Setup(e), "Test instance Setup must not error")
e.SetSaveTradeDataStatus(true)
msg := []byte(`{"table":"trade","action":"insert","data":[{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":100,"price":258.3,"tickDirection":"MinusTick","trdMatchID":"c427f7a0-6b26-1e10-5c4e-1bd74daf2a73","grossValue":2583000,"homeNotional":0.9904912836767037,"foreignNotional":255.84389857369254},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":100,"price":258.3,"tickDirection":"ZeroMinusTick","trdMatchID":"95eb9155-b58c-70e9-44b7-34efe50302e0","grossValue":2583000,"homeNotional":0.9904912836767037,"foreignNotional":255.84389857369254},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":100,"price":258.3,"tickDirection":"ZeroMinusTick","trdMatchID":"e607c187-f25c-86bc-cb39-8afff7aaf2d9","grossValue":2583000,"homeNotional":0.9904912836767037,"foreignNotional":255.84389857369254},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":17,"price":258.3,"tickDirection":"ZeroMinusTick","trdMatchID":"0f076814-a57d-9a59-8063-ad6b823a80ac","grossValue":439110,"homeNotional":0.1683835182250396,"foreignNotional":43.49346275752773},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":100,"price":258.25,"tickDirection":"MinusTick","trdMatchID":"f4ef3dfd-51c4-538f-37c1-e5071ba1c75d","grossValue":2582500,"homeNotional":0.9904912836767037,"foreignNotional":255.79437400950872},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":100,"price":258.25,"tickDirection":"ZeroMinusTick","trdMatchID":"81ef136b-8f4a-b1cf-78a8-fffbfa89bf40","grossValue":2582500,"homeNotional":0.9904912836767037,"foreignNotional":255.79437400950872},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":100,"price":258.25,"tickDirection":"ZeroMinusTick","trdMatchID":"65a87e8c-7563-34a4-d040-94e8513c5401","grossValue":2582500,"homeNotional":0.9904912836767037,"foreignNotional":255.79437400950872},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":15,"price":258.25,"tickDirection":"ZeroMinusTick","trdMatchID":"1d11a74e-a157-3f33-036d-35a101fba50b","grossValue":387375,"homeNotional":0.14857369255150554,"foreignNotional":38.369156101426306},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":1,"price":258.25,"tickDirection":"ZeroMinusTick","trdMatchID":"40d49df1-f018-f66f-4ca5-31d4997641d7","grossValue":25825,"homeNotional":0.009904912836767036,"foreignNotional":2.5579437400950873},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":100,"price":258.2,"tickDirection":"MinusTick","trdMatchID":"36135b51-73e5-c007-362b-a55be5830c6b","grossValue":2582000,"homeNotional":0.9904912836767037,"foreignNotional":255.7448494453249},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":100,"price":258.2,"tickDirection":"ZeroMinusTick","trdMatchID":"6ee19edb-99aa-3030-ba63-933ffb347ade","grossValue":2582000,"homeNotional":0.9904912836767037,"foreignNotional":255.7448494453249},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":100,"price":258.2,"tickDirection":"ZeroMinusTick","trdMatchID":"d44be603-cdb8-d676-e3e2-f91fb12b2a70","grossValue":2582000,"homeNotional":0.9904912836767037,"foreignNotional":255.7448494453249},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":5,"price":258.2,"tickDirection":"ZeroMinusTick","trdMatchID":"a14b43b3-50b4-c075-c54d-dfb0165de33d","grossValue":129100,"homeNotional":0.04952456418383518,"foreignNotional":12.787242472266245},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":8,"price":258.2,"tickDirection":"ZeroMinusTick","trdMatchID":"3c30e175-5194-320c-8f8c-01636c2f4a32","grossValue":206560,"homeNotional":0.07923930269413629,"foreignNotional":20.45958795562599},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":50,"price":258.2,"tickDirection":"ZeroMinusTick","trdMatchID":"5b803378-760b-4919-21fc-bfb275d39ace","grossValue":1291000,"homeNotional":0.49524564183835185,"foreignNotional":127.87242472266244},{"timestamp":"2020-02-17T01:35:36.442Z","symbol":"ETHUSD","side":"Sell","size":244,"price":258.2,"tickDirection":"ZeroMinusTick","trdMatchID":"cf57fec1-c444-b9e5-5e2d-4fb643f4fdb7","grossValue":6300080,"homeNotional":2.416798732171157,"foreignNotional":624.0174326465927}]}`)
require.NoError(t, e.wsHandleData(msg), "Must not error handling a standard stream of trades")
msg = []byte(`{"table":"trade","action":"insert","data":[{"timestamp":"2020-02-17T01:35:36.442Z","symbol":".BGCT","size":14,"price":258.2,"side":"sell"}]}`)
require.ErrorIs(t, e.wsHandleData(msg), exchange.ErrSymbolNotMatched, "Must error correctly with an unknown symbol")
msg = []byte(`{"table":"trade","action":"insert","data":[{"timestamp":"2020-02-17T01:35:36.442Z","symbol":".BGCT","size":0,"price":258.2,"side":"sell"}]}`)
require.NoError(t, e.wsHandleData(msg), "Must not error that symbol is unknown when index trade is ignored due to zero size")
}
func TestGetRecentTrades(t *testing.T) {
t.Parallel()
err := e.UpdateTradablePairs(t.Context())
require.NoError(t, err)
currencyPair := e.CurrencyPairs.Pairs[asset.Futures].Available[0]
_, err = e.GetRecentTrades(t.Context(), currencyPair, asset.Futures)
require.NoError(t, err)
}
func TestGetHistoricTrades(t *testing.T) {
t.Parallel()
err := e.UpdateTradablePairs(t.Context())
require.NoError(t, err)
currencyPair := e.CurrencyPairs.Pairs[asset.Futures].Available[0]
_, err = e.GetHistoricTrades(t.Context(), currencyPair, asset.Futures, time.Now().Add(-time.Minute), time.Now())
require.NoError(t, err)
}
func TestUpdateTicker(t *testing.T) {
t.Parallel()
cp := currency.NewPair(currency.ETH, currency.USD)
_, err := e.UpdateTicker(t.Context(), cp, asset.PerpetualContract)
require.NoError(t, err)
}
func TestUpdateTickers(t *testing.T) {
t.Parallel()
err := e.UpdateTickers(t.Context(), asset.PerpetualContract)
require.NoError(t, err)
}
func TestNormalizeWalletInfo(t *testing.T) {
w := &WalletInfo{
Currency: xbtCurr,
Amount: 1e+08,
}
normalizeWalletInfo(w)
assert.Equal(t, currency.BTC, w.Currency, "Currency should be correct")
assert.Equal(t, 1.0, w.Amount, "Amount should be correct")
}
func TestGetOrderType(t *testing.T) {
t.Parallel()
_, err := e.getOrderType(0)
assert.ErrorIs(t, err, order.ErrTypeIsInvalid)
o, err := e.getOrderType(1)
require.NoError(t, err)
assert.Equal(t, order.Market, o)
}
func TestGetActionFromString(t *testing.T) {
t.Parallel()
_, err := e.GetActionFromString("meow")
assert.ErrorIs(t, err, orderbook.ErrInvalidAction)
action, err := e.GetActionFromString("update")
require.NoError(t, err)
assert.Equal(t, orderbook.UpdateAction, action)
action, err = e.GetActionFromString("delete")
require.NoError(t, err)
assert.Equal(t, orderbook.DeleteAction, action)
action, err = e.GetActionFromString("insert")
require.NoError(t, err)
assert.Equal(t, orderbook.InsertAction, action)
action, err = e.GetActionFromString("update/insert")
require.NoError(t, err)
assert.Equal(t, orderbook.UpdateOrInsertAction, action)
}
func TestGetAccountFundingHistory(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
_, err := e.GetAccountFundingHistory(t.Context())
require.NoError(t, err)
}
func TestGetWithdrawalsHistory(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
_, err := e.GetWithdrawalsHistory(t.Context(), currency.BTC, asset.Spot)
require.NoError(t, err)
}
func TestGetOrderInfo(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
_, err := e.GetOrderInfo(t.Context(), "1234", currency.NewBTCUSD(), asset.Spot)
require.NoError(t, err)
}
func TestCancelBatchOrders(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
_, err := e.CancelBatchOrders(t.Context(), []order.Cancel{
{
OrderID: "1234",
AssetType: asset.Spot,
Pair: currency.NewBTCUSD(),
},
})
require.NoError(t, err)
}
func TestGetFuturesContractDetails(t *testing.T) {
t.Parallel()
_, err := e.GetFuturesContractDetails(t.Context(), asset.Spot)
assert.ErrorIs(t, err, futures.ErrNotFuturesAsset)
_, err = e.GetFuturesContractDetails(t.Context(), asset.USDTMarginedFutures)
assert.ErrorIs(t, err, asset.ErrNotSupported)
_, err = e.GetFuturesContractDetails(t.Context(), asset.Futures)
assert.NoError(t, err)
_, err = e.GetFuturesContractDetails(t.Context(), asset.PerpetualContract)
assert.NoError(t, err)
}
func TestGetLatestFundingRates(t *testing.T) {
t.Parallel()
_, err := e.GetLatestFundingRates(t.Context(), &fundingrate.LatestRateRequest{
Asset: asset.USDTMarginedFutures,
Pair: currency.NewBTCUSDT(),
IncludePredictedRate: true,
})
assert.ErrorIs(t, err, common.ErrFunctionNotSupported)
_, err = e.GetLatestFundingRates(t.Context(), &fundingrate.LatestRateRequest{
Asset: asset.Futures,
Pair: currency.NewPair(currency.BTC, currency.KLAY),
})
assert.ErrorIs(t, err, futures.ErrNotPerpetualFuture)
_, err = e.GetLatestFundingRates(t.Context(), &fundingrate.LatestRateRequest{
Asset: asset.PerpetualContract,
})
require.NoError(t, err)
cp, err := currency.NewPairFromString("ETHUSD")
require.NoError(t, err)
_, err = e.GetLatestFundingRates(t.Context(), &fundingrate.LatestRateRequest{
Asset: asset.PerpetualContract,
Pair: cp,
})
require.NoError(t, err)
}
func TestIsPerpetualFutureCurrency(t *testing.T) {
t.Parallel()
isPerp, err := e.IsPerpetualFutureCurrency(asset.Futures, currency.NewBTCUSD())
require.NoError(t, err)
require.False(t, isPerp)
isPerp, err = e.IsPerpetualFutureCurrency(asset.PerpetualContract, currency.NewBTCUSD())
require.NoError(t, err)
require.True(t, isPerp)
}
func TestGetOpenInterest(t *testing.T) {
t.Parallel()
cp1 := currency.NewPair(currency.XBT, currency.USD)
cp2 := currency.NewPair(currency.DOGE, currency.USD)
sharedtestvalues.SetupCurrencyPairsForExchangeAsset(t, e, asset.PerpetualContract, cp1, cp2)
resp, err := e.GetOpenInterest(t.Context(), key.PairAsset{
Base: currency.XBT.Item,
Quote: currency.USD.Item,
Asset: asset.PerpetualContract,
})
assert.NoError(t, err)
assert.NotEmpty(t, resp)
resp, err = e.GetOpenInterest(t.Context(),
key.PairAsset{
Base: currency.XBT.Item,
Quote: currency.USD.Item,
Asset: asset.PerpetualContract,
},
key.PairAsset{
Base: currency.DOGE.Item,
Quote: currency.USD.Item,
Asset: asset.PerpetualContract,
})
assert.NoError(t, err)
assert.NotEmpty(t, resp)
resp, err = e.GetOpenInterest(t.Context())
assert.NoError(t, err)
assert.NotEmpty(t, resp)
_, err = e.GetOpenInterest(t.Context(), key.PairAsset{
Base: currency.BTC.Item,
Quote: currency.USDT.Item,
Asset: asset.Spot,
})
assert.ErrorIs(t, err, asset.ErrNotSupported)
}
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])
require.NoError(t, err)
assert.NotEmpty(t, resp)
}
}
func TestGenerateSubscriptions(t *testing.T) {
t.Parallel()
e := new(Exchange)
require.NoError(t, testexch.Setup(e), "Test instance Setup must not error")
p := currency.Pairs{
currency.NewPair(currency.ETH, currency.USD),
currency.NewPair(currency.BCH, currency.NewCode("Z19")),
}
exp := subscription.List{
{QualifiedChannel: bitmexWSOrderbookL2 + ":" + p[1].String(), Channel: bitmexWSOrderbookL2, Asset: asset.Futures, Pairs: p[1:2]},
{QualifiedChannel: bitmexWSOrderbookL2 + ":" + p[0].String(), Channel: bitmexWSOrderbookL2, Asset: asset.PerpetualContract, Pairs: p[:1]},
{QualifiedChannel: bitmexWSTrade + ":" + p[1].String(), Channel: bitmexWSTrade, Asset: asset.Futures, Pairs: p[1:2]},
{QualifiedChannel: bitmexWSTrade + ":" + p[0].String(), Channel: bitmexWSTrade, Asset: asset.PerpetualContract, Pairs: p[:1]},
{QualifiedChannel: bitmexWSAffiliate, Channel: bitmexWSAffiliate, Authenticated: true},
{QualifiedChannel: bitmexWSOrder, Channel: bitmexWSOrder, Authenticated: true},
{QualifiedChannel: bitmexWSMargin, Channel: bitmexWSMargin, Authenticated: true},
{QualifiedChannel: bitmexWSTransact, Channel: bitmexWSTransact, Authenticated: true},
{QualifiedChannel: bitmexWSWallet, Channel: bitmexWSWallet, Authenticated: true},
{QualifiedChannel: bitmexWSExecution + ":" + p[0].String(), Channel: bitmexWSExecution, Authenticated: true, Asset: asset.PerpetualContract, Pairs: p[:1]},
{QualifiedChannel: bitmexWSPosition + ":" + p[0].String(), Channel: bitmexWSPosition, Authenticated: true, Asset: asset.PerpetualContract, Pairs: p[:1]},
}
e.Websocket.SetCanUseAuthenticatedEndpoints(true)
subs, err := e.generateSubscriptions()
require.NoError(t, err, "generateSubscriptions must not error")
testsubs.EqualLists(t, exp, subs)
for _, a := range e.GetAssetTypes(true) {
require.NoErrorf(t, e.CurrencyPairs.SetAssetEnabled(a, false), "SetAssetEnabled must not error for %s", a)
}
_, err = e.generateSubscriptions()
require.NoError(t, err, "generateSubscriptions must not error when no assets are enabled")
}
func TestSubscribe(t *testing.T) {
t.Parallel()
e := new(Exchange)
require.NoError(t, testexch.Setup(e), "Test instance Setup must not error")
subs, err := e.generateSubscriptions() // Note: We grab this before it's overwritten by SetupWs
require.NoError(t, err, "generateSubscriptions must not error")
testexch.SetupWs(t, e)
err = e.Subscribe(subs)
require.NoError(t, err, "Subscribe must not error")
for _, s := range subs {
assert.Equalf(t, subscription.SubscribedState, s.State(), "%s state should be subscribed", s.QualifiedChannel)
}
err = e.Unsubscribe(subs)
require.NoError(t, err, "Unsubscribe must not error")
for _, s := range subs {
assert.Equalf(t, subscription.UnsubscribedState, s.State(), "%s state should be unsusbscribed", s.QualifiedChannel)
}
err = e.Subscribe(subscription.List{{QualifiedChannel: "wibble", Channel: "wibble", Asset: asset.Spot}})
require.ErrorContains(t, err, "Unknown table: wibble", "Subscribe must receive errors through websocket.Match on request json")
}