Files
gocryptotrader/exchanges/deribit/deribit_test.go
Copilot a9f23018e3 Deribit: Remove deprecated RFQ endpoints (#2078)
* Initial plan

* Remove deprecated Deribit RFQ endpoints

Co-authored-by: thrasher- <4685270+thrasher-@users.noreply.github.com>

* Remove extra newline before TestGetTradeVolumes

Co-authored-by: thrasher- <4685270+thrasher-@users.noreply.github.com>

* Remove extra newline before TestSetMMPConfig

Co-authored-by: thrasher- <4685270+thrasher-@users.noreply.github.com>

---------

Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com>
Co-authored-by: thrasher- <4685270+thrasher-@users.noreply.github.com>
2025-10-14 09:08:45 +11:00

4225 lines
179 KiB
Go

package deribit
import (
"context"
"fmt"
"log"
"os"
"strconv"
"strings"
"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"
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/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"
"github.com/thrasher-corp/gocryptotrader/exchanges/trade"
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 keys here to do authenticated endpoint testing
const (
apiKey = ""
apiSecret = ""
canManipulateRealOrders = false
canManipulateAPIEndpoints = false
btcPerpInstrument = "BTC-PERPETUAL"
useTestNet = false
)
var (
e *Exchange
optionsTradablePair, optionComboTradablePair, futureComboTradablePair currency.Pair
spotTradablePair = currency.NewPairWithDelimiter(currencyBTC, "USDC", "_")
futuresTradablePair = currency.NewPairWithDelimiter(currencyBTC, perpString, "-")
assetTypeToPairsMap map[asset.Item]currency.Pair
)
func TestMain(m *testing.M) {
e = new(Exchange)
if err := testexch.Setup(e); err != nil {
log.Fatalf("Deribit Setup error: %s", err)
}
if apiKey != "" && apiSecret != "" {
e.API.AuthenticatedSupport = true
e.API.AuthenticatedWebsocketSupport = true
e.SetCredentials(apiKey, apiSecret, "", "", "", "")
e.Websocket.SetCanUseAuthenticatedEndpoints(true)
}
if useTestNet {
deribitWebsocketAddress = "wss://test.deribit.com/ws" + deribitAPIVersion
if err := e.Websocket.SetWebsocketURL(deribitWebsocketAddress, false, true); err != nil {
log.Fatalf("Deribit SetWebsocketURL error: %s", err)
}
for k, v := range e.API.Endpoints.GetURLMap() {
v = strings.Replace(v, "www.deribit.com", "test.deribit.com", 1)
if err := e.API.Endpoints.SetRunningURL(k, v); err != nil {
log.Fatalf("Deribit SetRunningURL error: %s", err)
}
}
}
instantiateTradablePairs()
assetTypeToPairsMap = map[asset.Item]currency.Pair{
asset.Futures: futuresTradablePair,
asset.Spot: spotTradablePair,
asset.Options: optionsTradablePair,
asset.OptionCombo: optionComboTradablePair,
asset.FutureCombo: futureComboTradablePair,
}
setupWs()
os.Exit(m.Run())
}
func instantiateTradablePairs() {
if err := e.UpdateTradablePairs(context.Background()); err != nil {
log.Fatalf("Failed to update tradable pairs. Error: %v", err)
}
handleError := func(err error, msg string) {
if err != nil {
log.Fatalf("%s. Error: %v", msg, err)
}
}
updateTradablePair := func(assetType asset.Item, tradablePair *currency.Pair) {
if e.CurrencyPairs.IsAssetEnabled(assetType) == nil {
pairs, err := e.GetEnabledPairs(assetType)
handleError(err, fmt.Sprintf("Failed to get enabled pairs for asset type %v", assetType))
if len(pairs) == 0 {
handleError(currency.ErrCurrencyPairsEmpty, fmt.Sprintf("No enabled pairs for asset type %v", assetType))
}
if assetType == asset.Options {
*tradablePair, err = e.FormatExchangeCurrency(pairs[0], assetType)
handleError(err, "Failed to format exchange currency for options pair")
} else {
*tradablePair = pairs[0]
}
}
}
updateTradablePair(asset.Options, &optionsTradablePair)
updateTradablePair(asset.OptionCombo, &optionComboTradablePair)
updateTradablePair(asset.FutureCombo, &futureComboTradablePair)
}
func TestUpdateTicker(t *testing.T) {
t.Parallel()
_, err := e.UpdateTicker(t.Context(), currency.Pair{}, asset.Margin)
require.ErrorIs(t, err, asset.ErrNotSupported)
for assetType, cp := range assetTypeToPairsMap {
result, err := e.UpdateTicker(t.Context(), cp, assetType)
require.NoErrorf(t, err, "expected nil, got %v for asset type %s pair %s", err, assetType, cp)
require.NotNilf(t, result, "expected result not to be nil for asset type %s pair %s", assetType, cp)
}
}
func TestUpdateOrderbook(t *testing.T) {
t.Parallel()
for assetType, cp := range assetTypeToPairsMap {
result, err := e.UpdateOrderbook(t.Context(), cp, assetType)
require.NoErrorf(t, err, "asset type: %v", assetType)
require.NotNil(t, result)
}
}
func TestGetHistoricTrades(t *testing.T) {
t.Parallel()
_, err := e.GetHistoricTrades(t.Context(), futureComboTradablePair, asset.FutureCombo, time.Now().Add(-time.Minute*10), time.Now())
require.ErrorIs(t, err, asset.ErrNotSupported)
for assetType, cp := range map[asset.Item]currency.Pair{asset.Spot: spotTradablePair, asset.Futures: futuresTradablePair} {
_, err = e.GetHistoricTrades(t.Context(), cp, assetType, time.Now().Add(-time.Minute*10), time.Now())
require.NoErrorf(t, err, "asset type: %v", assetType)
}
}
func TestFetchRecentTrades(t *testing.T) {
t.Parallel()
for assetType, cp := range assetTypeToPairsMap {
result, err := e.GetRecentTrades(t.Context(), cp, assetType)
require.NoErrorf(t, err, "expected nil, got %v for asset type %s pair %s", err, assetType, cp)
require.NotNilf(t, result, "expected result not to be nil for asset type %s pair %s", assetType, cp)
}
}
func TestGetHistoricCandles(t *testing.T) {
t.Parallel()
start := time.Now().Add(-time.Hour)
end := time.Now()
assetTypesToPairMap := map[asset.Item]struct {
Pair currency.Pair
Error error
}{
asset.Futures: {Pair: futuresTradablePair},
asset.Spot: {Pair: spotTradablePair},
asset.Options: {Pair: optionsTradablePair, Error: asset.ErrNotSupported},
asset.FutureCombo: {Pair: futureComboTradablePair, Error: asset.ErrNotSupported},
asset.OptionCombo: {Pair: optionComboTradablePair, Error: asset.ErrNotSupported},
}
for assetType, info := range assetTypesToPairMap {
resp, err := e.GetHistoricCandles(t.Context(), info.Pair, assetType, kline.FifteenMin, start, end)
require.ErrorIs(t, err, info.Error)
if info.Error == nil {
require.NotEmpty(t, resp)
}
}
}
func TestGetHistoricCandlesExtended(t *testing.T) {
t.Parallel()
start := time.Now().Add(-time.Hour * 24 * 90).Truncate(kline.OneDay.Duration()).UTC()
end := time.Now().UTC()
assetsToPairsMap := map[asset.Item]struct {
Pair currency.Pair
Error error
}{
asset.Futures: {Pair: futuresTradablePair},
asset.Spot: {Pair: spotTradablePair},
asset.Options: {Pair: optionsTradablePair, Error: asset.ErrNotSupported},
asset.FutureCombo: {Pair: futureComboTradablePair, Error: asset.ErrNotSupported},
asset.OptionCombo: {Pair: optionComboTradablePair, Error: asset.ErrNotSupported},
}
for assetType, instance := range assetsToPairsMap {
resp, err := e.GetHistoricCandlesExtended(t.Context(), instance.Pair, assetType, kline.OneDay, start, end)
require.ErrorIs(t, err, instance.Error)
if instance.Error == nil {
require.NotEmpty(t, resp)
}
}
}
func TestSubmitOrder(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
assetToPairStringMap := map[asset.Item]currency.Pair{
asset.Options: optionsTradablePair,
asset.FutureCombo: futureComboTradablePair,
asset.Futures: futuresTradablePair,
}
var result *order.SubmitResponse
var err error
var info *InstrumentData
for assetType, cp := range assetToPairStringMap {
info, err = e.GetInstrument(t.Context(), formatPairString(assetType, cp))
require.NoErrorf(t, err, "expected nil, got %v for asset type %s pair %s", err, assetType, cp)
require.NotNilf(t, result, "expected result not to be nil for asset type %s pair %s", assetType, cp)
result, err = e.SubmitOrder(
t.Context(),
&order.Submit{
Exchange: e.Name,
Price: 10,
Amount: info.ContractSize * 3,
Type: order.Limit,
AssetType: assetType,
Side: order.Buy,
Pair: cp,
},
)
require.NoErrorf(t, err, "expected nil, got %v for asset type %s pair %s", err, assetType, cp)
require.NotNilf(t, result, "expected result not to be nil for asset type %s pair %s", assetType, cp)
}
}
func TestGetMarkPriceHistory(t *testing.T) {
t.Parallel()
var resp []MarkPriceHistory
err := json.Unmarshal([]byte(`[[1608142381229,0.5165791606037885],[1608142380231,0.5165737855432504],[1608142379227,0.5165768236356326]]`), &resp)
require.NoError(t, err)
assert.Len(t, resp, 3)
_, err = e.GetMarkPriceHistory(t.Context(), "", time.Now().Add(-5*time.Minute), time.Now())
require.ErrorIs(t, err, errInvalidInstrumentName)
var result []MarkPriceHistory
for _, ps := range []string{
optionPairToString(optionsTradablePair),
spotTradablePair.String(),
btcPerpInstrument,
futureComboTradablePair.String(),
} {
result, err = e.GetMarkPriceHistory(t.Context(), ps, time.Now().Add(-5*time.Minute), time.Now())
require.NoErrorf(t, err, "expected nil, got %v for pair %s", err, ps)
require.NotNilf(t, result, "expected result not to be nil for pair %s", ps)
}
}
func TestWSRetrieveMarkPriceHistory(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveMarkPriceHistory(t.Context(), "", time.Now().Add(-4*time.Hour), time.Now())
require.ErrorIs(t, err, errInvalidInstrumentName)
var result []MarkPriceHistory
for _, ps := range []string{
optionPairToString(optionsTradablePair),
spotTradablePair.String(),
btcPerpInstrument,
futureComboTradablePair.String(),
} {
result, err = e.WSRetrieveMarkPriceHistory(t.Context(), ps, time.Now().Add(-4*time.Hour), time.Now())
require.NoErrorf(t, err, "expected %v, got %v currency pair %v", nil, err, ps)
require.NotNilf(t, result, "expected value not to be nil for pair: %v", ps)
}
}
func TestGetBookSummaryByCurrency(t *testing.T) {
t.Parallel()
var response BookSummaryData
err := json.Unmarshal([]byte(`{ "volume_usd": 0, "volume": 0, "quote_currency": "USD",
"price_change": -11.1896349, "open_interest": 0, "mid_price": null, "mark_price": 3579.73, "low": null,
"last": null, "instrument_name": "BTC-22FEB19", "high": null, "estimated_delivery_price": 3579.73, "creation_timestamp": 1550230036440,
"bid_price": null, "base_currency": "BTC", "ask_price": null}`), &response)
require.NoError(t, err)
_, err = e.GetBookSummaryByCurrency(t.Context(), currency.EMPTYCODE, "future")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.GetBookSummaryByCurrency(t.Context(), currency.BTC, "option")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveBookBySummary(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveBookBySummary(t.Context(), currency.EMPTYCODE, "")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.WSRetrieveBookBySummary(t.Context(), currency.SOL, "")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetBookSummaryByInstrument(t *testing.T) {
t.Parallel()
_, err := e.GetBookSummaryByInstrument(t.Context(), "")
require.ErrorIs(t, err, errInvalidInstrumentName)
for _, ps := range []string{
btcPerpInstrument,
spotTradablePair.String(),
futureComboTradablePair.String(),
optionPairToString(optionsTradablePair),
optionComboPairToString(optionComboTradablePair),
} {
t.Run(ps, func(t *testing.T) {
t.Parallel()
result, err := e.GetBookSummaryByInstrument(t.Context(), ps)
require.NoError(t, err, "GetBookSummaryByInstrument must not error")
require.NotNil(t, result, "result must not be nil")
})
}
}
func TestWSRetrieveBookSummaryByInstrument(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveBookSummaryByInstrument(t.Context(), "")
require.ErrorIs(t, err, errInvalidInstrumentName)
var result []BookSummaryData
for _, ps := range []string{
btcPerpInstrument,
spotTradablePair.String(),
futureComboTradablePair.String(),
optionPairToString(optionsTradablePair),
optionComboPairToString(optionComboTradablePair),
} {
result, err = e.WSRetrieveBookSummaryByInstrument(t.Context(), ps)
require.NoErrorf(t, err, "expected nil, got %v for pair %s", err, ps)
require.NotNilf(t, result, "expected result not to be nil for pair %s", ps)
}
}
func TestGetContractSize(t *testing.T) {
t.Parallel()
_, err := e.GetContractSize(t.Context(), "")
require.ErrorIs(t, err, errInvalidInstrumentName)
result, err := e.GetContractSize(t.Context(), btcPerpInstrument)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveContractSize(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveContractSize(t.Context(), "")
require.ErrorIs(t, err, errInvalidInstrumentName)
result, err := e.WSRetrieveContractSize(t.Context(), btcPerpInstrument)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetCurrencies(t *testing.T) {
t.Parallel()
result, err := e.GetCurrencies(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveCurrencies(t *testing.T) {
t.Parallel()
result, err := e.WSRetrieveCurrencies(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetDeliveryPrices(t *testing.T) {
t.Parallel()
_, err := e.GetDeliveryPrices(t.Context(), "", 0, 5)
require.ErrorIs(t, err, errUnsupportedIndexName)
result, err := e.GetDeliveryPrices(t.Context(), "btc_usd", 0, 5)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveDeliveryPrices(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveDeliveryPrices(t.Context(), "", 0, 5)
require.ErrorIs(t, err, errUnsupportedIndexName)
result, err := e.WSRetrieveDeliveryPrices(t.Context(), "btc_usd", 0, 5)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetFundingChartData(t *testing.T) {
t.Parallel()
// only for perpetual instruments
_, err := e.GetFundingChartData(t.Context(), "", "8h")
require.ErrorIs(t, err, errInvalidInstrumentName)
result, err := e.GetFundingChartData(t.Context(), btcPerpInstrument, "8h")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveFundingChartData(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveFundingChartData(t.Context(), "", "8h")
require.ErrorIs(t, err, errInvalidInstrumentName)
result, err := e.WSRetrieveFundingChartData(t.Context(), btcPerpInstrument, "8h")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetFundingRateHistory(t *testing.T) {
t.Parallel()
_, err := e.GetFundingRateHistory(t.Context(), "", time.Now().Add(-time.Hour), time.Now())
require.ErrorIs(t, err, errInvalidInstrumentName)
result, err := e.GetFundingRateHistory(t.Context(), btcPerpInstrument, time.Now().Add(-time.Hour), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveFundingRateHistory(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveFundingRateHistory(t.Context(), "", time.Now().Add(-time.Hour), time.Now())
require.ErrorIs(t, err, errInvalidInstrumentName)
result, err := e.WSRetrieveFundingRateHistory(t.Context(), btcPerpInstrument, time.Now().Add(-time.Hour), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetFundingRateValue(t *testing.T) {
t.Parallel()
_, err := e.GetFundingRateValue(t.Context(), "", time.Time{}, time.Time{})
require.ErrorIs(t, err, errInvalidInstrumentName)
_, err = e.GetFundingRateValue(t.Context(), btcPerpInstrument, time.Now(), time.Now().Add(-time.Hour*8))
require.ErrorIs(t, err, common.ErrStartAfterEnd)
result, err := e.GetFundingRateValue(t.Context(), btcPerpInstrument, time.Now().Add(-time.Hour*8), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveFundingRateValue(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveFundingRateValue(t.Context(), btcPerpInstrument, time.Now(), time.Now().Add(-time.Hour*8))
require.ErrorIs(t, err, common.ErrStartAfterEnd)
result, err := e.WSRetrieveFundingRateValue(t.Context(), btcPerpInstrument, time.Now().Add(-time.Hour*8), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestHistoricalVolatilityDataUnmarshalJSON(t *testing.T) {
t.Parallel()
data := []byte(`[[1746532800000,33.926694663144644],[1746536400000,33.86888345738641],[1746540000000,33.87689653120242],[1746543600000,33.92229949556179],[1746547200000,33.35430439982866],[1746550800000,33.405720857822644],[1746554400000,33.041661194903895],[1746558000000,33.026907604467596],[1746561600000,33.147012362654635],[1746565200000,32.948314953334105],[1746568800000,32.97264616801311],[1746572400000,32.97051874896058],[1746576000000,33.94405253940284],[1746579600000,34.01745935786804],[1746583200000,34.133772136604854],[1746586800000,33.89032454069847],[1746590400000,34.008502172420556],[1746594000000,34.01444591222428],[1746597600000,34.01154352323321],[1746601200000,33.97800061398224],[1746604800000,33.980501315033024]]`)
var targets []HistoricalVolatilityData
err := json.Unmarshal(data, &targets)
require.NoError(t, err)
require.Len(t, targets, 21)
assert.Equal(t, HistoricalVolatilityData{
Timestamp: types.Time(time.UnixMilli(1746532800000)),
Value: 33.926694663144644,
}, targets[0])
}
func TestGetHistoricalVolatility(t *testing.T) {
t.Parallel()
_, err := e.GetHistoricalVolatility(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.GetHistoricalVolatility(t.Context(), currency.BTC)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveHistoricalVolatility(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveHistoricalVolatility(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.WSRetrieveHistoricalVolatility(t.Context(), currency.SOL)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetCurrencyIndexPrice(t *testing.T) {
t.Parallel()
_, err := e.GetCurrencyIndexPrice(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.GetCurrencyIndexPrice(t.Context(), currency.BTC)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveCurrencyIndexPrice(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveCurrencyIndexPrice(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.WSRetrieveCurrencyIndexPrice(t.Context(), currency.BTC)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetIndexPrice(t *testing.T) {
t.Parallel()
_, err := e.GetIndexPrice(t.Context(), "")
require.ErrorIs(t, err, errUnsupportedIndexName)
result, err := e.GetIndexPrice(t.Context(), "ada_usd")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveIndexPrice(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveIndexPrice(t.Context(), "")
require.ErrorIs(t, err, errUnsupportedIndexName)
result, err := e.WSRetrieveIndexPrice(t.Context(), "ada_usd")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetIndexPriceNames(t *testing.T) {
t.Parallel()
result, err := e.GetIndexPriceNames(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveIndexPriceNames(t *testing.T) {
t.Parallel()
result, err := e.WSRetrieveIndexPriceNames(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetInstrumentData(t *testing.T) {
t.Parallel()
_, err := e.GetInstrument(t.Context(), "")
require.ErrorIs(t, err, errInvalidInstrumentName)
var result *InstrumentData
for assetType, cp := range assetTypeToPairsMap {
result, err = e.GetInstrument(t.Context(), formatPairString(assetType, cp))
require.NoErrorf(t, err, "expected nil, got %v for asset type %s pair %s", err, assetType, cp)
require.NotNilf(t, result, "expected result not to be nil for asset type %s pair %s", assetType, cp)
}
}
func TestWSRetrieveInstrumentData(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveInstrumentData(t.Context(), "")
require.ErrorIs(t, err, errInvalidInstrumentName)
for assetType, cp := range assetTypeToPairsMap {
t.Run(fmt.Sprintf("%s %s", assetType, cp), func(t *testing.T) {
t.Parallel()
result, err := e.WSRetrieveInstrumentData(request.WithVerbose(t.Context()), formatPairString(assetType, cp))
require.NoError(t, err)
require.NotNil(t, result, "result must not be nil")
})
}
}
func TestGetInstruments(t *testing.T) {
t.Parallel()
result, err := e.GetInstruments(t.Context(), currency.EMPTYCODE, "future", false)
require.NoError(t, err)
require.NotNil(t, result)
result, err = e.GetInstruments(t.Context(), currency.BTC, "", false)
require.NoError(t, err)
require.NotNil(t, result)
result, err = e.GetInstruments(t.Context(), currency.BTC, "", true)
require.NoError(t, err)
for a := range result {
require.Falsef(t, result[a].IsActive, "expected expired instrument, but got active instrument %s", result[a].InstrumentName)
}
}
func TestWSRetrieveInstrumentsData(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveInstrumentsData(t.Context(), currency.EMPTYCODE, "", false)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.WSRetrieveInstrumentsData(t.Context(), currency.BTC, "", false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetLastSettlementsByCurrency(t *testing.T) {
t.Parallel()
_, err := e.GetLastSettlementsByCurrency(t.Context(), currency.EMPTYCODE, "delivery", "5", 0, time.Now().Add(-time.Hour))
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.GetLastSettlementsByCurrency(t.Context(), currency.BTC, "delivery", "5", 0, time.Now().Add(-time.Hour))
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveLastSettlementsByCurrency(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveLastSettlementsByCurrency(t.Context(), currency.EMPTYCODE, "delivery", "5", 0, time.Now().Add(-time.Hour))
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.WSRetrieveLastSettlementsByCurrency(t.Context(), currency.BTC, "delivery", "5", 0, time.Now().Add(-time.Hour))
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveLastSettlementsByInstrument(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveLastSettlementsByInstrument(t.Context(), "", "settlement", "5", 0, time.Now().Add(-2*time.Hour))
require.ErrorIs(t, err, errInvalidInstrumentName)
result, err := e.WSRetrieveLastSettlementsByInstrument(t.Context(), formatFuturesTradablePair(futuresTradablePair), "settlement", "5", 0, time.Now().Add(-2*time.Hour))
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetLastSettlementsByInstrument(t *testing.T) {
t.Parallel()
_, err := e.GetLastSettlementsByInstrument(t.Context(), "", "settlement", "5", 0, time.Now().Add(-2*time.Hour))
require.ErrorIs(t, err, errInvalidInstrumentName)
result, err := e.GetLastSettlementsByInstrument(t.Context(), formatFuturesTradablePair(futuresTradablePair), "settlement", "5", 0, time.Now().Add(-2*time.Hour))
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetLastTradesByCurrency(t *testing.T) {
t.Parallel()
_, err := e.GetLastTradesByCurrency(t.Context(), currency.EMPTYCODE, "option", "36798", "36799", "asc", 0, true)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.GetLastTradesByCurrency(t.Context(), currency.BTC, "option", "36798", "36799", "asc", 0, true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveLastTradesByCurrency(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveLastTradesByCurrency(t.Context(), currency.EMPTYCODE, "option", "36798", "36799", "asc", 0, true)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.WSRetrieveLastTradesByCurrency(t.Context(), currency.BTC, "option", "36798", "36799", "asc", 0, true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetLastTradesByCurrencyAndTime(t *testing.T) {
t.Parallel()
_, err := e.GetLastTradesByCurrencyAndTime(t.Context(), currency.EMPTYCODE, "", "", 0, time.Now().Add(-8*time.Hour), time.Now())
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.GetLastTradesByCurrencyAndTime(t.Context(), currency.BTC, "", "", 0, time.Now().Add(-8*time.Hour), time.Now())
require.NoError(t, err)
require.NotNil(t, result)
result, err = e.GetLastTradesByCurrencyAndTime(t.Context(), currency.BTC, "option", "asc", 25, time.Now().Add(-8*time.Hour), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveLastTradesByCurrencyAndTime(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveLastTradesByCurrencyAndTime(t.Context(), currency.EMPTYCODE, "", "", 0, false, time.Now().Add(-8*time.Hour), time.Now())
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.WSRetrieveLastTradesByCurrencyAndTime(t.Context(), currency.BTC, "", "", 0, false, time.Now().Add(-8*time.Hour), time.Now())
require.NoError(t, err)
require.NotNil(t, result)
result, err = e.WSRetrieveLastTradesByCurrencyAndTime(t.Context(), currency.BTC, "option", "asc", 25, false, time.Now().Add(-8*time.Hour), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetLastTradesByInstrument(t *testing.T) {
t.Parallel()
_, err := e.GetLastTradesByInstrument(t.Context(), "", "", "", "", 0, false)
require.ErrorIs(t, err, errInvalidInstrumentName)
for assetType, cp := range assetTypeToPairsMap {
result, err := e.GetLastTradesByInstrument(t.Context(), formatPairString(assetType, cp), "30500", "31500", "desc", 0, true)
require.NoErrorf(t, err, "expected %v, got %v currency asset %v pair %v", nil, err, assetType, cp)
require.NotNilf(t, result, "expected value not to be nil for asset %v pair: %v", assetType, cp)
}
}
func TestWSRetrieveLastTradesByInstrument(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveLastTradesByInstrument(t.Context(), "", "", "", "", 0, false)
require.ErrorIs(t, err, errInvalidInstrumentName)
for assetType, cp := range assetTypeToPairsMap {
result, err := e.WSRetrieveLastTradesByInstrument(t.Context(), formatPairString(assetType, cp), "30500", "31500", "desc", 0, true)
require.NoErrorf(t, err, "expected %v, got %v currency asset %v pair %v", nil, err, assetType, cp)
require.NotNilf(t, result, "expected value not to be nil for asset %v pair: %v", assetType, cp)
}
}
func TestGetLastTradesByInstrumentAndTime(t *testing.T) {
t.Parallel()
_, err := e.GetLastTradesByInstrumentAndTime(t.Context(), "", "", 0, time.Now().Add(-8*time.Hour), time.Now())
require.ErrorIs(t, err, errInvalidInstrumentName)
for assetType, cp := range assetTypeToPairsMap {
result, err := e.GetLastTradesByInstrumentAndTime(t.Context(), formatPairString(assetType, cp), "", 0, time.Now().Add(-8*time.Hour), time.Now())
require.NoErrorf(t, err, "expected %v, got %v currency pair %v", nil, err, cp)
require.NotNilf(t, result, "expected value not to be nil for pair: %v", cp)
}
}
func TestWSRetrieveLastTradesByInstrumentAndTime(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveLastTradesByInstrumentAndTime(t.Context(), "", "", 0, false, time.Now().Add(-8*time.Hour), time.Now())
require.ErrorIs(t, err, errInvalidInstrumentName)
for assetType, cp := range assetTypeToPairsMap {
result, err := e.WSRetrieveLastTradesByInstrumentAndTime(t.Context(), formatPairString(assetType, cp), "", 0, true, time.Now().Add(-8*time.Hour), time.Now())
require.NoErrorf(t, err, "expected %v, got %v currency pair %v", nil, err, cp)
require.NotNilf(t, result, "expected value not to be nil for pair: %v", cp)
}
}
func TestWSProcessTrades(t *testing.T) {
t.Parallel()
e := new(Exchange) //nolint:govet // Intentional shadow
require.NoError(t, testexch.Setup(e), "Setup instance must not error")
testexch.FixtureToDataHandler(t, "testdata/wsAllTrades.json", e.wsHandleData)
close(e.Websocket.DataHandler)
a, p, err := getAssetPairByInstrument("BTC-PERPETUAL")
require.NoError(t, err, "getAssetPairByInstrument must not error")
exp := []trade.Data{
{
Exchange: e.Name,
CurrencyPair: p,
Timestamp: time.UnixMilli(1742627465811).UTC(),
Price: 84295.5,
Amount: 8430.0,
Side: order.Buy,
TID: "356130997",
AssetType: a,
},
{
Exchange: e.Name,
CurrencyPair: p,
Timestamp: time.UnixMilli(1742627361899).UTC(),
Price: 84319.0,
Amount: 580.0,
Side: order.Sell,
TID: "356130979",
AssetType: a,
},
}
require.Len(t, e.Websocket.DataHandler, len(exp), "Must see the correct number of trades")
for resp := range e.Websocket.DataHandler {
switch v := resp.(type) {
case trade.Data:
i := 1 - len(e.Websocket.DataHandler)
require.Equalf(t, exp[i], v, "Trade [%d] must be correct", i)
case error:
t.Error(v)
default:
t.Errorf("Unexpected type in DataHandler: %T(%s)", v, v)
}
}
}
func TestGetOrderbookData(t *testing.T) {
t.Parallel()
_, err := e.GetOrderbook(t.Context(), "", 0)
require.ErrorIs(t, err, errInvalidInstrumentName)
var result *Orderbook
for assetType, cp := range assetTypeToPairsMap {
result, err = e.GetOrderbook(t.Context(), formatPairString(assetType, cp), 0)
require.NoErrorf(t, err, "expected %v, got %v currency pair %v", nil, err, cp)
require.NotNilf(t, result, "expected value not to be nil for pair: %v", cp)
}
}
func TestWSRetrieveOrderbookData(t *testing.T) {
t.Parallel()
if !e.Websocket.IsConnected() {
t.Skip("websocket is not connected")
}
_, err := e.WSRetrieveOrderbookData(t.Context(), "", 0)
require.ErrorIs(t, err, errInvalidInstrumentName)
var result *Orderbook
for assetType, cp := range assetTypeToPairsMap {
result, err = e.WSRetrieveOrderbookData(t.Context(), formatPairString(assetType, cp), 0)
require.NoErrorf(t, err, "expected %v, got %v currency pair %v", nil, err, cp)
require.NotNilf(t, result, "expected value not to be nil for pair: %v", cp)
}
}
func TestGetOrderbookByInstrumentID(t *testing.T) {
t.Parallel()
combos, err := e.GetComboIDs(t.Context(), currency.BTC, "")
require.NoError(t, err)
if len(combos) == 0 {
t.Skip("no combo instance found for currency BTC")
}
_, err = e.GetOrderbookByInstrumentID(t.Context(), 0, 50)
require.ErrorIs(t, err, errInvalidInstrumentID)
comboD, err := e.GetComboDetails(t.Context(), combos[0])
require.NoError(t, err)
require.NotNil(t, comboD)
result, err := e.GetOrderbookByInstrumentID(t.Context(), comboD.InstrumentID, 50)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveOrderbookByInstrumentID(t *testing.T) {
t.Parallel()
combos, err := e.WSRetrieveComboIDs(t.Context(), currency.BTC, "")
require.NoError(t, err)
if len(combos) == 0 {
t.Skip("no combo instance found for currency BTC")
}
_, err = e.WSRetrieveOrderbookByInstrumentID(t.Context(), 0, 50)
require.ErrorIs(t, err, errInvalidInstrumentID)
comboD, err := e.WSRetrieveComboDetails(t.Context(), combos[0])
require.NoError(t, err)
require.NotNil(t, comboD)
result, err := e.WSRetrieveOrderbookByInstrumentID(t.Context(), comboD.InstrumentID, 50)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetSupportedIndexNames(t *testing.T) {
t.Parallel()
result, err := e.GetSupportedIndexNames(t.Context(), "derivative")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsRetrieveSupportedIndexNames(t *testing.T) {
t.Parallel()
result, err := e.WsRetrieveSupportedIndexNames(t.Context(), "derivative")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetTradeVolumes(t *testing.T) {
t.Parallel()
result, err := e.GetTradeVolumes(t.Context(), false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveTradeVolumes(t *testing.T) {
t.Parallel()
result, err := e.WSRetrieveTradeVolumes(t.Context(), false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetTradingViewChartData(t *testing.T) {
t.Parallel()
_, err := e.GetTradingViewChart(t.Context(), "", "60", time.Now().Add(-time.Hour), time.Now())
require.ErrorIs(t, err, errInvalidInstrumentName)
result, err := e.GetTradingViewChart(t.Context(), btcPerpInstrument, "60", time.Now().Add(-time.Hour), time.Now())
require.NoError(t, err)
require.NotNil(t, result)
result, err = e.GetTradingViewChart(t.Context(), spotTradablePair.String(), "60", time.Now().Add(-time.Hour), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrievesTradingViewChartData(t *testing.T) {
t.Parallel()
_, err := e.WSRetrievesTradingViewChartData(t.Context(), "", "60", time.Now().Add(-time.Hour), time.Now())
require.ErrorIs(t, err, errInvalidInstrumentName)
result, err := e.WSRetrievesTradingViewChartData(t.Context(), btcPerpInstrument, "60", time.Now().Add(-time.Hour), time.Now())
require.NoError(t, err)
require.NotNil(t, result)
result, err = e.WSRetrievesTradingViewChartData(t.Context(), spotTradablePair.String(), "60", time.Now().Add(-time.Hour), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetVolatilityIndexData(t *testing.T) {
t.Parallel()
_, err := e.GetVolatilityIndex(t.Context(), currency.EMPTYCODE, "60", time.Now().Add(-time.Hour), time.Now())
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.GetVolatilityIndex(t.Context(), currency.BTC, "", time.Now().Add(-time.Hour), time.Now())
require.ErrorIs(t, err, errResolutionNotSet)
_, err = e.GetVolatilityIndex(t.Context(), currency.BTC, "60", time.Now(), time.Now().Add(-time.Hour))
require.ErrorIs(t, err, common.ErrStartAfterEnd)
result, err := e.GetVolatilityIndex(t.Context(), currency.BTC, "60", time.Now().Add(-time.Hour), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveVolatilityIndexData(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveVolatilityIndexData(t.Context(), currency.EMPTYCODE, "60", time.Now().Add(-time.Hour), time.Now())
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.WSRetrieveVolatilityIndexData(t.Context(), currency.BTC, "", time.Now().Add(-time.Hour), time.Now())
require.ErrorIs(t, err, errResolutionNotSet)
_, err = e.WSRetrieveVolatilityIndexData(t.Context(), currency.BTC, "60", time.Now(), time.Now().Add(-time.Hour))
require.ErrorIs(t, err, common.ErrStartAfterEnd)
result, err := e.WSRetrieveVolatilityIndexData(t.Context(), currency.BTC, "60", time.Now().Add(-time.Hour), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetPublicTicker(t *testing.T) {
t.Parallel()
_, err := e.GetPublicTicker(t.Context(), "")
require.ErrorIs(t, err, errInvalidInstrumentName)
result, err := e.GetPublicTicker(t.Context(), btcPerpInstrument)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrievePublicTicker(t *testing.T) {
t.Parallel()
_, err := e.WSRetrievePublicTicker(t.Context(), "")
require.ErrorIs(t, err, errInvalidInstrumentName)
result, err := e.WSRetrievePublicTicker(t.Context(), btcPerpInstrument)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetAccountSummary(t *testing.T) {
t.Parallel()
_, err := e.GetAccountSummary(t.Context(), currency.EMPTYCODE, false)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetAccountSummary(t.Context(), currency.BTC, false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveAccountSummary(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveAccountSummary(t.Context(), currency.EMPTYCODE, false)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveAccountSummary(t.Context(), currency.BTC, false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestCancelTransferByID(t *testing.T) {
t.Parallel()
_, err := e.CancelTransferByID(t.Context(), currency.EMPTYCODE, "", 23487)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.CancelTransferByID(t.Context(), currency.BTC, "", 0)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.CancelTransferByID(t.Context(), currency.BTC, "", 23487)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSCancelTransferByID(t *testing.T) {
t.Parallel()
_, err := e.WSCancelTransferByID(t.Context(), currency.EMPTYCODE, "", 23487)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.WSCancelTransferByID(t.Context(), currency.BTC, "", 0)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSCancelTransferByID(t.Context(), currency.BTC, "", 23487)
require.NoError(t, err)
assert.NotNil(t, result)
}
const getTransferResponseJSON = `{"count": 2, "data":[{"amount": 0.2, "created_timestamp": 1550579457727, "currency": "BTC", "direction": "payment", "id": 2, "other_side": "2MzyQc5Tkik61kJbEpJV5D5H9VfWHZK9Sgy", "state": "prepared", "type": "user", "updated_timestamp": 1550579457727}, { "amount": 0.3, "created_timestamp": 1550579255800, "currency": "BTC", "direction": "payment", "id": 1, "other_side": "new_user_1_1", "state": "confirmed", "type": "subaccount", "updated_timestamp": 1550579255800} ] }`
func TestGetTransfers(t *testing.T) {
t.Parallel()
var resp *TransfersData
err := json.Unmarshal([]byte(getTransferResponseJSON), &resp)
require.NoError(t, err)
_, err = e.GetTransfers(t.Context(), currency.EMPTYCODE, 0, 0)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetTransfers(t.Context(), currency.BTC, 0, 0)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveTransfers(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveTransfers(t.Context(), currency.EMPTYCODE, 0, 0)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveTransfers(t.Context(), currency.BTC, 0, 0)
require.NoError(t, err)
assert.NotNil(t, result)
}
const cancelWithdrawlPushDataJSON = `{"address": "2NBqqD5GRJ8wHy1PYyCXTe9ke5226FhavBz", "amount": 0.5, "confirmed_timestamp": null, "created_timestamp": 1550571443070, "currency": "BTC", "fee": 0.0001, "id": 1, "priority": 0.15, "state": "cancelled", "transaction_id": null, "updated_timestamp": 1550571443070}`
func TestCancelWithdrawal(t *testing.T) {
t.Parallel()
var resp *CancelWithdrawalData
err := json.Unmarshal([]byte(cancelWithdrawlPushDataJSON), &resp)
require.NoError(t, err)
_, err = e.CancelWithdrawal(t.Context(), currency.EMPTYCODE, 123844)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.CancelWithdrawal(t.Context(), currency.BTC, 0)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.CancelWithdrawal(t.Context(), currency.BTC, 123844)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSCancelWithdrawal(t *testing.T) {
t.Parallel()
_, err := e.WSCancelWithdrawal(t.Context(), currency.EMPTYCODE, 123844)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.WSCancelWithdrawal(t.Context(), currency.BTC, 0)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSCancelWithdrawal(t.Context(), currency.BTC, 123844)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestCreateDepositAddress(t *testing.T) {
t.Parallel()
_, err := e.CreateDepositAddress(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.CreateDepositAddress(t.Context(), currency.SOL)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSCreateDepositAddress(t *testing.T) {
t.Parallel()
_, err := e.WSCreateDepositAddress(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSCreateDepositAddress(t.Context(), currency.SOL)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetCurrentDepositAddress(t *testing.T) {
t.Parallel()
_, err := e.GetCurrentDepositAddress(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetCurrentDepositAddress(t.Context(), currency.ETH)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveCurrentDepositAddress(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveCurrentDepositAddress(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveCurrentDepositAddress(t.Context(), currency.ETH)
require.NoError(t, err)
assert.NotNil(t, result)
}
const getDepositPushDataJSON = `{"count": 1, "data": [ { "address": "2N35qDKDY22zmJq9eSyiAerMD4enJ1xx6ax", "amount": 5, "currency": "BTC", "received_timestamp": 1549295017670, "state": "completed", "transaction_id": "230669110fdaf0a0dbcdc079b6b8b43d5af29cc73683835b9bc6b3406c065fda", "updated_timestamp": 1549295130159} ] }`
func TestGetDeposits(t *testing.T) {
t.Parallel()
var resp *DepositsData
err := json.Unmarshal([]byte(getDepositPushDataJSON), &resp)
require.NoError(t, err)
_, err = e.GetDeposits(t.Context(), currency.EMPTYCODE, 25, 0)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetDeposits(t.Context(), currency.BTC, 25, 0)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveDeposits(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveDeposits(t.Context(), currency.EMPTYCODE, 25, 0)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveDeposits(t.Context(), currency.BTC, 25, 0)
require.NoError(t, err)
assert.NotNil(t, result)
}
const getWithdrawalResponseJSON = `{"count": 1, "data": [ { "address": "2NBqqD5GRJ8wHy1PYyCXTe9ke5226FhavBz", "amount": 0.5, "confirmed_timestamp": null, "created_timestamp": 1550571443070, "currency": "BTC", "fee": 0.0001, "id": 1, "priority": 0.15, "state": "unconfirmed", "transaction_id": null, "updated_timestamp": 1550571443070} ] }`
func TestGetWithdrawals(t *testing.T) {
t.Parallel()
var resp *WithdrawalsData
err := json.Unmarshal([]byte(getWithdrawalResponseJSON), &resp)
require.NoError(t, err)
_, err = e.GetWithdrawals(t.Context(), currency.EMPTYCODE, 25, 0)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetWithdrawals(t.Context(), currency.BTC, 25, 0)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveWithdrawals(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveWithdrawals(t.Context(), currency.EMPTYCODE, 25, 0)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveWithdrawals(t.Context(), currency.BTC, 25, 0)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSubmitTransferBetweenSubAccounts(t *testing.T) {
t.Parallel()
_, err := e.SubmitTransferBetweenSubAccounts(t.Context(), currency.EMPTYCODE, 12345, 2, "")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.SubmitTransferBetweenSubAccounts(t.Context(), currency.EURR, 0, 2, "")
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.SubmitTransferBetweenSubAccounts(t.Context(), currency.EURR, 12345, -1, "")
require.ErrorIs(t, err, errInvalidDestinationID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitTransferBetweenSubAccounts(t.Context(), currency.EURR, 12345, 4, "")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsSubmitTransferBetweenSubAccounts(t *testing.T) {
t.Parallel()
_, err := e.WsSubmitTransferBetweenSubAccounts(t.Context(), currency.EMPTYCODE, 12345, 2, "")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.WsSubmitTransferBetweenSubAccounts(t.Context(), currency.EURR, 0, 2, "")
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.WsSubmitTransferBetweenSubAccounts(t.Context(), currency.EURR, 12345, -1, "")
require.ErrorIs(t, err, errInvalidDestinationID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WsSubmitTransferBetweenSubAccounts(t.Context(), currency.EURR, 12345, 2, "")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSubmitTransferToSubAccount(t *testing.T) {
t.Parallel()
_, err := e.SubmitTransferToSubAccount(t.Context(), currency.EMPTYCODE, 0.01, 13434)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.SubmitTransferToSubAccount(t.Context(), currency.BTC, 0, 13434)
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.SubmitTransferToSubAccount(t.Context(), currency.BTC, 0.01, 0)
require.ErrorIs(t, err, errInvalidDestinationID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitTransferToSubAccount(t.Context(), currency.BTC, 0.01, 13434)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitTransferToSubAccount(t *testing.T) {
t.Parallel()
_, err := e.WSSubmitTransferToSubAccount(t.Context(), currency.EMPTYCODE, 0.01, 13434)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.WSSubmitTransferToSubAccount(t.Context(), currency.BTC, 0, 13434)
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.WSSubmitTransferToSubAccount(t.Context(), currency.BTC, 0.01, 0)
require.ErrorIs(t, err, errInvalidDestinationID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitTransferToSubAccount(t.Context(), currency.BTC, 0.01, 13434)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSubmitTransferToUser(t *testing.T) {
t.Parallel()
_, err := e.SubmitTransferToUser(t.Context(), currency.EMPTYCODE, "", "0x4aa0753d798d668056920094d65321a8e8913e26", 0.001)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.SubmitTransferToUser(t.Context(), currency.BTC, "", "0x4aa0753d798d668056920094d65321a8e8913e26", 0)
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.SubmitTransferToUser(t.Context(), currency.BTC, "", "", 0.001)
require.ErrorIs(t, err, errInvalidCryptoAddress)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitTransferToUser(t.Context(), currency.BTC, "", "13434", 0.001)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitTransferToUser(t *testing.T) {
t.Parallel()
_, err := e.WSSubmitTransferToUser(t.Context(), currency.EMPTYCODE, "", "0x4aa0753d798d668056920094d65321a8e8913e26", 0.001)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.WSSubmitTransferToUser(t.Context(), currency.BTC, "", "0x4aa0753d798d668056920094d65321a8e8913e26", 0)
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.WSSubmitTransferToUser(t.Context(), currency.BTC, "", "", 0.001)
require.ErrorIs(t, err, errInvalidCryptoAddress)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitTransferToUser(t.Context(), currency.BTC, "", "", 0.001)
require.NoError(t, err)
assert.NotNil(t, result)
}
const submitWithdrawalResponseJSON = `{"address": "2NBqqD5GRJ8wHy1PYyCXTe9ke5226FhavBz", "amount": 0.4, "confirmed_timestamp": null, "created_timestamp": 1550574558607, "currency": "BTC", "fee": 0.0001, "id": 4, "priority": 1, "state": "unconfirmed", "transaction_id": null, "updated_timestamp": 1550574558607}`
func TestSubmitWithdraw(t *testing.T) {
t.Parallel()
var resp *WithdrawData
err := json.Unmarshal([]byte(submitWithdrawalResponseJSON), &resp)
require.NoError(t, err)
_, err = e.SubmitWithdraw(t.Context(), currency.EMPTYCODE, core.BitcoinDonationAddress, "", 0.001)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.SubmitWithdraw(t.Context(), currency.BTC, core.BitcoinDonationAddress, "", 0)
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.SubmitWithdraw(t.Context(), currency.BTC, "", "", 0.001)
require.ErrorIs(t, err, errInvalidCryptoAddress)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitWithdraw(t.Context(), currency.BTC, core.BitcoinDonationAddress, "", 0.001)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitWithdraw(t *testing.T) {
_, err := e.WSSubmitWithdraw(t.Context(), currency.EMPTYCODE, core.BitcoinDonationAddress, "", 0.001)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.WSSubmitWithdraw(t.Context(), currency.BTC, core.BitcoinDonationAddress, "", 0)
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.WSSubmitWithdraw(t.Context(), currency.BTC, "", "", 0.001)
require.ErrorIs(t, err, errInvalidCryptoAddress)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitWithdraw(t.Context(), currency.BTC, core.BitcoinDonationAddress, "", 0.001)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetAnnouncements(t *testing.T) {
t.Parallel()
result, err := e.GetAnnouncements(t.Context(), time.Now(), 5)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveAnnouncements(t *testing.T) {
t.Parallel()
result, err := e.WSRetrieveAnnouncements(t.Context(), time.Now(), 5)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetAccessLog(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetAccessLog(t.Context(), 0, 0)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveAccessLog(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveAccessLog(t.Context(), 0, 0)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestChangeAPIKeyName(t *testing.T) {
t.Parallel()
_, err := e.ChangeAPIKeyName(t.Context(), 0, "TestKey123")
require.ErrorIs(t, err, errInvalidID)
_, err = e.ChangeAPIKeyName(t.Context(), 2, "TestKey123$")
require.ErrorIs(t, err, errUnacceptableAPIKey)
_, err = e.ChangeAPIKeyName(t.Context(), 2, "#$#")
require.ErrorIs(t, err, errUnacceptableAPIKey)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.ChangeAPIKeyName(t.Context(), 1, "TestKey123")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSChangeAPIKeyName(t *testing.T) {
t.Parallel()
_, err := e.WSChangeAPIKeyName(t.Context(), 0, "TestKey123")
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.WSChangeAPIKeyName(t.Context(), 1, "TestKey123")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestChangeMarginModel(t *testing.T) {
t.Parallel()
_, err := e.ChangeMarginModel(t.Context(), 2, "", false)
require.ErrorIs(t, err, errInvalidMarginModel)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.ChangeMarginModel(t.Context(), 2, "segregated_pm", false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsChangeMarginModel(t *testing.T) {
t.Parallel()
_, err := e.WsChangeMarginModel(t.Context(), 2, "", false)
require.ErrorIs(t, err, errInvalidMarginModel)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WsChangeMarginModel(t.Context(), 2, "segregated_pm", false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestChangeScopeInAPIKey(t *testing.T) {
t.Parallel()
_, err := e.ChangeScopeInAPIKey(t.Context(), -1, "account:read_write")
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.ChangeScopeInAPIKey(t.Context(), 1, "account:read_write")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSChangeScopeInAPIKey(t *testing.T) {
t.Parallel()
_, err := e.WSChangeScopeInAPIKey(t.Context(), 0, "account:read_write")
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.WSChangeScopeInAPIKey(t.Context(), 1, "account:read_write")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestChangeSubAccountName(t *testing.T) {
t.Parallel()
err := e.ChangeSubAccountName(t.Context(), 0, "new_sub")
require.ErrorIs(t, err, errInvalidID)
err = e.ChangeSubAccountName(t.Context(), 312313, "")
require.ErrorIs(t, err, errInvalidUsername)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err = e.ChangeSubAccountName(t.Context(), 1, "new_sub")
assert.NoError(t, err)
}
func TestWSChangeSubAccountName(t *testing.T) {
t.Parallel()
err := e.WSChangeSubAccountName(t.Context(), 0, "new_sub")
require.ErrorIs(t, err, errInvalidID)
err = e.WSChangeSubAccountName(t.Context(), 312313, "")
require.ErrorIs(t, err, errInvalidUsername)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err = e.WSChangeSubAccountName(t.Context(), 1, "new_sub")
assert.NoError(t, err)
}
func TestCreateAPIKey(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.CreateAPIKey(t.Context(), "account:read_write", "new_sub", false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSCreateAPIKey(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.WSCreateAPIKey(t.Context(), "account:read_write", "new_sub", false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestCreateSubAccount(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.CreateSubAccount(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSCreateSubAccount(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSCreateSubAccount(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestDisableAPIKey(t *testing.T) {
t.Parallel()
_, err := e.DisableAPIKey(t.Context(), 0)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.DisableAPIKey(t.Context(), 1)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSDisableAPIKey(t *testing.T) {
t.Parallel()
_, err := e.WSDisableAPIKey(t.Context(), 0)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.WSDisableAPIKey(t.Context(), 1)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestEditAPIKey(t *testing.T) {
t.Parallel()
_, err := e.EditAPIKey(t.Context(), 0, "trade", "", false, []string{"read", "read_write"}, []string{})
require.ErrorIs(t, err, errInvalidAPIKeyID)
_, err = e.EditAPIKey(t.Context(), 1234, "", "", false, []string{"read", "read_write"}, []string{})
require.ErrorIs(t, err, errMaxScopeIsRequired)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.EditAPIKey(t.Context(), 1234, "trade", "", false, []string{"read", "read_write"}, []string{})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsEditAPIKey(t *testing.T) {
t.Parallel()
_, err := e.WsEditAPIKey(t.Context(), 0, "trade", "", false, []string{"read", "read_write"}, []string{})
require.ErrorIs(t, err, errInvalidAPIKeyID)
_, err = e.WsEditAPIKey(t.Context(), 1234, "", "", false, []string{"read", "read_write"}, []string{})
require.ErrorIs(t, err, errMaxScopeIsRequired)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.WsEditAPIKey(t.Context(), 1234, "trade", "", false, []string{"read", "read_write"}, []string{})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestEnableAffiliateProgram(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err := e.EnableAffiliateProgram(t.Context())
assert.NoError(t, err)
}
func TestWSEnableAffiliateProgram(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err := e.WSEnableAffiliateProgram(t.Context())
assert.NoError(t, err)
}
func TestEnableAPIKey(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.EnableAPIKey(t.Context(), 1)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSEnableAPIKey(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.WSEnableAPIKey(t.Context(), 1)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetAffiliateProgramInfo(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetAffiliateProgramInfo(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveAffiliateProgramInfo(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveAffiliateProgramInfo(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetEmailLanguage(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetEmailLanguage(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveEmailLanguage(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveEmailLanguage(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetNewAnnouncements(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetNewAnnouncements(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveNewAnnouncements(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveNewAnnouncements(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetPosition(t *testing.T) {
t.Parallel()
_, err := e.GetPosition(t.Context(), "")
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetPosition(t.Context(), btcPerpInstrument)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrievePosition(t *testing.T) {
t.Parallel()
_, err := e.WSRetrievePosition(t.Context(), "")
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrievePosition(t.Context(), btcPerpInstrument)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetSubAccounts(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetSubAccounts(t.Context(), false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveSubAccounts(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveSubAccounts(t.Context(), false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetSubAccountDetails(t *testing.T) {
t.Parallel()
_, err := e.GetSubAccountDetails(t.Context(), currency.EMPTYCODE, false)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetSubAccountDetails(t.Context(), currency.BTC, false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveSubAccountDetails(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveSubAccountDetails(t.Context(), currency.EMPTYCODE, false)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveSubAccountDetails(t.Context(), currency.BTC, false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetPositions(t *testing.T) {
t.Parallel()
_, err := e.GetPositions(t.Context(), currency.EMPTYCODE, "option")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetPositions(t.Context(), currency.BTC, "option")
require.NoError(t, err)
require.NotNil(t, result)
result, err = e.GetPositions(t.Context(), currency.ETH, "")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrievePositions(t *testing.T) {
t.Parallel()
_, err := e.WSRetrievePositions(t.Context(), currency.EMPTYCODE, "option")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrievePositions(t.Context(), currency.BTC, "option")
require.NoError(t, err)
require.NotNil(t, result)
result, err = e.WSRetrievePositions(t.Context(), currency.ETH, "")
require.NoError(t, err)
assert.NotNil(t, result)
}
const getTransactionLogResponseJSON = `{"logs": [ { "username": "TestUser", "user_seq": 6009, "user_id": 7, "type": "transfer", "trade_id": null, "timestamp": 1613659830333, "side": "-", "price": null, "position": null, "order_id": null, "interest_pl": null, "instrument_name": null, "info": { "transfer_type": "subaccount", "other_user_id": 27, "other_user": "Subaccount" }, "id": 61312, "equity": 3000.9275869, "currency": "BTC", "commission": 0, "change": -2.5, "cashflow": -2.5, "balance": 3001.22270418 } ], "continuation": 61282 }`
func TestGetTransactionLog(t *testing.T) {
t.Parallel()
var resp *TransactionsData
err := json.Unmarshal([]byte(getTransactionLogResponseJSON), &resp)
require.NoError(t, err)
_, err = e.GetTransactionLog(t.Context(), currency.EMPTYCODE, "trade", time.Now().Add(-24*time.Hour), time.Now(), 5, 0)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetTransactionLog(t.Context(), currency.BTC, "trade", time.Now().Add(-24*time.Hour), time.Now(), 5, 0)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveTransactionLog(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveTransactionLog(t.Context(), currency.EMPTYCODE, "trade", time.Now().Add(-24*time.Hour), time.Now(), 5, 0)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveTransactionLog(t.Context(), currency.BTC, "trade", time.Now().Add(-24*time.Hour), time.Now(), 5, 0)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetUserLocks(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetUserLocks(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveUserLocks(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveUserLocks(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestListAPIKeys(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.ListAPIKeys(t.Context(), "")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSListAPIKeys(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSListAPIKeys(t.Context(), "")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetCustodyAccounts(t *testing.T) {
t.Parallel()
_, err := e.GetCustodyAccounts(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetCustodyAccounts(t.Context(), currency.BTC)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsRetrieveCustodyAccounts(t *testing.T) {
t.Parallel()
_, err := e.WsRetrieveCustodyAccounts(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WsRetrieveCustodyAccounts(t.Context(), currency.BTC)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestRemoveAPIKey(t *testing.T) {
t.Parallel()
err := e.RemoveAPIKey(t.Context(), 0)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
err = e.RemoveAPIKey(t.Context(), 1)
assert.NoError(t, err)
}
func TestWSRemoveAPIKey(t *testing.T) {
t.Parallel()
err := e.WSRemoveAPIKey(t.Context(), 0)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
err = e.WSRemoveAPIKey(t.Context(), 1)
assert.NoError(t, err)
}
func TestRemoveSubAccount(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
err := e.RemoveSubAccount(t.Context(), 1)
assert.NoError(t, err)
}
func TestWSRemoveSubAccount(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
err := e.WSRemoveSubAccount(t.Context(), 1)
assert.NoError(t, err)
}
func TestResetAPIKey(t *testing.T) {
t.Parallel()
_, err := e.ResetAPIKey(t.Context(), 0)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.ResetAPIKey(t.Context(), 1)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSResetAPIKey(t *testing.T) {
t.Parallel()
err := e.WSResetAPIKey(t.Context(), 0)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
err = e.WSResetAPIKey(t.Context(), 1)
assert.NoError(t, err)
}
func TestSetAnnouncementAsRead(t *testing.T) {
t.Parallel()
err := e.SetAnnouncementAsRead(t.Context(), 0)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err = e.SetAnnouncementAsRead(t.Context(), 1)
assert.NoError(t, err)
}
func TestSetEmailForSubAccount(t *testing.T) {
t.Parallel()
err := e.SetEmailForSubAccount(t.Context(), 0, "wrongemail@wrongemail.com")
require.ErrorIs(t, err, errInvalidID)
err = e.SetEmailForSubAccount(t.Context(), 1, "")
require.ErrorIs(t, err, errInvalidEmailAddress)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err = e.SetEmailForSubAccount(t.Context(), 1, "wrongemail@wrongemail.com")
assert.NoError(t, err)
}
func TestWSSetEmailForSubAccount(t *testing.T) {
t.Parallel()
err := e.WSSetEmailForSubAccount(t.Context(), 0, "wrongemail@wrongemail.com")
require.ErrorIs(t, err, errInvalidID)
err = e.WSSetEmailForSubAccount(t.Context(), 1, "")
require.ErrorIs(t, err, errInvalidEmailAddress)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err = e.WSSetEmailForSubAccount(t.Context(), 1, "wrongemail@wrongemail.com")
assert.NoError(t, err)
}
func TestSetEmailLanguage(t *testing.T) {
t.Parallel()
err := e.SetEmailLanguage(t.Context(), "")
require.ErrorIs(t, err, errLanguageIsRequired)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err = e.SetEmailLanguage(t.Context(), "en")
assert.NoError(t, err)
}
func TestWSSetEmailLanguage(t *testing.T) {
t.Parallel()
err := e.WSSetEmailLanguage(t.Context(), "")
require.ErrorIs(t, err, errLanguageIsRequired)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err = e.WSSetEmailLanguage(t.Context(), "en")
assert.NoError(t, err)
}
func TestSetSelfTradingConfig(t *testing.T) {
t.Parallel()
_, err := e.SetSelfTradingConfig(t.Context(), "", false)
require.ErrorIs(t, err, errTradeModeIsRequired)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SetSelfTradingConfig(t.Context(), "reject_taker", false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsSetSelfTradingConfig(t *testing.T) {
t.Parallel()
_, err := e.WsSetSelfTradingConfig(t.Context(), "", false)
require.ErrorIs(t, err, errTradeModeIsRequired)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WsSetSelfTradingConfig(t.Context(), "reject_taker", false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestToggleNotificationsFromSubAccount(t *testing.T) {
t.Parallel()
err := e.ToggleNotificationsFromSubAccount(t.Context(), 0, false)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err = e.ToggleNotificationsFromSubAccount(t.Context(), 1, false)
assert.NoError(t, err)
}
func TestWSToggleNotificationsFromSubAccount(t *testing.T) {
t.Parallel()
err := e.WSToggleNotificationsFromSubAccount(t.Context(), 0, false)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err = e.WSToggleNotificationsFromSubAccount(t.Context(), 1, false)
assert.NoError(t, err)
}
func TestTogglePortfolioMargining(t *testing.T) {
t.Parallel()
_, err := e.TogglePortfolioMargining(t.Context(), 0, false, false)
require.ErrorIs(t, err, errUserIDRequired)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.TogglePortfolioMargining(t.Context(), 1234, false, false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSTogglePortfolioMargining(t *testing.T) {
t.Parallel()
_, err := e.WSTogglePortfolioMargining(t.Context(), 0, false, false)
require.ErrorIs(t, err, errUserIDRequired)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSTogglePortfolioMargining(t.Context(), 1234, false, false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestToggleSubAccountLogin(t *testing.T) {
t.Parallel()
err := e.ToggleSubAccountLogin(t.Context(), -1, false)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err = e.ToggleSubAccountLogin(t.Context(), 1, false)
assert.NoError(t, err)
}
func TestWSToggleSubAccountLogin(t *testing.T) {
t.Parallel()
err := e.WSToggleSubAccountLogin(t.Context(), -1, false)
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err = e.WSToggleSubAccountLogin(t.Context(), 1, false)
assert.NoError(t, err)
}
func TestSubmitBuy(t *testing.T) {
t.Parallel()
pairs, err := e.GetEnabledPairs(asset.Futures)
require.NoError(t, err)
_, err = e.SubmitBuy(t.Context(), &OrderBuyAndSellParams{})
require.ErrorIs(t, err, common.ErrNilPointer)
_, err = e.SubmitBuy(t.Context(), &OrderBuyAndSellParams{
Instrument: "", OrderType: "limit",
Label: "testOrder", TimeInForce: "",
Trigger: "", Advanced: "",
Amount: 30, Price: 500000,
MaxShow: 0, TriggerPrice: 0,
})
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitBuy(t.Context(), &OrderBuyAndSellParams{
Instrument: pairs[0].String(), OrderType: "limit",
Label: "testOrder", TimeInForce: "",
Trigger: "", Advanced: "",
Amount: 30, Price: 500000,
MaxShow: 0, TriggerPrice: 0,
PostOnly: false, RejectPostOnly: false,
ReduceOnly: false, MMP: false,
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitBuy(t *testing.T) {
t.Parallel()
_, err := e.WSSubmitBuy(t.Context(), &OrderBuyAndSellParams{})
require.ErrorIs(t, err, common.ErrNilPointer)
_, err = e.WSSubmitBuy(t.Context(), &OrderBuyAndSellParams{
Instrument: "", OrderType: "limit",
Label: "testOrder", TimeInForce: "",
Trigger: "", Advanced: "",
Amount: 30, Price: 500000,
MaxShow: 0, TriggerPrice: 0,
})
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitBuy(t.Context(), &OrderBuyAndSellParams{
Instrument: btcPerpInstrument, OrderType: "limit",
Label: "testOrder", TimeInForce: "",
Trigger: "", Advanced: "",
Amount: 30, Price: 500000,
MaxShow: 0, TriggerPrice: 0,
PostOnly: false, RejectPostOnly: false,
ReduceOnly: false, MMP: false,
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSubmitSell(t *testing.T) {
t.Parallel()
_, err := e.SubmitSell(t.Context(), &OrderBuyAndSellParams{})
require.ErrorIs(t, err, common.ErrNilPointer)
info, err := e.GetInstrument(t.Context(), btcPerpInstrument)
require.NoError(t, err)
_, err = e.SubmitSell(t.Context(), &OrderBuyAndSellParams{OrderType: "limit", Label: "testOrder", TimeInForce: "", Trigger: "", Advanced: "", Amount: info.ContractSize * 3, Price: 500000, MaxShow: 0, TriggerPrice: 0, PostOnly: false, RejectPostOnly: false, ReduceOnly: false, MMP: false})
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitSell(t.Context(), &OrderBuyAndSellParams{Instrument: btcPerpInstrument, OrderType: "limit", Label: "testOrder", TimeInForce: "", Trigger: "", Advanced: "", Amount: info.ContractSize * 3, Price: 500000, MaxShow: 0, TriggerPrice: 0, PostOnly: false, RejectPostOnly: false, ReduceOnly: false, MMP: false})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitSell(t *testing.T) {
t.Parallel()
info, err := e.GetInstrument(t.Context(), btcPerpInstrument)
require.NoError(t, err)
_, err = e.WSSubmitSell(t.Context(), &OrderBuyAndSellParams{})
require.ErrorIs(t, err, common.ErrNilPointer)
_, err = e.WSSubmitSell(t.Context(), &OrderBuyAndSellParams{
Instrument: "", OrderType: "limit",
Label: "testOrder", TimeInForce: "",
Trigger: "", Advanced: "", Amount: info.ContractSize * 3,
Price: 500000, MaxShow: 0, TriggerPrice: 0, PostOnly: false,
RejectPostOnly: false, ReduceOnly: false, MMP: false,
})
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitSell(t.Context(), &OrderBuyAndSellParams{
Instrument: btcPerpInstrument, OrderType: "limit",
Label: "testOrder", TimeInForce: "",
Trigger: "", Advanced: "", Amount: info.ContractSize * 3,
Price: 500000, MaxShow: 0, TriggerPrice: 0, PostOnly: false,
RejectPostOnly: false, ReduceOnly: false, MMP: false,
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestEditOrderByLabel(t *testing.T) {
t.Parallel()
_, err := e.EditOrderByLabel(t.Context(), &OrderBuyAndSellParams{})
require.ErrorIs(t, err, common.ErrNilPointer)
_, err = e.EditOrderByLabel(t.Context(), &OrderBuyAndSellParams{
Label: "incorrectUserLabel", Instrument: "",
Advanced: "", Amount: 1, Price: 30000, TriggerPrice: 0, PostOnly: false, ReduceOnly: false, RejectPostOnly: false, MMP: false,
})
require.ErrorIs(t, err, errInvalidInstrumentName)
_, err = e.EditOrderByLabel(t.Context(), &OrderBuyAndSellParams{
Label: "incorrectUserLabel", Instrument: btcPerpInstrument,
Advanced: "", Amount: 0, Price: 30000, TriggerPrice: 0, PostOnly: false, ReduceOnly: false, RejectPostOnly: false, MMP: false,
})
require.ErrorIs(t, err, errInvalidAmount)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.EditOrderByLabel(t.Context(), &OrderBuyAndSellParams{
Label: "incorrectUserLabel", Instrument: btcPerpInstrument,
Advanced: "", Amount: 1, Price: 30000, TriggerPrice: 0, PostOnly: false, ReduceOnly: false, RejectPostOnly: false, MMP: false,
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSEditOrderByLabel(t *testing.T) {
t.Parallel()
_, err := e.WSEditOrderByLabel(t.Context(), &OrderBuyAndSellParams{})
require.ErrorIs(t, err, common.ErrNilPointer)
_, err = e.WSEditOrderByLabel(t.Context(), &OrderBuyAndSellParams{
Label: "incorrectUserLabel", Instrument: "",
Advanced: "", Amount: 1, Price: 30000, TriggerPrice: 0, PostOnly: false, ReduceOnly: false, RejectPostOnly: false, MMP: false,
})
require.ErrorIs(t, err, errInvalidInstrumentName)
_, err = e.WSEditOrderByLabel(t.Context(), &OrderBuyAndSellParams{
Label: "incorrectUserLabel", Instrument: btcPerpInstrument,
Advanced: "", Amount: 0, Price: 30000, TriggerPrice: 0, PostOnly: false, ReduceOnly: false, RejectPostOnly: false, MMP: false,
})
require.ErrorIs(t, err, errInvalidAmount)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSEditOrderByLabel(t.Context(), &OrderBuyAndSellParams{
Label: "incorrectUserLabel", Instrument: btcPerpInstrument,
Advanced: "", Amount: 1, Price: 30000, TriggerPrice: 0, PostOnly: false, ReduceOnly: false, RejectPostOnly: false, MMP: false,
})
require.NoError(t, err)
assert.NotNil(t, result)
}
const submitCancelResponseJSON = `{"triggered": false, "trigger": "index_price", "time_in_force": "good_til_cancelled", "trigger_price": 144.73, "reduce_only": false, "profit_loss": 0, "price": "1234", "post_only": false, "order_type": "stop_market", "order_state": "untriggered", "order_id": "ETH-SLIS-12", "max_show": 5, "last_update_timestamp": 1550575961291, "label": "", "is_liquidation": false, "instrument_name": "ETH-PERPETUAL", "direction": "sell", "creation_timestamp": 1550575961291, "api": false, "amount": 5 }`
func TestSubmitCancel(t *testing.T) {
t.Parallel()
var resp *PrivateCancelData
err := json.Unmarshal([]byte(submitCancelResponseJSON), &resp)
require.NoError(t, err)
_, err = e.SubmitCancel(t.Context(), "")
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitCancel(t.Context(), "incorrectID")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitCancel(t *testing.T) {
t.Parallel()
_, err := e.WSSubmitCancel(t.Context(), "")
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitCancel(t.Context(), "incorrectID")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSubmitCancelAll(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitCancelAll(t.Context(), false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitCancelAll(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitCancelAll(t.Context(), true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSubmitCancelAllByCurrency(t *testing.T) {
t.Parallel()
_, err := e.SubmitCancelAllByCurrency(t.Context(), currency.EMPTYCODE, "option", "", true)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitCancelAllByCurrency(t.Context(), currency.BTC, "option", "", true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitCancelAllByCurrency(t *testing.T) {
t.Parallel()
_, err := e.WSSubmitCancelAllByCurrency(t.Context(), currency.EMPTYCODE, "option", "", true)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitCancelAllByCurrency(t.Context(), currency.BTC, "option", "", true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSubmitCancelAllByKind(t *testing.T) {
t.Parallel()
_, err := e.SubmitCancelAllByKind(t.Context(), currency.EMPTYCODE, "option_combo", "trigger_all", true)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitCancelAllByKind(t.Context(), currency.ETH, "option_combo", "trigger_all", true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsSubmitCancelAllByKind(t *testing.T) {
t.Parallel()
_, err := e.WsSubmitCancelAllByKind(t.Context(), currency.EMPTYCODE, "option_combo", "trigger_all", true)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WsSubmitCancelAllByKind(t.Context(), currency.ETH, "option_combo", "trigger_all", true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSubmitCancelAllByInstrument(t *testing.T) {
t.Parallel()
_, err := e.SubmitCancelAllByInstrument(t.Context(), "", "all", true, true)
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitCancelAllByInstrument(t.Context(), btcPerpInstrument, "all", true, true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitCancelAllByInstrument(t *testing.T) {
t.Parallel()
_, err := e.WSSubmitCancelAllByInstrument(t.Context(), "", "all", true, true)
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitCancelAllByInstrument(t.Context(), btcPerpInstrument, "all", true, true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSubmitCancelByLabel(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitCancelByLabel(t.Context(), "incorrectOrderLabel", currency.EMPTYCODE, true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitCancelByLabel(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitCancelByLabel(t.Context(), "incorrectOrderLabel", currency.EMPTYCODE, true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSubmitCancelQuotes(t *testing.T) {
t.Parallel()
_, err := e.SubmitCancelQuotes(t.Context(), currency.EMPTYCODE, 0, 0, "all", "", formatFuturesTradablePair(futuresTradablePair), "future", true)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitCancelQuotes(t.Context(), currency.BTC, 0, 0, "all", "", formatFuturesTradablePair(futuresTradablePair), "future", true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitCancelQuotes(t *testing.T) {
t.Parallel()
_, err := e.WSSubmitCancelQuotes(t.Context(), currency.EMPTYCODE, 0, 0, "all", "", formatFuturesTradablePair(futuresTradablePair), "future", true)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitCancelQuotes(t.Context(), currency.BTC, 0, 0, "all", "", formatFuturesTradablePair(futuresTradablePair), "future", true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSubmitClosePosition(t *testing.T) {
t.Parallel()
_, err := e.SubmitClosePosition(t.Context(), "", "limit", 35000)
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitClosePosition(t.Context(), formatFuturesTradablePair(futuresTradablePair), "limit", 35000)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitClosePosition(t *testing.T) {
t.Parallel()
_, err := e.WSSubmitClosePosition(t.Context(), "", "limit", 35000)
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitClosePosition(t.Context(), formatFuturesTradablePair(futuresTradablePair), "limit", 35000)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetMargins(t *testing.T) {
t.Parallel()
_, err := e.GetMargins(t.Context(), "", 5, 35000)
require.ErrorIs(t, err, errInvalidInstrumentName)
_, err = e.GetMargins(t.Context(), formatFuturesTradablePair(futuresTradablePair), 0, 35000)
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.GetMargins(t.Context(), formatFuturesTradablePair(futuresTradablePair), 5, -1)
require.ErrorIs(t, err, errInvalidPrice)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetMargins(t.Context(), formatFuturesTradablePair(futuresTradablePair), 5, 35000)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveMargins(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveMargins(t.Context(), "", 5, 35000)
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveMargins(t.Context(), formatFuturesTradablePair(futuresTradablePair), 5, 35000)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetMMPConfig(t *testing.T) {
t.Parallel()
_, err := e.GetMMPConfig(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetMMPConfig(t.Context(), currency.ETH)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveMMPConfig(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveMMPConfig(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveMMPConfig(t.Context(), currency.ETH)
require.NoError(t, err)
assert.NotNil(t, result)
}
const getOpenOrdersByCurrencyResponseJSON = `[{ "time_in_force": "good_til_cancelled", "reduce_only": false, "profit_loss": 0, "price": 0.0028, "post_only": false, "order_type": "limit", "order_state": "open", "order_id": "146062", "max_show": 10, "last_update_timestamp": 1550050597036, "label": "", "is_liquidation": false, "instrument_name": "BTC-15FEB19-3250-P", "filled_amount": 0, "direction": "buy", "creation_timestamp": 1550050597036, "commission": 0, "average_price": 0, "api": true, "amount": 10 } ]`
func TestGetOpenOrdersByCurrency(t *testing.T) {
t.Parallel()
var resp []OrderData
err := json.Unmarshal([]byte(getOpenOrdersByCurrencyResponseJSON), &resp)
require.NoError(t, err)
_, err = e.GetOpenOrdersByCurrency(t.Context(), currency.EMPTYCODE, "option", "all")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetOpenOrdersByCurrency(t.Context(), currency.BTC, "option", "all")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveOpenOrdersByCurrency(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveOpenOrdersByCurrency(t.Context(), currency.EMPTYCODE, "option", "all")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveOpenOrdersByCurrency(t.Context(), currency.BTC, "option", "all")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetOpenOrdersByLabel(t *testing.T) {
t.Parallel()
_, err := e.GetOpenOrdersByLabel(t.Context(), currency.EMPTYCODE, "the-label")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetOpenOrdersByLabel(t.Context(), currency.EURR, "the-label")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveOpenOrdersByLabel(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveOpenOrdersByLabel(t.Context(), currency.EMPTYCODE, "the-label")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSRetrieveOpenOrdersByLabel(t.Context(), currency.EURR, "the-label")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetOpenOrdersByInstrument(t *testing.T) {
t.Parallel()
_, err := e.GetOpenOrdersByInstrument(t.Context(), "", "all")
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetOpenOrdersByInstrument(t.Context(), btcPerpInstrument, "all")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveOpenOrdersByInstrument(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveOpenOrdersByInstrument(t.Context(), "", "all")
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveOpenOrdersByInstrument(t.Context(), btcPerpInstrument, "all")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetOrderHistoryByCurrency(t *testing.T) {
t.Parallel()
_, err := e.GetOrderHistoryByCurrency(t.Context(), currency.EMPTYCODE, "future", 0, 0, false, false)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetOrderHistoryByCurrency(t.Context(), currency.BTC, "future", 0, 0, false, false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveOrderHistoryByCurrency(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveOrderHistoryByCurrency(t.Context(), currency.EMPTYCODE, "future", 0, 0, false, false)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveOrderHistoryByCurrency(t.Context(), currency.BTC, "future", 0, 0, false, false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetOrderHistoryByInstrument(t *testing.T) {
t.Parallel()
_, err := e.GetOrderHistoryByInstrument(t.Context(), "", 0, 0, false, false)
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetOrderHistoryByInstrument(t.Context(), btcPerpInstrument, 0, 0, false, false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveOrderHistoryByInstrument(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveOrderHistoryByInstrument(t.Context(), "", 0, 0, false, false)
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveOrderHistoryByInstrument(t.Context(), btcPerpInstrument, 0, 0, false, false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetOrderMarginsByID(t *testing.T) {
t.Parallel()
_, err := e.GetOrderMarginsByID(t.Context(), []string{})
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetOrderMarginsByID(t.Context(), []string{"21422175153", "21422175154"})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveOrderMarginsByID(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveOrderMarginsByID(t.Context(), []string{})
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveOrderMarginsByID(t.Context(), []string{"ETH-349280", "ETH-349279", "ETH-349278"})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetOrderState(t *testing.T) {
t.Parallel()
_, err := e.GetOrderState(t.Context(), "")
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetOrderState(t.Context(), "brokenid123")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrievesOrderState(t *testing.T) {
t.Parallel()
_, err := e.WSRetrievesOrderState(t.Context(), "")
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrievesOrderState(t.Context(), "brokenid123")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetOrderStateByLabel(t *testing.T) {
t.Parallel()
_, err := e.GetOrderStateByLabel(t.Context(), currency.EMPTYCODE, "the-label")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetOrderStateByLabel(t.Context(), currency.EURR, "the-label")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsRetrieveOrderStateByLabel(t *testing.T) {
t.Parallel()
_, err := e.WsRetrieveOrderStateByLabel(t.Context(), currency.EMPTYCODE, "the-label")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WsRetrieveOrderStateByLabel(t.Context(), currency.EURR, "the-label")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetTriggerOrderHistory(t *testing.T) {
t.Parallel()
_, err := e.GetTriggerOrderHistory(t.Context(), currency.EMPTYCODE, "", "", 0)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetTriggerOrderHistory(t.Context(), currency.ETH, "", "", 0)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveTriggerOrderHistory(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveTriggerOrderHistory(t.Context(), currency.EMPTYCODE, "", "", 0)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveTriggerOrderHistory(t.Context(), currency.ETH, "", "", 0)
require.NoError(t, err)
assert.NotNil(t, result)
}
const getUserTradesByCurrencyResponseJSON = `{"trades": [ { "underlying_price": 204.5, "trade_seq": 3, "trade_id": "ETH-2696060", "timestamp": 1590480363130, "tick_direction": 2, "state": "filled", "reduce_only": false, "price": 0.361, "post_only": false, "order_type": "limit", "order_id": "ETH-584827850", "matching_id": null, "mark_price": 0.364585, "liquidity": "T", "iv": 0, "instrument_name": "ETH-29MAY20-130-C", "index_price": 203.72, "fee_currency": "ETH", "fee": 0.002, "direction": "sell", "amount": 5 }, { "underlying_price": 204.82, "trade_seq": 3, "trade_id": "ETH-2696062", "timestamp": 1590480416119, "tick_direction": 0, "state": "filled", "reduce_only": false, "price": 0.015, "post_only": false, "order_type": "limit", "order_id": "ETH-584828229", "matching_id": null, "mark_price": 0.000596, "liquidity": "T", "iv": 352.91, "instrument_name": "ETH-29MAY20-140-P", "index_price": 204.06, "fee_currency": "ETH", "fee": 0.002, "direction": "buy", "amount": 5 } ], "has_more": true }`
func TestGetUserTradesByCurrency(t *testing.T) {
t.Parallel()
var resp *UserTradesData
err := json.Unmarshal([]byte(getUserTradesByCurrencyResponseJSON), &resp)
require.NoError(t, err)
_, err = e.GetUserTradesByCurrency(t.Context(), currency.EMPTYCODE, "future", "", "", "asc", 0, false)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetUserTradesByCurrency(t.Context(), currency.ETH, "future", "", "", "asc", 0, false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveUserTradesByCurrency(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveUserTradesByCurrency(t.Context(), currency.EMPTYCODE, "future", "", "", "asc", 0, false)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveUserTradesByCurrency(t.Context(), currency.ETH, "future", "", "", "asc", 0, false)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetUserTradesByCurrencyAndTime(t *testing.T) {
t.Parallel()
_, err := e.GetUserTradesByCurrencyAndTime(t.Context(), currency.EMPTYCODE, "future", "default", 5, time.Now().Add(-time.Hour*10), time.Now().Add(-time.Hour*1))
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetUserTradesByCurrencyAndTime(t.Context(), currency.ETH, "future", "default", 5, time.Now().Add(-time.Hour*10), time.Now().Add(-time.Hour*1))
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveUserTradesByCurrencyAndTime(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveUserTradesByCurrencyAndTime(t.Context(), currency.EMPTYCODE, "future", "default", 5, time.Now().Add(-time.Hour*4), time.Now())
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveUserTradesByCurrencyAndTime(t.Context(), currency.ETH, "future", "default", 5, time.Now().Add(-time.Hour*4), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetUserTradesByInstrument(t *testing.T) {
t.Parallel()
_, err := e.GetUserTradesByInstrument(t.Context(), "", "asc", 5, 10, 4, true)
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetUserTradesByInstrument(t.Context(), btcPerpInstrument, "asc", 5, 10, 4, true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsRetrieveUserTradesByInstrument(t *testing.T) {
t.Parallel()
_, err := e.WsRetrieveUserTradesByInstrument(t.Context(), "", "asc", 5, 10, 4, true)
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WsRetrieveUserTradesByInstrument(t.Context(), btcPerpInstrument, "asc", 5, 10, 4, true)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetUserTradesByInstrumentAndTime(t *testing.T) {
t.Parallel()
_, err := e.GetUserTradesByInstrumentAndTime(t.Context(), "", "asc", 10, time.Now().Add(-time.Hour), time.Now())
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetUserTradesByInstrumentAndTime(t.Context(), btcPerpInstrument, "asc", 10, time.Now().Add(-time.Hour), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveUserTradesByInstrumentAndTime(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveUserTradesByInstrumentAndTime(t.Context(), "", "asc", 10, false, time.Now().Add(-time.Hour), time.Now())
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveUserTradesByInstrumentAndTime(t.Context(), btcPerpInstrument, "asc", 10, false, time.Now().Add(-time.Hour), time.Now())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetUserTradesByOrder(t *testing.T) {
t.Parallel()
_, err := e.GetUserTradesByOrder(t.Context(), "", "default")
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetUserTradesByOrder(t.Context(), "wrongOrderID", "default")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveUserTradesByOrder(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveUserTradesByOrder(t.Context(), "", "default")
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveUserTradesByOrder(t.Context(), "wrongOrderID", "default")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestResetMMP(t *testing.T) {
t.Parallel()
err := e.ResetMMP(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
err = e.ResetMMP(t.Context(), currency.BTC)
assert.NoError(t, err)
}
func TestWSResetMMP(t *testing.T) {
t.Parallel()
err := e.WSResetMMP(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
err = e.WSResetMMP(t.Context(), currency.BTC)
assert.NoError(t, err)
}
func TestSetMMPConfig(t *testing.T) {
t.Parallel()
err := e.SetMMPConfig(t.Context(), currency.EMPTYCODE, kline.FiveMin, 5, 0, 0)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
err = e.SetMMPConfig(t.Context(), currency.BTC, kline.FiveMin, 5, 0, 0)
assert.NoError(t, err)
}
func TestWSSetMMPConfig(t *testing.T) {
t.Parallel()
err := e.WSSetMMPConfig(t.Context(), currency.EMPTYCODE, kline.FiveMin, 5, 0, 0)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
err = e.WSSetMMPConfig(t.Context(), currency.BTC, kline.FiveMin, 5, 0, 0)
assert.NoError(t, err)
}
func TestGetSettlementHistoryByCurency(t *testing.T) {
t.Parallel()
_, err := e.GetSettlementHistoryByCurency(t.Context(), currency.EMPTYCODE, "settlement", "", 10, time.Now().Add(-time.Hour))
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetSettlementHistoryByCurency(t.Context(), currency.BTC, "settlement", "", 10, time.Now().Add(-time.Hour))
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveSettlementHistoryByCurency(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveSettlementHistoryByCurency(t.Context(), currency.EMPTYCODE, "settlement", "", 10, time.Now().Add(-time.Hour))
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveSettlementHistoryByCurency(t.Context(), currency.BTC, "settlement", "", 10, time.Now().Add(-time.Hour))
require.NoError(t, err)
assert.NotNil(t, result)
}
const getSettlementHistoryByInstrumentResponseJSON = `{"settlements": [ { "type": "settlement", "timestamp": 1550475692526, "session_profit_loss": 0.038358299, "profit_loss": -0.001783937, "position": -66, "mark_price": 121.67, "instrument_name": "ETH-22FEB19", "index_price": 119.8 } ], "continuation": "xY7T6cusbMBNpH9SNmKb94jXSBxUPojJEdCPL4YociHBUgAhWQvEP" }`
func TestGetSettlementHistoryByInstrument(t *testing.T) {
t.Parallel()
var result *PrivateSettlementsHistoryData
err := json.Unmarshal([]byte(getSettlementHistoryByInstrumentResponseJSON), &result)
require.NoError(t, err)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err = e.GetSettlementHistoryByInstrument(t.Context(), btcPerpInstrument, "settlement", "", 10, time.Now().Add(-time.Hour))
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveSettlementHistoryByInstrument(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveSettlementHistoryByInstrument(t.Context(), "", "settlement", "", 10, time.Now().Add(-time.Hour))
require.ErrorIs(t, err, errInvalidInstrumentName)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveSettlementHistoryByInstrument(t.Context(), btcPerpInstrument, "settlement", "", 10, time.Now().Add(-time.Hour))
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSubmitEdit(t *testing.T) {
t.Parallel()
_, err := e.SubmitEdit(t.Context(), &OrderBuyAndSellParams{})
require.ErrorIs(t, err, common.ErrNilPointer)
_, err = e.SubmitEdit(t.Context(), &OrderBuyAndSellParams{OrderID: "", Advanced: "", TriggerPrice: 0.001, Price: 100000, Amount: 123})
require.ErrorIs(t, err, errInvalidID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.SubmitEdit(t.Context(), &OrderBuyAndSellParams{OrderID: "incorrectID", Advanced: "", TriggerPrice: 0.001, Price: 100000, Amount: 123})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSSubmitEdit(t *testing.T) {
t.Parallel()
_, err := e.WSSubmitEdit(t.Context(), &OrderBuyAndSellParams{})
require.ErrorIs(t, err, common.ErrNilPointer)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSSubmitEdit(t.Context(), &OrderBuyAndSellParams{
OrderID: "incorrectID",
Advanced: "",
TriggerPrice: 0.001,
Price: 100000,
Amount: 123,
})
require.NoError(t, err)
assert.NotNil(t, result)
}
// Combo Books Endpoints
func TestGetComboIDS(t *testing.T) {
t.Parallel()
_, err := e.GetComboIDs(t.Context(), currency.EMPTYCODE, "")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.GetComboIDs(t.Context(), currency.BTC, "")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveComboIDS(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveComboIDs(t.Context(), currency.EMPTYCODE, "")
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
combos, err := e.WSRetrieveComboIDs(t.Context(), currency.BTC, "")
require.NoError(t, err)
assert.NotEmpty(t, combos)
}
func TestGetComboDetails(t *testing.T) {
t.Parallel()
_, err := e.GetComboDetails(t.Context(), "")
require.ErrorIs(t, err, errInvalidComboID)
result, err := e.GetComboDetails(t.Context(), futureComboTradablePair.String())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveComboDetails(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveComboDetails(t.Context(), "")
require.ErrorIs(t, err, errInvalidComboID)
result, err := e.WSRetrieveComboDetails(t.Context(), futureComboTradablePair.String())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetCombos(t *testing.T) {
t.Parallel()
_, err := e.GetCombos(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.GetCombos(t.Context(), currency.BTC)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestCreateCombo(t *testing.T) {
t.Parallel()
_, err := e.CreateCombo(t.Context(), []ComboParam{})
require.ErrorIs(t, err, errNoArgumentPassed)
instruments, err := e.GetEnabledPairs(asset.Futures)
require.NoError(t, err)
if len(instruments) < 2 {
t.Skip("no enough instrument found")
}
_, err = e.CreateCombo(t.Context(), []ComboParam{
{
InstrumentName: instruments[0].String(),
Direction: "sell",
},
{
InstrumentName: instruments[1].String(),
Direction: "sell",
Amount: 1200,
},
})
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.CreateCombo(t.Context(), []ComboParam{
{
InstrumentName: instruments[0].String(),
Amount: 123,
},
{
InstrumentName: instruments[1].String(),
Direction: "sell",
Amount: 1200,
},
})
require.ErrorIs(t, err, errInvalidOrderSideOrDirection)
_, err = e.CreateCombo(t.Context(), []ComboParam{
{
InstrumentName: instruments[0].String(),
Direction: "buy",
Amount: 123,
},
{
InstrumentName: instruments[1].String(),
Direction: "buy",
Amount: 1200,
},
})
require.ErrorIs(t, err, errDifferentInstruments)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.CreateCombo(t.Context(), []ComboParam{
{
InstrumentName: instruments[0].String(),
Direction: "buy",
Amount: 123,
},
{
InstrumentName: instruments[0].String(),
Direction: "sell",
Amount: 1200,
},
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSCreateCombo(t *testing.T) {
t.Parallel()
_, err := e.WSCreateCombo(t.Context(), []ComboParam{})
require.ErrorIs(t, err, errNoArgumentPassed)
instruments, err := e.GetEnabledPairs(asset.Futures)
require.NoError(t, err)
if len(instruments) < 2 {
t.Skip("no enough instrument found")
}
_, err = e.WSCreateCombo(t.Context(), []ComboParam{})
require.ErrorIs(t, err, errNoArgumentPassed)
_, err = e.WSCreateCombo(t.Context(), []ComboParam{
{
InstrumentName: instruments[0].String(),
Direction: "sell",
},
{
InstrumentName: instruments[1].String(),
Direction: "sell",
Amount: 1200,
},
})
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.WSCreateCombo(t.Context(), []ComboParam{
{
InstrumentName: instruments[0].String(),
Amount: 123,
},
{
InstrumentName: instruments[1].String(),
Direction: "sell",
Amount: 1200,
},
})
require.ErrorIs(t, err, errInvalidOrderSideOrDirection)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSCreateCombo(t.Context(), []ComboParam{
{
InstrumentName: instruments[0].String(),
Direction: "sell",
Amount: 123,
},
{
InstrumentName: instruments[1].String(),
Direction: "buy",
Amount: 1200,
},
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestVerifyBlockTrade(t *testing.T) {
t.Parallel()
_, err := e.VerifyBlockTrade(t.Context(), time.Now(), "", "maker", currency.EMPTYCODE, []BlockTradeParam{})
require.ErrorIs(t, err, errMissingNonce)
_, err = e.VerifyBlockTrade(t.Context(), time.Now(), "nonce-string", "", currency.EMPTYCODE, []BlockTradeParam{})
require.ErrorIs(t, err, errInvalidTradeRole)
_, err = e.VerifyBlockTrade(t.Context(), time.Now(), "nonce-string", "maker", currency.EMPTYCODE, []BlockTradeParam{})
require.ErrorIs(t, err, errNoArgumentPassed)
info, err := e.GetInstrument(t.Context(), btcPerpInstrument)
require.NoError(t, err)
require.NotNil(t, info)
_, err = e.VerifyBlockTrade(t.Context(), time.Time{}, "nonce-string", "maker", currency.EMPTYCODE, []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: btcPerpInstrument,
Direction: "buy",
Amount: info.MinimumTradeAmount*5 + (200000 - info.MinimumTradeAmount*5) + 10,
},
})
require.ErrorIs(t, err, errZeroTimestamp)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.VerifyBlockTrade(t.Context(), time.Now(), "something", "maker", currency.EMPTYCODE, []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: btcPerpInstrument,
Direction: order.Buy.Lower(),
Amount: info.MinimumTradeAmount*5 + (200000 - info.MinimumTradeAmount*5) + 10,
},
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSVerifyBlockTrade(t *testing.T) {
t.Parallel()
_, err := e.WSVerifyBlockTrade(t.Context(), time.Now(), "", "maker", currency.EMPTYCODE, []BlockTradeParam{})
require.ErrorIs(t, err, errMissingNonce)
_, err = e.WSVerifyBlockTrade(t.Context(), time.Now(), "nonce-string", "", currency.EMPTYCODE, []BlockTradeParam{})
require.ErrorIs(t, err, errInvalidTradeRole)
_, err = e.WSVerifyBlockTrade(t.Context(), time.Now(), "nonce-string", "maker", currency.EMPTYCODE, []BlockTradeParam{})
require.ErrorIs(t, err, errNoArgumentPassed)
info, err := e.GetInstrument(t.Context(), btcPerpInstrument)
require.NoError(t, err)
require.NotNil(t, info)
_, err = e.WSVerifyBlockTrade(t.Context(), time.Time{}, "nonce-string", "maker", currency.EMPTYCODE, []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: btcPerpInstrument,
Direction: "buy",
Amount: info.MinimumTradeAmount*5 + (200000 - info.MinimumTradeAmount*5) + 10,
},
})
require.ErrorIs(t, err, errZeroTimestamp)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSVerifyBlockTrade(t.Context(), time.Now(), "sdjkafdad", "maker", currency.EMPTYCODE, []BlockTradeParam{
{
Price: 0.777 * 28000,
InstrumentName: btcPerpInstrument,
Direction: "buy",
Amount: info.MinimumTradeAmount*5 + (200000 - info.MinimumTradeAmount*5) + 10,
},
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestInvalidateBlockTradeSignature(t *testing.T) {
t.Parallel()
err := e.WsInvalidateBlockTradeSignature(t.Context(), "")
require.ErrorIs(t, err, errMissingSignature)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
err = e.InvalidateBlockTradeSignature(t.Context(), "verified_signature_string")
assert.NoError(t, err)
}
func TestWsInvalidateBlockTradeSignature(t *testing.T) {
t.Parallel()
err := e.WsInvalidateBlockTradeSignature(t.Context(), "")
require.ErrorIs(t, err, errMissingSignature)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
err = e.WsInvalidateBlockTradeSignature(t.Context(), "verified_signature_string")
assert.NoError(t, err)
}
func TestExecuteBlockTrade(t *testing.T) {
t.Parallel()
_, err := e.ExecuteBlockTrade(t.Context(), time.Now(), "", "maker", currency.EMPTYCODE, []BlockTradeParam{})
require.ErrorIs(t, err, errMissingNonce)
_, err = e.ExecuteBlockTrade(t.Context(), time.Now(), "nonce-string", "", currency.EMPTYCODE, []BlockTradeParam{})
require.ErrorIs(t, err, errInvalidTradeRole)
_, err = e.ExecuteBlockTrade(t.Context(), time.Now(), "nonce-string", "maker", currency.EMPTYCODE, []BlockTradeParam{})
require.ErrorIs(t, err, errNoArgumentPassed)
info, err := e.GetInstrument(t.Context(), btcPerpInstrument)
require.NoError(t, err)
require.NotNil(t, info)
_, err = e.ExecuteBlockTrade(t.Context(), time.Time{}, "nonce-string", "maker", currency.EMPTYCODE, []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: btcPerpInstrument,
Direction: "buy",
Amount: info.MinimumTradeAmount*5 + (200000 - info.MinimumTradeAmount*5) + 10,
},
})
require.ErrorIs(t, err, errZeroTimestamp)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.ExecuteBlockTrade(t.Context(), time.Now(), "something", "maker", currency.EMPTYCODE, []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: btcPerpInstrument,
Direction: "buy",
Amount: info.MinimumTradeAmount*5 + (200000 - info.MinimumTradeAmount*5) + 10,
},
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSExecuteBlockTrade(t *testing.T) {
t.Parallel()
_, err := e.WSExecuteBlockTrade(t.Context(), time.Now(), "", "maker", currency.EMPTYCODE, []BlockTradeParam{})
require.ErrorIs(t, err, errMissingNonce)
_, err = e.WSExecuteBlockTrade(t.Context(), time.Now(), "nonce-string", "", currency.EMPTYCODE, []BlockTradeParam{})
require.ErrorIs(t, err, errInvalidTradeRole)
_, err = e.WSExecuteBlockTrade(t.Context(), time.Now(), "nonce-string", "maker", currency.EMPTYCODE, []BlockTradeParam{})
require.ErrorIs(t, err, errNoArgumentPassed)
info, err := e.GetInstrument(t.Context(), btcPerpInstrument)
require.NoError(t, err)
require.NotNil(t, info)
_, err = e.WSExecuteBlockTrade(t.Context(), time.Time{}, "nonce-string", "maker", currency.EMPTYCODE, []BlockTradeParam{{
Price: 0.777 * 22000,
InstrumentName: btcPerpInstrument,
Direction: "buy",
Amount: info.MinimumTradeAmount*5 + (200000 - info.MinimumTradeAmount*5) + 10,
}})
require.ErrorIs(t, err, errZeroTimestamp)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSExecuteBlockTrade(t.Context(), time.Now(), "sdjkafdad", "maker", currency.EMPTYCODE, []BlockTradeParam{
{
Price: 0.777 * 22000,
InstrumentName: btcPerpInstrument,
Direction: "buy",
Amount: info.MinimumTradeAmount*5 + (200000 - info.MinimumTradeAmount*5) + 10,
},
})
require.NoError(t, err)
assert.NotNil(t, result)
}
const getUserBlocTradeResponseJSON = `[ { "trade_seq": 37, "trade_id": "92437", "timestamp": 1565089523719, "tick_direction": 3, "state": "filled", "price": 0.0001, "order_type": "limit", "order_id": "343062", "matching_id": null, "liquidity": "T", "iv": 0, "instrument_name": "BTC-9AUG19-10250-C", "index_price": 11738, "fee_currency": "BTC", "fee": 0.00025, "direction": "sell", "block_trade_id": "61", "amount": 10 }, { "trade_seq": 25350, "trade_id": "92435", "timestamp": 1565089523719, "tick_direction": 3, "state": "filled", "price": 11590, "order_type": "limit", "order_id": "343058", "matching_id": null, "liquidity": "T", "instrument_name": "BTC-PERPETUAL", "index_price": 11737.98, "fee_currency": "BTC", "fee": 0.00000164, "direction": "buy", "block_trade_id": "61", "amount": 190 } ]`
func TestGetUserBlocTrade(t *testing.T) {
t.Parallel()
var resp []BlockTradeData
err := json.Unmarshal([]byte(getUserBlocTradeResponseJSON), &resp)
require.NoError(t, err)
_, err = e.GetUserBlockTrade(t.Context(), "")
require.ErrorIs(t, err, errMissingBlockTradeID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetUserBlockTrade(t.Context(), "12345567")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveUserBlockTrade(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveUserBlockTrade(t.Context(), "")
require.ErrorIs(t, err, errMissingBlockTradeID)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveUserBlockTrade(t.Context(), "12345567")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetLastBlockTradesbyCurrency(t *testing.T) {
t.Parallel()
_, err := e.GetLastBlockTradesByCurrency(t.Context(), currency.EMPTYCODE, "", "", 5)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetLastBlockTradesByCurrency(t.Context(), currency.SOL, "", "", 5)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSRetrieveLastBlockTradesByCurrency(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveLastBlockTradesByCurrency(t.Context(), currency.EMPTYCODE, "", "", 5)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WSRetrieveLastBlockTradesByCurrency(t.Context(), currency.SOL, "", "", 5)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestMovePositions(t *testing.T) {
t.Parallel()
_, err := e.MovePositions(t.Context(), currency.EMPTYCODE, 123, 345, []BlockTradeParam{})
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.MovePositions(t.Context(), currency.BTC, 0, 345, []BlockTradeParam{})
require.ErrorIs(t, err, errMissingSubAccountID)
_, err = e.MovePositions(t.Context(), currency.BTC, 123, 0, []BlockTradeParam{})
require.ErrorIs(t, err, errMissingSubAccountID)
_, err = e.MovePositions(t.Context(), currency.BTC, 123, 345, []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: "",
Direction: "buy",
Amount: 100,
},
})
require.ErrorIs(t, err, errInvalidInstrumentName)
_, err = e.MovePositions(t.Context(), currency.BTC, 123, 345, []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: "BTC-PERPETUAL",
Direction: "buy",
Amount: 0,
},
})
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.MovePositions(t.Context(), currency.BTC, 123, 345, []BlockTradeParam{
{
Price: -4,
InstrumentName: "BTC-PERPETUAL",
Direction: "buy",
Amount: 20,
},
})
require.ErrorIs(t, err, errInvalidPrice)
info, err := e.GetInstrument(t.Context(), "BTC-PERPETUAL")
require.NoError(t, err)
require.NotNil(t, info)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.MovePositions(t.Context(), currency.BTC, 123, 345, []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: "BTC-PERPETUAL",
Direction: "buy",
Amount: info.MinimumTradeAmount*5 + (200000 - info.MinimumTradeAmount*5) + 10,
},
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWSMovePositions(t *testing.T) {
t.Parallel()
_, err := e.WSMovePositions(t.Context(), currency.EMPTYCODE, 123, 345, []BlockTradeParam{})
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
_, err = e.WSMovePositions(t.Context(), currency.BTC, 0, 345, []BlockTradeParam{})
require.ErrorIs(t, err, errMissingSubAccountID)
_, err = e.WSMovePositions(t.Context(), currency.BTC, 123, 0, []BlockTradeParam{})
require.ErrorIs(t, err, errMissingSubAccountID)
_, err = e.WSMovePositions(t.Context(), currency.BTC, 123, 345, []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: "",
Direction: "buy",
Amount: 100,
},
})
require.ErrorIs(t, err, errInvalidInstrumentName)
_, err = e.WSMovePositions(t.Context(), currency.BTC, 123, 345, []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: "BTC-PERPETUAL",
Direction: "buy",
Amount: 0,
},
})
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.WSMovePositions(t.Context(), currency.BTC, 123, 345, []BlockTradeParam{
{
Price: -4,
InstrumentName: "BTC-PERPETUAL",
Direction: "buy",
Amount: 20,
},
})
require.ErrorIs(t, err, errInvalidPrice)
info, err := e.GetInstrument(t.Context(), "BTC-PERPETUAL")
require.NoError(t, err)
require.NotNil(t, info)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WSMovePositions(t.Context(), currency.BTC, 123, 345, []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: btcPerpInstrument,
Direction: "buy",
Amount: info.MinimumTradeAmount*5 + (200000 - info.MinimumTradeAmount*5) + 10,
},
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSimulateBlockTrade(t *testing.T) {
t.Parallel()
_, err := e.SimulateBlockTrade(t.Context(), "", []BlockTradeParam{})
require.ErrorIs(t, err, errInvalidTradeRole)
_, err = e.SimulateBlockTrade(t.Context(), "maker", []BlockTradeParam{})
require.ErrorIs(t, err, errNoArgumentPassed)
_, err = e.SimulateBlockTrade(t.Context(), "maker", []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: "",
Direction: "buy",
Amount: 10,
},
})
require.ErrorIs(t, err, errInvalidInstrumentName)
_, err = e.SimulateBlockTrade(t.Context(), "maker", []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: btcPerpInstrument,
Direction: "",
Amount: 10,
},
})
require.ErrorIs(t, err, errInvalidOrderSideOrDirection)
_, err = e.SimulateBlockTrade(t.Context(), "maker", []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: btcPerpInstrument,
Direction: "sell",
Amount: 0,
},
})
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.SimulateBlockTrade(t.Context(), "maker", []BlockTradeParam{
{
Price: -1,
InstrumentName: btcPerpInstrument,
Direction: "sell",
Amount: 10,
},
})
require.ErrorIs(t, err, errInvalidPrice)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
info, err := e.GetInstrument(t.Context(), "BTC-PERPETUAL")
require.NoError(t, err)
result, err := e.SimulateBlockTrade(t.Context(), "maker", []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: btcPerpInstrument,
Direction: "buy",
Amount: info.MinimumTradeAmount*5 + (200000 - info.MinimumTradeAmount*5) + 10,
},
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsSimulateBlockTrade(t *testing.T) {
t.Parallel()
_, err := e.WsSimulateBlockTrade(t.Context(), "", []BlockTradeParam{})
require.ErrorIs(t, err, errInvalidTradeRole)
_, err = e.WsSimulateBlockTrade(t.Context(), "maker", []BlockTradeParam{})
require.ErrorIs(t, err, errNoArgumentPassed)
_, err = e.WsSimulateBlockTrade(t.Context(), "maker", []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: "",
Direction: "buy",
Amount: 10,
},
})
require.ErrorIs(t, err, errInvalidInstrumentName)
_, err = e.WsSimulateBlockTrade(t.Context(), "maker", []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: btcPerpInstrument,
Direction: "",
Amount: 100,
},
})
require.ErrorIs(t, err, errInvalidOrderSideOrDirection)
_, err = e.WsSimulateBlockTrade(t.Context(), "maker", []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: btcPerpInstrument,
Direction: "sell",
Amount: 0,
},
})
require.ErrorIs(t, err, errInvalidAmount)
_, err = e.WsSimulateBlockTrade(t.Context(), "maker", []BlockTradeParam{
{
Price: -1,
InstrumentName: btcPerpInstrument,
Direction: "sell",
Amount: 100,
},
})
require.ErrorIs(t, err, errInvalidPrice)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
info, err := e.GetInstrument(t.Context(), "BTC-PERPETUAL")
require.NoError(t, err)
require.NotNil(t, info)
result, err := e.WsSimulateBlockTrade(t.Context(), "taker", []BlockTradeParam{
{
Price: 0.777 * 25000,
InstrumentName: btcPerpInstrument,
Direction: "buy",
Amount: info.MinimumTradeAmount*5 + (200000 - info.MinimumTradeAmount*5) + 10,
},
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func setupWs() {
if !e.Websocket.IsEnabled() {
return
}
if !sharedtestvalues.AreAPICredentialsSet(e) {
e.Websocket.SetCanUseAuthenticatedEndpoints(false)
}
err := e.WsConnect()
if err != nil {
log.Fatal(err)
}
}
func TestGenerateSubscriptions(t *testing.T) {
t.Parallel()
e := new(Exchange) //nolint:govet // Intentional shadow
require.NoError(t, testexch.Setup(e), "Test instance Setup must not error")
e.Websocket.SetCanUseAuthenticatedEndpoints(true)
subs, err := e.generateSubscriptions()
require.NoError(t, err)
exp := subscription.List{}
for _, s := range e.Features.Subscriptions {
for _, a := range e.GetAssetTypes(true) {
if !e.IsAssetWebsocketSupported(a) {
continue
}
pairs, err := e.GetEnabledPairs(a)
require.NoErrorf(t, err, "GetEnabledPairs %s must not error", a)
s := s.Clone() //nolint:govet // Intentional lexical scope shadow
s.Asset = a
if isSymbolChannel(s) {
for i, p := range pairs {
s := s.Clone() //nolint:govet // Intentional lexical scope shadow
s.QualifiedChannel = channelName(s) + "." + p.String()
if s.Interval != 0 {
s.QualifiedChannel += "." + channelInterval(s)
}
s.Pairs = pairs[i : i+1]
exp = append(exp, s)
}
} else {
s.Pairs = pairs
s.QualifiedChannel = channelName(s)
exp = append(exp, s)
}
}
}
testsubs.EqualLists(t, exp, subs)
}
func TestChannelInterval(t *testing.T) {
t.Parallel()
for _, i := range []int64{1, 3, 5, 10, 15, 30, 60, 120, 180, 360, 720} {
a := channelInterval(&subscription.Subscription{Channel: subscription.CandlesChannel, Interval: kline.Interval(i * int64(time.Minute))})
assert.Equal(t, strconv.Itoa(int(i)), a)
}
a := channelInterval(&subscription.Subscription{Channel: subscription.CandlesChannel, Interval: kline.OneDay})
assert.Equal(t, "1D", a)
assert.Panics(t, func() {
channelInterval(&subscription.Subscription{Channel: subscription.CandlesChannel, Interval: kline.OneMonth})
})
a = channelInterval(&subscription.Subscription{Channel: subscription.OrderbookChannel, Interval: kline.ThousandMilliseconds})
assert.Equal(t, "agg2", a, "1 second should expand to agg2")
a = channelInterval(&subscription.Subscription{Channel: subscription.OrderbookChannel, Interval: kline.HundredMilliseconds})
assert.Equal(t, "100ms", a, "100ms should expand correctly")
a = channelInterval(&subscription.Subscription{Channel: subscription.OrderbookChannel, Interval: kline.Raw})
assert.Equal(t, "raw", a, "raw should expand correctly")
assert.Panics(t, func() {
channelInterval(&subscription.Subscription{Channel: subscription.OrderbookChannel, Interval: kline.OneMonth})
})
a = channelInterval(&subscription.Subscription{Channel: userAccessLogChannel})
assert.Empty(t, a, "Anything else should return empty")
}
func TestChannelName(t *testing.T) {
t.Parallel()
assert.Equal(t, tickerChannel, channelName(&subscription.Subscription{Channel: subscription.TickerChannel}))
assert.Equal(t, userLockChannel, channelName(&subscription.Subscription{Channel: userLockChannel}))
assert.Panics(t, func() { channelName(&subscription.Subscription{Channel: "wibble"}) }, "Unknown channels should panic")
}
func TestUpdateAccountInfo(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.UpdateAccountInfo(t.Context(), asset.Futures)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetFundingHistory(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetAccountFundingHistory(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetWithdrawalsHistory(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetWithdrawalsHistory(t.Context(), currency.BTC, asset.Empty)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetRecentTrades(t *testing.T) {
t.Parallel()
for assetType, cp := range assetTypeToPairsMap {
t.Run(fmt.Sprintf("%s %s", assetType, cp), func(t *testing.T) {
t.Parallel()
result, err := e.GetRecentTrades(t.Context(), cp, assetType)
require.NoError(t, err, "GetRecentTrades must not error")
require.NotNil(t, result, "result must not be nil")
})
}
}
func TestCancelAllOrders(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
orderCancellation := &order.Cancel{
OrderID: "1",
AccountID: "1",
Pair: futuresTradablePair,
AssetType: asset.Futures,
}
var result order.CancelAllResponse
var err error
for assetType, cp := range assetTypeToPairsMap {
orderCancellation.AssetType = assetType
orderCancellation.Pair = cp
result, err = e.CancelAllOrders(t.Context(), orderCancellation)
require.NoErrorf(t, err, "expected nil, got %v for asset type %s pair %s", err, assetType, cp)
require.NotNilf(t, result, "expected result not to be nil for asset type %s pair %s", assetType, cp)
}
}
func TestGetOrderInfo(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
for assetType, cp := range assetTypeToPairsMap {
result, err := e.GetOrderInfo(t.Context(), "1234", cp, assetType)
require.NoErrorf(t, err, "expected nil, got %v for asset type %s pair %s", err, assetType, cp)
require.NotNilf(t, result, "expected result not to be nil for asset type %s pair %s", assetType, cp)
}
}
func TestGetDepositAddress(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.GetDepositAddress(t.Context(), currency.BTC, "", "")
require.ErrorIs(t, err, common.ErrNoResponse)
assert.NotNil(t, result)
}
func TestWithdraw(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WithdrawCryptocurrencyFunds(t.Context(), &withdraw.Request{
Exchange: e.Name,
Amount: 1,
Currency: currency.BTC,
Description: "WITHDRAW IT ALL",
Crypto: withdraw.CryptoRequest{
Address: "0x1nv4l1d",
Chain: "tetheruse",
},
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetActiveOrders(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
getOrdersRequest := order.MultiOrderRequest{
Type: order.AnyType, AssetType: asset.Futures,
Side: order.AnySide, Pairs: currency.Pairs{futuresTradablePair},
}
for assetType, cp := range assetTypeToPairsMap {
getOrdersRequest.Pairs = []currency.Pair{cp}
getOrdersRequest.AssetType = assetType
result, err := e.GetActiveOrders(t.Context(), &getOrdersRequest)
require.NoErrorf(t, err, "expected nil, got %v for asset type %s pair %s", err, assetType, cp)
require.NotNilf(t, result, "expected result not to be nil for asset type %s pair %s", assetType, cp)
}
}
func TestGetOrderHistory(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
for assetType, cp := range assetTypeToPairsMap {
result, err := e.GetOrderHistory(t.Context(), &order.MultiOrderRequest{
Type: order.AnyType, AssetType: assetType,
Side: order.AnySide, Pairs: []currency.Pair{cp},
})
require.NoErrorf(t, err, "expected nil, got %v for asset type %s pair %s", err, assetType, cp)
require.NotNilf(t, result, "expected result not to be nil for asset type %s pair %s", assetType, cp)
}
}
func TestGetAssetPairByInstrument(t *testing.T) {
t.Parallel()
for _, assetType := range []asset.Item{asset.Spot, asset.Futures, asset.Options, asset.OptionCombo, asset.FutureCombo} {
availablePairs, err := e.GetAvailablePairs(assetType)
require.NoErrorf(t, err, "expected nil, got %v for asset type %s", err, assetType)
require.NotNilf(t, availablePairs, "expected result not to be nil for asset type %s", assetType)
for _, cp := range availablePairs {
instrument := formatPairString(assetType, cp)
t.Run(fmt.Sprintf("%s %s", assetType, instrument), func(t *testing.T) {
t.Parallel()
extractedAsset, extractedPair, err := getAssetPairByInstrument(instrument)
assert.NoError(t, err)
fPair, err := e.FormatExchangeCurrency(extractedPair, assetType)
require.NoError(t, err, "FormatExchangeCurrency must not error")
assert.Equal(t, cp.String(), fPair.String())
assert.Equal(t, assetType.String(), extractedAsset.String(), "asset should match for")
})
}
}
t.Run("empty asset, empty pair", func(t *testing.T) {
t.Parallel()
_, _, err := getAssetPairByInstrument("")
assert.ErrorIs(t, err, currency.ErrSymbolStringEmpty)
})
t.Run("thisIsAFakeCurrency", func(t *testing.T) {
t.Parallel()
_, _, err := getAssetPairByInstrument("thisIsAFakeCurrency")
assert.ErrorIs(t, err, errUnsupportedInstrumentFormat)
})
}
func TestGetAssetFromInstrument(t *testing.T) {
t.Parallel()
tc := []struct {
instrument string
expectedAsset asset.Item
expectedError error
}{
{"BNB_USDC", asset.Spot, nil},
{"BTC-30DEC22", asset.Futures, nil},
{"BTCDVOL_USDC-1OCT25", asset.Futures, nil},
{"ADA_USDC-PERPETUAL", asset.Futures, nil},
{"PAXG_USDC-12SEP25-3320-P", asset.Options, nil},
{"ETH-3OCT25-4800-P", asset.Options, nil},
{"ETH-FS-26JUN26_26DEC25", asset.FutureCombo, nil},
{"BTC_USDC-PBUT-31OCT25-90000_100000_102000", asset.OptionCombo, nil},
{"XRP_USDC-CBUT-26SEP25-2d9_3d2_3d4", asset.OptionCombo, nil},
{"ETH-CS-26SEP25-5000_5500", asset.OptionCombo, nil},
{"HELLOMOTO", asset.Empty, errUnsupportedInstrumentFormat},
{"hi-my-name-is-moto", asset.Empty, errUnsupportedInstrumentFormat},
}
for _, test := range tc {
t.Run(test.instrument, func(t *testing.T) {
t.Parallel()
a, err := getAssetFromInstrument(test.instrument)
if test.expectedError != nil {
assert.ErrorIs(t, err, test.expectedError)
} else {
assert.NoError(t, err)
assert.Equal(t, test.expectedAsset, a)
}
})
}
}
func TestGetFeeByTypeOfflineTradeFee(t *testing.T) {
feeBuilder := &exchange.FeeBuilder{
Amount: 1,
FeeType: exchange.CryptocurrencyTradeFee,
Pair: futuresTradablePair,
IsMaker: false,
PurchasePrice: 1,
FiatCurrency: currency.USD,
BankTransactionType: exchange.WireTransfer,
}
result, err := e.GetFeeByType(t.Context(), feeBuilder)
require.NoError(t, err)
require.NotNil(t, result)
if !sharedtestvalues.AreAPICredentialsSet(e) {
assert.Equalf(t, exchange.OfflineTradeFee, feeBuilder.FeeType, "expected %v, received %v", exchange.OfflineTradeFee, feeBuilder.FeeType)
} else {
assert.Equalf(t, exchange.CryptocurrencyTradeFee, feeBuilder.FeeType, "expected %v, received %v", exchange.CryptocurrencyTradeFee, feeBuilder.FeeType)
}
}
func TestCalculateTradingFee(t *testing.T) {
t.Parallel()
feeBuilder := &exchange.FeeBuilder{
FeeType: exchange.CryptocurrencyTradeFee,
Pair: currency.Pair{Base: currency.BTC, Quote: currency.USD, Delimiter: currency.DashDelimiter},
IsMaker: true,
Amount: 1,
PurchasePrice: 1000,
}
var result float64
result, err := calculateTradingFee(feeBuilder)
require.NoError(t, err)
require.NotNil(t, result)
require.Equalf(t, 1e-1, result, "expected result %f, got %f", 1e-1, result)
// futures
feeBuilder.Pair, err = currency.NewPairFromString("BTC-21OCT22")
require.NoError(t, err)
result, err = calculateTradingFee(feeBuilder)
require.NoError(t, err)
require.NotNil(t, result)
require.Equalf(t, 0.1, result, "expected 0.1 but found %f", result)
// options
feeBuilder.Pair, err = currency.NewPairFromString("SOL-21OCT22-20-C")
require.NoError(t, err)
feeBuilder.IsMaker = false
result, err = calculateTradingFee(feeBuilder)
require.NoError(t, err)
require.NotNil(t, result)
require.Equalf(t, 0.3, result, "expected 0.3 but found %f", result)
// options
feeBuilder.Pair, err = currency.NewPairFromString("SOL-21OCT22-20-C,SOL-21OCT22-20-P")
require.NoError(t, err)
feeBuilder.IsMaker = true
_, err = calculateTradingFee(feeBuilder)
require.NoError(t, err)
require.NotNil(t, result)
require.Equalf(t, 0.3, result, "expected 0.3 but found %f", result)
// option_combo
feeBuilder.Pair, err = currency.NewPairFromString("BTC-STRG-21OCT22-19000_21000")
require.NoError(t, err)
feeBuilder.IsMaker = false
result, err = calculateTradingFee(feeBuilder)
require.NoError(t, err)
require.NotNil(t, result)
// future_combo
feeBuilder.Pair, err = currency.NewPairFromString("SOL-FS-30DEC22_28OCT22")
require.NoError(t, err)
feeBuilder.IsMaker = false
result, err = calculateTradingFee(feeBuilder)
require.NoError(t, err)
require.NotNil(t, result)
feeBuilder.Pair, err = currency.NewPairFromString("some_instrument_builder")
require.NoError(t, err)
require.NotNil(t, result)
_, err = calculateTradingFee(feeBuilder)
assert.ErrorIs(t, err, asset.ErrNotSupported)
}
func TestGetTime(t *testing.T) {
t.Parallel()
result, err := e.GetTime(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWrapperGetServerTime(t *testing.T) {
t.Parallel()
result, err := e.GetServerTime(t.Context(), asset.Empty)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestModifyOrder(t *testing.T) {
t.Parallel()
modifyParamToErrorMap := map[*order.Modify]error{
{OrderID: "1234"}: order.ErrPairIsEmpty,
{AssetType: asset.Spot}: order.ErrPairIsEmpty,
{AssetType: asset.Margin, OrderID: "1234", Pair: spotTradablePair}: asset.ErrNotSupported,
{AssetType: asset.Futures, OrderID: "1234", Pair: futuresTradablePair}: errInvalidAmount,
{AssetType: asset.Futures, Pair: futuresTradablePair, Amount: 2}: order.ErrOrderIDNotSet,
}
for param, errIncoming := range modifyParamToErrorMap {
_, err := e.ModifyOrder(t.Context(), param)
require.ErrorIs(t, err, errIncoming)
}
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.ModifyOrder(t.Context(), &order.Modify{AssetType: asset.Futures, OrderID: "1234", Pair: futuresTradablePair, Amount: 2})
require.NoError(t, err)
require.NotNil(t, result)
result, err = e.ModifyOrder(t.Context(), &order.Modify{AssetType: asset.Options, OrderID: "1234", Pair: optionsTradablePair, Amount: 2})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestCancelOrder(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
orderCancellation := &order.Cancel{
OrderID: "1",
AccountID: "1",
}
for assetType, cp := range assetTypeToPairsMap {
orderCancellation.AssetType = assetType
orderCancellation.Pair = cp
err := e.CancelOrder(t.Context(), orderCancellation)
require.NoError(t, err)
}
}
var websocketPushData = map[string]string{
"Announcement": `{"jsonrpc": "2.0","method": "subscription","params": {"channel": "announcements","data": { "action": "new", "body": "Lorem ipsum dolor sit amet, consectetur adipiscing elit.", "id": 1532593832021, "important": true, "publication_timestamp": 1532593832021, "title": "Example announcement" } }}`,
"Orderbook": `{"jsonrpc": "2.0", "method": "subscription", "params": { "channel": "book.BTC-PERPETUAL.100ms", "data": { "type": "snapshot", "timestamp": 1677589058217, "instrument_name": "BTC-PERPETUAL", "change_id": 53639437695, "bids": [ [ "new", 23461.0, 47800.0 ], [ "new", 23460.5, 37820.0 ], [ "new", 23460.0, 45720.0 ], [ "new", 23459.5, 24030.0 ], [ "new", 23459.0, 63600.0 ], [ "new", 23458.5, 60480.0 ], [ "new", 23458.0, 7960.0 ], [ "new", 23457.5, 2310.0 ], [ "new", 23457.0, 4270.0 ], [ "new", 23456.5, 44070.0 ], [ "new", 23456.0, 88690.0 ], [ "new", 23455.5, 5650.0 ], [ "new", 23455.0, 13420.0 ], [ "new", 23454.5, 116710.0 ], [ "new", 23454.0, 2010.0 ], [ "new", 23453.5, 200000.0 ], [ "new", 23452.5, 19950.0 ], [ "new", 23452.0, 39360.0 ], [ "new", 23451.5, 10000.0 ], [ "new", 23451.0, 239510.0 ], [ "new", 23450.5, 6250.0 ], [ "new", 23450.0, 40080.0 ], [ "new", 23449.5, 2000.0 ], [ "new", 23448.5, 500.0 ], [ "new", 23447.5, 179810.0 ], [ "new", 23447.0, 11000.0 ], [ "new", 23446.0, 57730.0 ], [ "new", 23445.0, 3640.0 ], [ "new", 23444.0, 17640.0 ], [ "new", 23443.5, 50000.0 ], [ "new", 23443.0, 6250.0 ], [ "new", 23441.5, 30330.0 ], [ "new", 23440.5, 76990.0 ], [ "new", 23440.0, 23910.0 ], [ "new", 23439.5, 3000.0 ], [ "new", 23439.0, 990.0 ], [ "new", 23438.0, 20760.0 ], [ "new", 23437.5, 500.0 ], [ "new", 23437.0, 84970.0 ], [ "new", 23436.5, 30040.0 ], [ "new", 23435.5, 322380.0 ], [ "new", 23434.0, 86280.0 ], [ "new", 23433.5, 187860.0 ], [ "new", 23433.0, 102360.0 ], [ "new", 23432.5, 48250.0 ], [ "new", 23432.0, 29070.0 ], [ "new", 23430.0, 119780.0 ], [ "new", 23429.0, 10.0 ], [ "new", 23428.0, 1510.0 ], [ "new", 23427.5, 2000.0 ], [ "new", 23427.0, 10.0 ], [ "new", 23426.5, 1840.0 ], [ "new", 23425.5, 2000.0 ], [ "new", 23425.0, 2250.0 ], [ "new", 23424.5, 600000.0 ], [ "new", 23424.0, 40870.0 ], [ "new", 23423.0, 117200.0 ], [ "new", 23422.0, 5000.0 ], [ "new", 23421.5, 80970.0 ], [ "new", 23420.0, 2420.0 ], [ "new", 23419.5, 200.0 ], [ "new", 23418.5, 40000.0 ], [ "new", 23415.0, 8020.0 ], [ "new", 23414.5, 57730.0 ], [ "new", 23413.5, 133250.0 ], [ "new", 23412.0, 40000.0 ], [ "new", 23410.5, 24000.0 ], [ "new", 23410.0, 80.0 ], [ "new", 23408.0, 36000.0 ], [ "new", 23407.0, 550000.0 ], [ "new", 23406.0, 30.0 ], [ "new", 23404.5, 230.0 ], [ "new", 23402.5, 57730.0 ], [ "new", 23401.0, 300010.0 ], [ "new", 23400.0, 520.0 ], [ "new", 23398.0, 28980.0 ], [ "new", 23394.5, 10.0 ], [ "new", 23391.5, 200.0 ], [ "new", 23391.0, 150000.0 ], [ "new", 23390.0, 80.0 ], [ "new", 23387.0, 403640.0 ], [ "new", 23385.5, 110.0 ], [ "new", 23385.0, 50.0 ], [ "new", 23384.5, 4690.0 ], [ "new", 23381.0, 200.0 ], [ "new", 23101.0, 9240.0 ], [ "new", 23100.5, 2320.0 ], [ "new", 23100.0, 15360.0 ], [ "new", 23096.0, 3000.0 ], [ "new", 23090.0, 90.0 ], [ "new", 23088.0, 3000.0 ], [ "new", 23087.0, 60.0 ], [ "new", 23081.5, 100.0 ], [ "new", 23080.0, 5400.0 ], [ "new", 23072.0, 3000.0 ], [ "new", 23070.0, 80.0 ], [ "new", 23064.0, 3000.0 ], [ "new", 23062.0, 3270.0 ], [ "new", 23060.0, 80.0 ], [ "new", 23056.0, 98000.0 ], [ "new", 23053.0, 3500.0 ], [ "new", 23050.5, 2370.0 ], [ "new", 23050.0, 32510.0 ], [ "new", 23048.0, 3000.0 ], [ "new", 23040.0, 3080.0 ], [ "new", 23038.0, 1000.0 ], [ "new", 23032.0, 5310.0 ], [ "new", 23030.0, 100.0 ], [ "new", 23024.0, 29000.0 ], [ "new", 23021.0, 2080.0 ], [ "new", 23020.0, 80.0 ], [ "new", 23016.0, 4150.0 ], [ "new", 23010.0, 80.0 ], [ "new", 23008.0, 3000.0 ], [ "new", 23005.0, 80.0 ], [ "new", 23004.5, 79200.0 ], [ "new", 23002.0, 20470.0 ], [ "new", 23001.0, 1000.0 ], [ "new", 23000.0, 8940.0 ], [ "new", 22992.0, 3000.0 ], [ "new", 22990.0, 2080.0 ], [ "new", 22984.0, 3000.0 ], [ "new", 22980.5, 2320.0 ], [ "new", 22980.0, 80.0 ], [ "new", 22976.0, 3000.0 ], [ "new", 22975.0, 52000.0 ], [ "new", 22971.0, 3600.0 ], [ "new", 22970.0, 2400.0 ], [ "new", 22968.0, 3000.0 ], [ "new", 22965.0, 270.0 ], [ "new", 22960.0, 3080.0 ], [ "new", 22956.0, 1000.0 ], [ "new", 22952.0, 3000.0 ], [ "new", 22951.0, 60.0 ], [ "new", 22950.0, 40200.0 ], [ "new", 22949.0, 1500.0 ], [ "new", 22944.0, 3000.0 ], [ "new", 22936.0, 3000.0 ], [ "new", 22934.0, 3000.0 ], [ "new", 22928.0, 3000.0 ], [ "new", 22925.0, 2370.0 ], [ "new", 22922.0, 80.0 ], [ "new", 22920.0, 3000.0 ], [ "new", 22916.0, 1150.0 ], [ "new", 22912.0, 3000.0 ], [ "new", 22904.5, 220.0 ], [ "new", 22904.0, 3000.0 ], [ "new", 22900.0, 273290.0 ], [ "new", 22896.0, 3000.0 ], [ "new", 22889.5, 100.0 ], [ "new", 22888.0, 7580.0 ], [ "new", 22880.0, 683400.0 ], [ "new", 22875.0, 400.0 ], [ "new", 22872.0, 3000.0 ], [ "new", 22870.0, 100.0 ], [ "new", 22864.0, 3000.0 ], [ "new", 22860.0, 2320.0 ], [ "new", 22856.0, 3000.0 ], [ "new", 22854.0, 10.0 ], [ "new", 22853.0, 500.0 ], [ "new", 22850.0, 1020.0 ], [ "new", 22848.0, 3000.0 ], [ "new", 22844.0, 25730.0 ], [ "new", 22840.0, 3000.0 ], [ "new", 22834.0, 3000.0 ], [ "new", 22832.0, 3000.0 ], [ "new", 22831.0, 200.0 ], [ "new", 22827.0, 40120.0 ], [ "new", 22824.0, 3000.0 ], [ "new", 22816.0, 4140.0 ], [ "new", 22808.0, 3000.0 ], [ "new", 22804.5, 220.0 ], [ "new", 22802.0, 50.0 ], [ "new", 22801.0, 1150.0 ], [ "new", 22800.0, 14050.0 ], [ "new", 22797.0, 10.0 ], [ "new", 22792.0, 3000.0 ], [ "new", 22789.0, 3000.0 ], [ "new", 22787.5, 5000.0 ], [ "new", 22784.0, 3000.0 ], [ "new", 22776.0, 3000.0 ], [ "new", 22775.0, 10000.0 ], [ "new", 22770.0, 200.0 ], [ "new", 22768.0, 14380.0 ], [ "new", 22760.0, 3000.0 ], [ "new", 22756.5, 2370.0 ], [ "new", 22752.0, 3000.0 ], [ "new", 22751.0, 47780.0 ], [ "new", 22750.0, 59970.0 ], [ "new", 22749.0, 50.0 ], [ "new", 22744.0, 3000.0 ], [ "new", 22736.0, 3000.0 ], [ "new", 22728.0, 3000.0 ], [ "new", 22726.0, 2320.0 ], [ "new", 22725.0, 20000.0 ], [ "new", 22720.0, 3000.0 ], [ "new", 22713.5, 250.0 ], [ "new", 22712.0, 3000.0 ], [ "new", 22709.0, 25000.0 ], [ "new", 22704.5, 220.0 ], [ "new", 22704.0, 3000.0 ], [ "new", 22702.0, 50.0 ], [ "new", 22700.0, 10230.0 ], [ "new", 22697.5, 10.0 ], [ "new", 22696.0, 3000.0 ], [ "new", 22688.0, 3000.0 ], [ "new", 22684.0, 10.0 ], [ "new", 22680.0, 3000.0 ], [ "new", 22672.0, 3000.0 ], [ "new", 22667.0, 2270.0 ], [ "new", 22664.0, 3000.0 ], [ "new", 22662.5, 2320.0 ], [ "new", 22657.5, 2340.0 ], [ "new", 22656.0, 3000.0 ], [ "new", 22655.0, 50.0 ], [ "new", 22653.0, 500.0 ], [ "new", 22650.0, 360120.0 ], [ "new", 22648.0, 3000.0 ], [ "new", 22640.0, 5320.0 ], [ "new", 22635.5, 2350.0 ], [ "new", 22632.0, 3000.0 ], [ "new", 22628.5, 2000.0 ], [ "new", 22626.5, 2350.0 ], [ "new", 22625.0, 400.0 ], [ "new", 22624.0, 3000.0 ], [ "new", 22616.0, 3000.0 ], [ "new", 22608.0, 3000.0 ], [ "new", 22604.5, 220.0 ], [ "new", 22601.0, 22600.0 ], [ "new", 22600.0, 696120.0 ], [ "new", 22598.5, 2320.0 ], [ "new", 22592.0, 3000.0 ], [ "new", 22584.0, 3000.0 ], [ "new", 22576.0, 3000.0 ], [ "new", 22568.0, 3000.0 ], [ "new", 22560.0, 25560.0 ], [ "new", 22552.5, 20.0 ], [ "new", 22550.0, 35760.0 ], [ "new", 22533.0, 2320.0 ], [ "new", 22530.0, 2320.0 ], [ "new", 22520.0, 1000.0 ], [ "new", 22505.0, 20000.0 ], [ "new", 22504.5, 220.0 ], [ "new", 22501.0, 45000.0 ], [ "new", 22500.0, 27460.0 ], [ "new", 22497.5, 1500.0 ], [ "new", 22485.0, 810.0 ], [ "new", 22481.0, 300.0 ], [ "new", 22465.5, 2320.0 ], [ "new", 22456.0, 2350.0 ], [ "new", 22453.0, 500.0 ], [ "new", 22450.0, 25000.0 ], [ "new", 22433.0, 141000.0 ], [ "new", 22431.0, 1940.0 ], [ "new", 22420.0, 2320.0 ], [ "new", 22419.5, 1000000.0 ], [ "new", 22400.0, 14280.0 ], [ "new", 22388.5, 30.0 ], [ "new", 22381.0, 100.0 ] ] } } }`,
"Orderbook Update": `{"params" : {"data" : {"type" : "snapshot","timestamp" : 1554373962454,"instrument_name" : "BTC-PERPETUAL","change_id" : 297217,"bids" : [["new",5042.34,30],["new",5041.94,20]],"asks" : [["new",5042.64,40],["new",5043.3,40]]},"channel" : "book.BTC-PERPETUAL.100ms"}, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Candlestick": `{"params" : {"data" : {"volume" : 0.05219351,"tick" : 1573645080000,"open" : 8869.79,"low" : 8788.25,"high" : 8870.31,"cost" : 460,"close" : 8791.25},"channel" : "chart.trades.BTC-PERPETUAL.1"}, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Index Price": `{"params" : {"data" : {"timestamp" : 1550588002899,"price" : 3937.89,"index_name" : "btc_usd"},"channel" : "deribit_price_index.btc_usd" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Price Ranking": `{"params" : {"data" :[{"weight" : 14.29,"timestamp" : 1573202284040,"price" : 9109.35,"original_price" : 9109.35,"identifier" : "bitfinex","enabled" : true},{"weight" : 14.29,"timestamp" : 1573202284055,"price" : 9084.83,"original_price" : 9084.83,"identifier" : "bitstamp","enabled" : true }, { "weight" : 14.29,"timestamp" : 1573202283191,"price" : 9079.91,"original_price" : 9079.91,"identifier" : "bittrex","enabled" : true }, { "weight" : 14.29,"timestamp" : 1573202284094,"price" : 9085.81,"original_price" : 9085.81,"identifier" : "coinbase","enabled" : true }, { "weight" : 14.29,"timestamp" : 1573202283881,"price" : 9086.27,"original_price" : 9086.27,"identifier" : "gemini","enabled" : true }, { "weight" : 14.29,"timestamp" : 1573202283420,"price" : 9088.38,"original_price" : 9088.38,"identifier" : "itbit","enabled" : true }, { "weight" : 14.29,"timestamp" : 1573202283459,"price" : 9083.6,"original_price" : 9083.6,"identifier" : "kraken","enabled" : true }, { "weight" : 0,"timestamp" : 0,"price" : null,"original_price" : null,"identifier" : "lmax","enabled" : false } ], "channel" : "deribit_price_ranking.btc_usd" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Price Statistics": `{"params" : {"data" : {"low24h" : 58012.08,"index_name" : "btc_usd","high_volatility" : false,"high24h" : 59311.42,"change24h" : 1009.61},"channel" : "deribit_price_statistics.btc_usd"}, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Volatility Index": `{"params" : {"data" : {"volatility" : 129.36,"timestamp" : 1619777946007,"index_name" : "btc_usd","estimated_delivery" : 129.36},"channel" : "deribit_volatility_index.btc_usd" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Estimated Expiration Price": `{"params" : {"data" : {"seconds" : 180929,"price" : 3939.73,"is_estimated" : false},"channel" : "estimated_expiration_price.btc_usd" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Incremental Ticker": `{"jsonrpc": "2.0", "method": "subscription", "params": { "channel": "incremental_ticker.BTC-PERPETUAL", "data": { "type": "snapshot", "timestamp": 1677592580023, "stats": { "volume_usd": 224579520.0, "volume": 9581.70741368, "price_change": -1.2945, "low": 23123.5, "high": 23900.0 }, "state": "open", "settlement_price": 23240.71, "open_interest": 333091400, "min_price": 23057.4, "max_price": 23759.65, "mark_price": 23408.41, "last_price": 23409.0, "interest_value": 0.0, "instrument_name": "BTC-PERPETUAL", "index_price": 23406.85, "funding_8h": 0.0, "estimated_delivery_price": 23406.85, "current_funding": 0.0, "best_bid_price": 23408.5, "best_bid_amount": 53270.0, "best_ask_price": 23409.0, "best_ask_amount": 46990.0 } } }`,
"Instrument State": `{"params" : {"data" : {"timestamp" : 1553080940000,"state" : "created","instrument_name" : "BTC-22MAR19"},"channel" : "instrument.state.any.any"}, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Currency Trades": `{"params":{"data":[{"trade_seq":2,"trade_id" : "48079289","timestamp" : 1590484589306,"tick_direction" : 2,"price" : 0.0075,"mark_price" : 0.01062686,"iv" : 47.58,"instrument_name" : "BTC-27MAY20-9000-C","index_price" : 8956.17,"direction" : "sell","amount" : 3}],"channel" : "trades.option.BTC.raw"},"method":"subscription","jsonrpc":"2.0"}`,
"Change Updates": `{"params" : {"data" : {"trades" : [{"trade_seq" : 866638,"trade_id" : "1430914","timestamp" : 1605780344032,"tick_direction" : 1,"state" : "filled","self_trade" : false,"reduce_only" : false,"profit_loss" : 0.00004898,"price" : 17391,"post_only" : false,"order_type" : "market","order_id" : "3398016","matching_id" : null,"mark_price" : 17391,"liquidity" : "T","instrument_name" : "BTC-PERPETUAL","index_price" : 17501.88,"fee_currency" : "BTC","fee" : 1.6e-7,"direction" : "sell","amount" : 10 } ],"positions" : [ { "total_profit_loss" : 1.69711368, "size_currency" : 10.646886321, "size" : 185160, "settlement_price" : 16025.83, "realized_profit_loss" : 0.012454598, "realized_funding" : 0.01235663, "open_orders_margin" : 0, "mark_price" : 17391, "maintenance_margin" : 0.234575865, "leverage" : 33, "kind" : "future", "interest_value" : 1.7362511643080387, "instrument_name" : "BTC-PERPETUAL", "initial_margin" : 0.319750953, "index_price" : 17501.88, "floating_profit_loss" : 0.906961435, "direction" : "buy", "delta" : 10.646886321, "average_price" : 15000 } ],"orders" : [ { "web" : true, "time_in_force" : "good_til_cancelled", "replaced" : false, "reduce_only" : false, "profit_loss" : 0.00009166, "price" : 15665.5, "post_only" : false, "order_type" : "market", "order_state" : "filled", "order_id" : "3398016", "max_show" : 10, "last_update_timestamp" : 1605780344032, "label" : "", "is_liquidation" : false, "instrument_name" : "BTC-PERPETUAL", "filled_amount" : 10, "direction" : "sell", "creation_timestamp" : 1605780344032, "commission" : 1.6e-7, "average_price" : 17391, "api" : false, "amount" : 10}],"instrument_name" : "BTC-PERPETUAL" }, "channel" : "user.changes.BTC-PERPETUAL.raw" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Currency Changes Updates": `{"params" : {"data" : {"trades" : [{"trade_seq" : 866638,"trade_id" : "1430914","timestamp" : 1605780344032,"tick_direction" : 1,"state" : "filled","self_trade" : false,"reduce_only" : false,"profit_loss" : 0.00004898,"price" : 17391,"post_only" : false,"order_type" : "market","order_id" : "3398016","matching_id" : null,"mark_price" : 17391,"liquidity" : "T","instrument_name" : "BTC-PERPETUAL","index_price" : 17501.88,"fee_currency" : "BTC","fee" : 1.6e-7,"direction" : "sell","amount" : 10 } ],"positions" : [ { "total_profit_loss" : 1.69711368, "size_currency" : 10.646886321, "size" : 185160, "settlement_price" : 16025.83, "realized_profit_loss" : 0.012454598, "realized_funding" : 0.01235663, "open_orders_margin" : 0, "mark_price" : 17391, "maintenance_margin" : 0.234575865, "leverage" : 33, "kind" : "future", "interest_value" : 1.7362511643080387, "instrument_name" : "BTC-PERPETUAL", "initial_margin" : 0.319750953, "index_price" : 17501.88, "floating_profit_loss" : 0.906961435, "direction" : "buy", "delta" : 10.646886321, "average_price" : 15000 } ],"orders" : [ { "web" : true, "time_in_force" : "good_til_cancelled", "replaced" : false, "reduce_only" : false, "profit_loss" : 0.00009166, "price" : 15665.5, "post_only" : false, "order_type" : "market", "order_state" : "filled", "order_id" : "3398016", "max_show" : 10, "last_update_timestamp" : 1605780344032, "label" : "", "is_liquidation" : false, "instrument_name" : "BTC-PERPETUAL", "filled_amount" : 10, "direction" : "sell", "creation_timestamp" : 1605780344032, "commission" : 1.6e-7, "average_price" : 17391, "api" : false, "amount" : 10 } ],"instrument_name" : "BTC-PERPETUAL" }, "channel" : "user.changes.future.BTC.raw" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"User Orders Raw Instrument": `{"params" : {"data" : {"time_in_force" : "good_til_cancelled","replaced" : false,"reduce_only" : false,"profit_loss" : 0,"price" : 10502.52,"post_only" : false,"original_order_type" : "market","order_type" : "limit","order_state" : "open","order_id" : "5","max_show" : 200,"last_update_timestamp" : 1581507423789,"label" : "","is_liquidation" : false,"instrument_name" : "BTC-PERPETUAL","filled_amount" : 0,"direction" : "buy","creation_timestamp" : 1581507423789,"commission" : 0,"average_price" : 0,"api" : false,"amount" : 200 }, "channel" : "user.orders.BTC-PERPETUAL.raw" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"User Orders By Instrument WithInterval": `{"params" : {"data" : [{"time_in_force" : "good_til_cancelled","replaced" : false,"reduce_only" : false,"profit_loss" : 0,"price" : 10460.43,"post_only" : false,"original_order_type" : "market","order_type" : "limit","order_state" : "open","order_id" : "4","max_show" : 200,"last_update_timestamp" : 1581507159533,"label" : "","is_liquidation" : false,"instrument_name" : "BTC-PERPETUAL","filled_amount" : 0,"direction" : "buy","creation_timestamp" : 1581507159533,"commission" : 0,"average_price" : 0,"api" : false,"amount" : 200 } ], "channel" : "user.orders.BTC-PERPETUAL.100ms" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"User Order By Currency Raw": `{"params" : {"data" : {"time_in_force" : "good_til_cancelled","replaced" : false,"reduce_only" : false,"profit_loss" : 0,"price" : 10542.68,"post_only" : false,"original_order_type" : "market","order_type" : "limit","order_state" : "open","order_id" : "6","max_show" : 200,"last_update_timestamp" : 1581507583024,"label" : "","is_liquidation" : false,"instrument_name" : "BTC-PERPETUAL","filled_amount" : 0,"direction" : "buy","creation_timestamp" : 1581507583024,"commission" : 0,"average_price" : 0,"api" : false,"amount" : 200 }, "channel" : "user.orders.any.any.raw" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"User Order By Currency WithInterval": `{"params" : {"data" : [{"time_in_force" : "good_til_cancelled","reduce_only" : false,"profit_loss" : 0,"price" : 3928.5,"post_only" : false,"order_type" : "limit","order_state" : "open","order_id" : "476137","max_show" : 120,"last_update_timestamp" : 1550826337209,"label" : "","is_liquidation" : false,"instrument_name" : "BTC-PERPETUAL","filled_amount" : 0,"direction" : "buy","creation_timestamp" : 1550826337209,"commission" : 0,"average_price" : 0,"api" : false,"amount" : 120 } ], "channel" : "user.orders.future.BTC.100ms" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"User Portfolio": `{"params" : {"data" : {"total_pl" : 0.00000425,"session_upl" : 0.00000425,"session_rpl" : -2e-8,"projected_maintenance_margin" : 0.00009141,"projected_initial_margin" : 0.00012542,"projected_delta_total" : 0.0043,"portfolio_margining_enabled" : false,"options_vega" : 0,"options_value" : 0,"options_theta" : 0,"options_session_upl" : 0,"options_session_rpl" : 0,"options_pl" : 0,"options_gamma" : 0,"options_delta" : 0,"margin_balance" : 0.2340038,"maintenance_margin" : 0.00009141,"initial_margin" : 0.00012542,"futures_session_upl" : 0.00000425,"futures_session_rpl" : -2e-8,"futures_pl" : 0.00000425,"estimated_liquidation_ratio" : 0.01822795,"equity" : 0.2340038,"delta_total" : 0.0043,"currency" : "BTC","balance" : 0.23399957,"available_withdrawal_funds" : 0.23387415,"available_funds" : 0.23387838 }, "channel" : "user.portfolio.btc" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"User Trades": `{"params" : {"data" : [{"trade_seq" :30289432,"trade_id":"48079254","timestamp":1590484156350,"tick_direction" : 0,"state" : "filled","self_trade" : false,"reduce_only" : false,"price" : 8954,"post_only" : false,"order_type" : "market","order_id" : "4008965646","matching_id" : null,"mark_price" : 8952.86,"liquidity" : "T","instrument_name" : "BTC-PERPETUAL","index_price" : 8956.73,"fee_currency" : "BTC","fee" : 0.00000168,"direction" : "sell","amount" : 20 }, { "trade_seq" : 30289433,"trade_id" : "48079255","timestamp" : 1590484156350,"tick_direction" : 1,"state" : "filled","self_trade" : false,"reduce_only" : false,"price" : 8954,"post_only" : false,"order_type" : "market","order_id" : "4008965646","matching_id" : null,"mark_price" : 8952.86,"liquidity" : "T","instrument_name" : "BTC-PERPETUAL","index_price" : 8956.73,"fee_currency" : "BTC","fee" : 0.00000168,"direction" : "sell","amount" : 20 }],"channel" : "user.trades.BTC-PERPETUAL.raw" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"User Trades With Currency": `{"params" : {"data" : [{"trade_seq" :74405, "trade_id":"48079262","timestamp":1590484255886,"tick_direction" : 2,"state" : "filled","self_trade" : false,"reduce_only" : false,"price" : 8947,"post_only" : false,"order_type" : "limit","order_id" : "4008978075","matching_id" : null,"mark_price" : 8970.03,"liquidity" : "T","instrument_name" : "BTC-25SEP20","index_price" : 8953.53,"fee_currency" : "BTC","fee" : 0.00049961,"direction" : "sell","amount" : 8940 } ], "channel" : "user.trades.future.BTC.100ms" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Mark Price Options": `{"params" : {"data" : [{"timestamp" : 1622470378005,"mark_price" : 0.0333,"iv" : 0.9,"instrument_name" : "BTC-2JUN21-37000-P"},{"timestamp" : 1622470378005,"mark_price" : 0.117,"iv" : 0.9,"instrument_name" : "BTC-4JUN21-40500-P"},{"timestamp" : 1622470378005,"mark_price" : 0.0177,"iv" : 0.9,"instrument_name" : "BTC-4JUN21-38250-C"},{"timestamp" : 1622470378005,"mark_price" : 0.0098,"iv" : 0.9,"instrument_name" : "BTC-1JUN21-37000-C" }, { "timestamp" : 1622470378005,"mark_price" : 0.0371,"iv" : 0.9,"instrument_name" : "BTC-4JUN21-36500-P" } ], "channel" : "markprice.options.btc_usd" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Platform State": `{"params" : {"data" : {"allow_unauthenticated_public_requests" : true},"channel" : "platform_state.public_methods_state" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Quote Ticker": `{"params" : {"data" : {"timestamp" : 1550658624149,"instrument_name" : "BTC-PERPETUAL","best_bid_price" : 3914.97,"best_bid_amount" : 40,"best_ask_price" : 3996.61,"best_ask_amount" : 50},"channel" : "quote.BTC-PERPETUAL" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Rfq": `{"params" : {"data" : {"state" : true,"side" : null,"last_rfq_tstamp" : 1634816143836,"instrument_name" : "BTC-PERPETUAL","amount" : null },"channel" : "rfq.btc" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
"Instrument Trades": `{"params":{"data":[{"trade_seq":30289442,"trade_id" : "48079269","timestamp" : 1590484512188,"tick_direction" : 2,"price" : 8950,"mark_price" : 8948.9,"instrument_name" : "BTC-PERPETUAL","index_price" : 8955.88,"direction" : "sell","amount" : 10}],"channel" : "trades.BTC-PERPETUAL.raw"},"method":"subscription","jsonrpc":"2.0"}`,
"Instruments Ticker": `{"params" : {"data" : {"timestamp" : 1623060194301,"stats" : {"volume_usd" : 284061480,"volume" : 7871.02139035,"price_change" : 0.7229,"low" : 35213.5,"high" : 36824.5},"state" : "open","settlement_price" : 36169.49,"open_interest" : 502097590,"min_price" : 35898.37,"max_price" : 36991.72,"mark_price" : 36446.51,"last_price" : 36457.5,"interest_value" : 1.7362511643080387,"instrument_name" : "BTC-PERPETUAL","index_price" : 36441.64,"funding_8h" : 0.0000211,"estimated_delivery_price" : 36441.64,"current_funding" : 0,"best_bid_price" : 36442.5,"best_bid_amount" : 5000,"best_ask_price" : 36443,"best_ask_amount" : 100 }, "channel" : "ticker.BTC-PERPETUAL.raw" }, "method" : "subscription", "jsonrpc" : "2.0" }`,
}
func TestProcessPushData(t *testing.T) {
t.Parallel()
for k, v := range websocketPushData {
t.Run(k, func(t *testing.T) {
t.Parallel()
err := e.wsHandleData(t.Context(), []byte(v))
require.NoError(t, err, "wsHandleData must not error")
})
}
}
func TestFormatFuturesTradablePair(t *testing.T) {
t.Parallel()
futuresInstrumentsOutputList := map[currency.Pair]string{
{Delimiter: currency.DashDelimiter, Base: currency.BTC, Quote: currency.NewCode(perpString)}: "BTC-PERPETUAL",
{Delimiter: currency.DashDelimiter, Base: currency.AVAX, Quote: currency.NewCode("USDC-PERPETUAL")}: "AVAX_USDC-PERPETUAL",
{Delimiter: currency.DashDelimiter, Base: currency.ETH, Quote: currency.NewCode("30DEC22")}: "ETH-30DEC22",
{Delimiter: currency.DashDelimiter, Base: currency.SOL, Quote: currency.NewCode("30DEC22")}: "SOL-30DEC22",
{Delimiter: currency.DashDelimiter, Base: currency.NewCode("BTCDVOL"), Quote: currency.NewCode("USDC-28JUN23")}: "BTCDVOL_USDC-28JUN23",
}
for pair, instrumentID := range futuresInstrumentsOutputList {
t.Run(instrumentID, func(t *testing.T) {
t.Parallel()
instrument := formatFuturesTradablePair(pair)
require.Equal(t, instrumentID, instrument)
})
}
}
func TestOptionPairToString(t *testing.T) {
t.Parallel()
for pair, exp := range map[currency.Pair]string{
{Delimiter: currency.DashDelimiter, Base: currency.BTC, Quote: currency.NewCode("30MAY24-61000-C")}: "BTC-30MAY24-61000-C",
{Delimiter: currency.DashDelimiter, Base: currency.ETH, Quote: currency.NewCode("1JUN24-3200-P")}: "ETH-1JUN24-3200-P",
{Delimiter: currency.DashDelimiter, Base: currency.SOL, Quote: currency.NewCode("USDC-31MAY24-162-P")}: "SOL_USDC-31MAY24-162-P",
{Delimiter: currency.DashDelimiter, Base: currency.MATIC, Quote: currency.NewCode("USDC-6APR24-0d98-P")}: "MATIC_USDC-6APR24-0d98-P",
{Delimiter: currency.DashDelimiter, Base: currency.MATIC, Quote: currency.NewCode("USDC-8JUN24-0D99-P")}: "MATIC_USDC-8JUN24-0d99-P",
{Delimiter: currency.DashDelimiter, Base: currency.MATIC, Quote: currency.NewCode("USDC-6DEC29-0D87-C")}: "MATIC_USDC-6DEC29-0d87-C",
} {
assert.Equal(t, exp, optionPairToString(pair), "optionPairToString should return correctly")
}
}
func TestWSRetrieveCombos(t *testing.T) {
t.Parallel()
_, err := e.WSRetrieveCombos(t.Context(), currency.EMPTYCODE)
require.ErrorIs(t, err, currency.ErrCurrencyCodeEmpty)
result, err := e.WSRetrieveCombos(t.Context(), futureComboTradablePair.Base)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetLatestFundingRates(t *testing.T) {
t.Parallel()
_, err := e.GetLatestFundingRates(t.Context(), &fundingrate.LatestRateRequest{
Asset: asset.USDTMarginedFutures,
Pair: currency.NewBTCUSDT(),
IncludePredictedRate: true,
})
require.ErrorIs(t, err, asset.ErrNotSupported)
result, err := e.GetLatestFundingRates(t.Context(), &fundingrate.LatestRateRequest{
Asset: asset.Futures,
Pair: futuresTradablePair,
})
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestUpdateOrderExecutionLimits(t *testing.T) {
t.Parallel()
for _, a := range e.GetAssetTypes(false) {
t.Run(a.String(), func(t *testing.T) {
t.Parallel()
require.NoError(t, e.UpdateOrderExecutionLimits(t.Context(), a), "UpdateOrderExecutionLimits must not error")
pairs, err := e.CurrencyPairs.GetPairs(a, true)
require.NoError(t, err, "GetPairs must not error")
l, err := e.GetOrderExecutionLimits(a, pairs[0])
require.NoError(t, err, "GetOrderExecutionLimits must not error")
assert.Positive(t, l.MinimumBaseAmount, "MinimumBaseAmount should be positive")
assert.Positive(t, l.PriceStepIncrementSize, "PriceStepIncrementSize should be positive")
})
}
}
func TestGetLockedStatus(t *testing.T) {
t.Parallel()
result, err := e.GetLockedStatus(t.Context())
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestSayHello(t *testing.T) {
t.Parallel()
result, err := e.SayHello(t.Context(), "Thrasher", "")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsRetrieveCancelOnDisconnect(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WsRetrieveCancelOnDisconnect(t.Context(), "connection")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsDisableCancelOnDisconnect(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WsDisableCancelOnDisconnect(t.Context(), "connection")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestEnableCancelOnDisconnect(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.EnableCancelOnDisconnect(t.Context(), "account")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsEnableCancelOnDisconnect(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
result, err := e.WsEnableCancelOnDisconnect(t.Context(), "connection")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestLogout(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateRealOrders)
err := e.WsLogout(t.Context(), true)
assert.NoError(t, err)
}
func TestExchangeToken(t *testing.T) {
t.Parallel()
_, err := e.ExchangeToken(t.Context(), "", 1234)
require.ErrorIs(t, err, errRefreshTokenRequired)
_, err = e.ExchangeToken(t.Context(), "1568800656974.1CWcuzUS.MGy49NK4hpTwvR1OYWfpqMEkH4T4oDg4tNIcrM7KdeyxXRcSFqiGzA_D4Cn7mqWocHmlS89FFmUYcmaN2H7lNKKTnhRg5EtrzsFCCiuyN0Wv9y-LbGLV3-Ojv_kbD50FoScQ8BDXS5b_w6Ir1MqEdQ3qFZ3MLcvlPiIgG2BqyJX3ybYnVpIlrVrrdYD1-lkjLcjxOBNJvvUKNUAzkQ", 0)
require.ErrorIs(t, err, errSubjectIDRequired)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.ExchangeToken(t.Context(), "1568800656974.1CWcuzUS.MGy49NK4hpTwvR1OYWfpqMEkH4T4oDg4tNIcrM7KdeyxXRcSFqiGzA_D4Cn7mqWocHmlS89FFmUYcmaN2H7lNKKTnhRg5EtrzsFCCiuyN0Wv9y-LbGLV3-Ojv_kbD50FoScQ8BDXS5b_w6Ir1MqEdQ3qFZ3MLcvlPiIgG2BqyJX3ybYnVpIlrVrrdYD1-lkjLcjxOBNJvvUKNUAzkQ", 1234)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsExchangeToken(t *testing.T) {
t.Parallel()
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
result, err := e.WsExchangeToken(t.Context(), "1568800656974.1CWcuzUS.MGy49NK4hpTwvR1OYWfpqMEkH4T4oDg4tNIcrM7KdeyxXRcSFqiGzA_D4Cn7mqWocHmlS89FFmUYcmaN2H7lNKKTnhRg5EtrzsFCCiuyN0Wv9y-LbGLV3-Ojv_kbD50FoScQ8BDXS5b_w6Ir1MqEdQ3qFZ3MLcvlPiIgG2BqyJX3ybYnVpIlrVrrdYD1-lkjLcjxOBNJvvUKNUAzkQ", 1234)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestForkToken(t *testing.T) {
t.Parallel()
_, err := e.ForkToken(t.Context(), "", "Sami")
require.ErrorIs(t, err, errRefreshTokenRequired)
_, err = e.ForkToken(t.Context(), "1568800656974.1CWcuzUS.MGy49NK4hpTwvR1OYWfpqMEkH4T4oDg4tNIcrM7KdeyxXRcSFqiGzA_D4Cn7mqWocHmlS89FFmUYcmaN2H7lNKKTnhRg5EtrzsFCCiuyN0Wv9y-LbGLV3-Ojv_kbD50FoScQ8BDXS5b_w6Ir1MqEdQ3qFZ3MLcvlPiIgG2BqyJX3ybYnVpIlrVrrdYD1-lkjLcjxOBNJvvUKNUAzkQ", "")
require.ErrorIs(t, err, errSessionNameRequired)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.ForkToken(t.Context(), "1568800656974.1CWcuzUS.MGy49NK4hpTwvR1OYWfpqMEkH4T4oDg4tNIcrM7KdeyxXRcSFqiGzA_D4Cn7mqWocHmlS89FFmUYcmaN2H7lNKKTnhRg5EtrzsFCCiuyN0Wv9y-LbGLV3-Ojv_kbD50FoScQ8BDXS5b_w6Ir1MqEdQ3qFZ3MLcvlPiIgG2BqyJX3ybYnVpIlrVrrdYD1-lkjLcjxOBNJvvUKNUAzkQ", "Sami")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestWsForkToken(t *testing.T) {
t.Parallel()
_, err := e.WsForkToken(t.Context(), "", "Sami")
require.ErrorIs(t, err, errRefreshTokenRequired)
_, err = e.WsForkToken(t.Context(), "1568800656974.1CWcuzUS.MGy49NK4hpTwvR1OYWfpqMEkH4T4oDg4tNIcrM7KdeyxXRcSFqiGzA_D4Cn7mqWocHmlS89FFmUYcmaN2H7lNKKTnhRg5EtrzsFCCiuyN0Wv9y-LbGLV3-Ojv_kbD50FoScQ8BDXS5b_w6Ir1MqEdQ3qFZ3MLcvlPiIgG2BqyJX3ybYnVpIlrVrrdYD1-lkjLcjxOBNJvvUKNUAzkQ", "")
require.ErrorIs(t, err, errSessionNameRequired)
sharedtestvalues.SkipTestIfCredentialsUnset(t, e, canManipulateAPIEndpoints)
result, err := e.WsForkToken(t.Context(), "1568800656974.1CWcuzUS.MGy49NK4hpTwvR1OYWfpqMEkH4T4oDg4tNIcrM7KdeyxXRcSFqiGzA_D4Cn7mqWocHmlS89FFmUYcmaN2H7lNKKTnhRg5EtrzsFCCiuyN0Wv9y-LbGLV3-Ojv_kbD50FoScQ8BDXS5b_w6Ir1MqEdQ3qFZ3MLcvlPiIgG2BqyJX3ybYnVpIlrVrrdYD1-lkjLcjxOBNJvvUKNUAzkQ", "Sami")
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetFuturesContractDetails(t *testing.T) {
t.Parallel()
_, err := e.GetFuturesContractDetails(t.Context(), asset.Binary)
require.ErrorIs(t, err, futures.ErrNotFuturesAsset)
result, err := e.GetFuturesContractDetails(t.Context(), asset.Futures)
require.NoError(t, err)
assert.NotNil(t, result)
_, err = e.GetFuturesContractDetails(t.Context(), asset.FutureCombo)
require.ErrorIs(t, err, asset.ErrNotSupported)
}
func TestGetFuturesPositionSummary(t *testing.T) {
t.Parallel()
paramToErrorMap := map[*futures.PositionSummaryRequest]error{
nil: common.ErrNilPointer,
{}: futures.ErrNotPerpetualFuture,
{Asset: asset.Futures}: currency.ErrCurrencyPairEmpty,
}
for param, errIncoming := range paramToErrorMap {
_, err := e.GetFuturesPositionSummary(t.Context(), param)
require.ErrorIs(t, err, errIncoming)
}
sharedtestvalues.SkipTestIfCredentialsUnset(t, e)
req := &futures.PositionSummaryRequest{
Asset: asset.Futures,
Pair: currency.NewPair(currency.BTC, currency.NewCode(perpString)),
}
result, err := e.GetFuturesPositionSummary(t.Context(), req)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestGetOpenInterest(t *testing.T) {
t.Parallel()
_, err := e.GetOpenInterest(t.Context(), key.PairAsset{
Base: currency.SOL.Item,
Quote: currency.USDC.Item,
Asset: asset.Spot,
})
require.ErrorIs(t, err, asset.ErrNotSupported)
_, err = e.GetOpenInterest(t.Context(), key.PairAsset{
Base: optionsTradablePair.Base.Item,
Quote: optionsTradablePair.Quote.Item,
Asset: asset.Options,
})
require.NoError(t, err)
_, err = e.GetOpenInterest(t.Context(), key.PairAsset{
Base: currency.BTC.Item,
Quote: currency.NewCode(perpString).Item,
Asset: asset.Futures,
})
require.NoError(t, err)
_, err = e.GetOpenInterest(t.Context(), key.PairAsset{
Base: currency.NewCode("XRP").Item,
Quote: currency.NewCode("USDC-PERPETUAL").Item,
Asset: asset.Futures,
})
require.NoError(t, err)
_, err = e.GetOpenInterest(t.Context(), key.PairAsset{
Base: futureComboTradablePair.Base.Item,
Quote: futureComboTradablePair.Quote.Item,
Asset: asset.FutureCombo,
})
require.NoError(t, err)
}
func TestIsPerpetualFutureCurrency(t *testing.T) {
t.Parallel()
assetPairToErrorMap := map[asset.Item][]struct {
Pair currency.Pair
Error error
Response bool
}{
asset.Spot: {
{Pair: currency.EMPTYPAIR, Error: currency.ErrCurrencyPairEmpty, Response: false},
{Pair: spotTradablePair, Error: nil, Response: false},
},
asset.Futures: {
{Pair: currency.NewPair(currency.BTC, currency.NewCode(perpString)), Response: true},
},
asset.FutureCombo: {
{Pair: currency.NewPair(currency.NewCode("BTC"), currency.NewCode("FS-27SEP24_PERP")), Response: false},
},
asset.OptionCombo: {
{Pair: currency.NewPair(currency.NewCode(currencyBTC), currency.NewCode("STRG-21OCT22")), Error: nil, Response: false},
},
}
for assetType, instances := range assetPairToErrorMap {
for i := range instances {
t.Run(fmt.Sprintf("Asset: %s Pair: %s", assetType.String(), instances[i].Pair.String()), func(t *testing.T) {
t.Parallel()
is, err := e.IsPerpetualFutureCurrency(assetType, instances[i].Pair)
require.ErrorIs(t, err, instances[i].Error)
require.Equal(t, is, instances[i].Response)
})
}
}
}
func TestGetHistoricalFundingRates(t *testing.T) {
t.Parallel()
cp, err := currency.NewPairFromString("BTC-PERPETUAL")
require.NoError(t, err)
r := &fundingrate.HistoricalRatesRequest{
Asset: asset.Spot,
Pair: cp,
PaymentCurrency: currency.USDT,
StartDate: time.Now().Add(-time.Hour * 24 * 2),
EndDate: time.Now(),
}
_, err = e.GetHistoricalFundingRates(t.Context(), r)
require.ErrorIs(t, err, asset.ErrNotSupported)
r.Asset = asset.Futures
result, err := e.GetHistoricalFundingRates(t.Context(), r)
require.NoError(t, err)
assert.NotNil(t, result)
}
func TestMultipleCancelResponseUnmarshalJSON(t *testing.T) {
t.Parallel()
resp := &struct {
Result *MultipleCancelResponse `json:"result"`
}{}
data := `{ "jsonrpc": "2.0", "id": 8748, "result": 37 }`
err := json.Unmarshal([]byte(data), &resp)
require.NoError(t, err)
require.Equal(t, int64(37), resp.Result.CancelCount)
data = `{"jsonrpc":"2.0","id":1599612810505,"result":[{"instrument_name":"BTC_USDC","currency":"BTC_USDC","result":[{"is_rebalance":false,"risk_reducing":false,"order_type":"limit","creation_timestamp":1715302998260,"order_state":"cancelled","contracts":300000.0,"average_price":0.0,"post_only":false,"last_update_timestamp":1715303041949,"filled_amount":0.0,"replaced":false,"web":false,"api":true,"mmp":false,"cancel_reason":"user_request","instrument_name":"BTC_USDC","order_id":"BTC_USDC-13133482","max_show":30.0,"time_in_force":"good_til_cancelled","direction":"buy","amount":30.0,"price":30.0,"label":"test"}],"type":"limit"}]}`
err = json.Unmarshal([]byte(data), &resp)
require.NoError(t, err)
require.Equal(t, int64(1), resp.Result.CancelCount)
require.Len(t, resp.Result.CancelDetails, 1)
require.Len(t, resp.Result.CancelDetails, 1)
}
func TestGetResolutionFromInterval(t *testing.T) {
t.Parallel()
intervalStringMap := []struct {
Interval kline.Interval
IntervalString string
Error error
}{
{Interval: kline.HundredMilliseconds, IntervalString: "100ms"},
{Interval: kline.OneMin, IntervalString: "1"},
{Interval: kline.ThreeMin, IntervalString: "3"},
{Interval: kline.FiveMin, IntervalString: "5"},
{Interval: kline.TenMin, IntervalString: "10"},
{Interval: kline.FifteenMin, IntervalString: "15"},
{Interval: kline.ThirtyMin, IntervalString: "30"},
{Interval: kline.OneHour, IntervalString: "60"},
{Interval: kline.TwoHour, IntervalString: "120"},
{Interval: kline.ThreeHour, IntervalString: "180"},
{Interval: kline.SixHour, IntervalString: "360"},
{Interval: kline.TwelveHour, IntervalString: "720"},
{Interval: kline.OneDay, IntervalString: "1D"},
{Interval: kline.Raw, IntervalString: "raw"},
{Interval: kline.FourHour, Error: kline.ErrUnsupportedInterval},
}
for x := range intervalStringMap {
result, err := e.GetResolutionFromInterval(intervalStringMap[x].Interval)
require.ErrorIs(t, err, intervalStringMap[x].Error)
require.Equal(t, intervalStringMap[x].IntervalString, result)
}
}
func TestGetValidatedCurrencyCode(t *testing.T) {
t.Parallel()
pairs := map[currency.Pair]string{
currency.NewPairWithDelimiter(currencySOL, "21OCT22-20-C", "-"): currencySOL,
currency.NewPairWithDelimiter(currencyBTC, perpString, "-"): currencyBTC,
currency.NewPairWithDelimiter(currencyETH, perpString, "-"): currencyETH,
currency.NewPairWithDelimiter(currencySOL, perpString, "-"): currencySOL,
currency.NewPairWithDelimiter("AVAX_USDC", perpString, "-"): currencyUSDC,
currency.NewPairWithDelimiter(currencyBTC, "USDC", "_"): currencyBTC,
currency.NewPairWithDelimiter(currencyETH, "USDC", "_"): currencyETH,
currency.NewPairWithDelimiter("DOT", "USDC-PERPETUAL", "_"): currencyUSDC,
currency.NewPairWithDelimiter("DOT", "USDT-PERPETUAL", "_"): currencyUSDT,
currency.EMPTYPAIR: "any",
}
for x := range pairs {
result := getValidatedCurrencyCode(x)
require.Equalf(t, pairs[x], result, "expected: %s actual : %s for currency pair: %v", x, result, pairs[x])
}
}
func TestGetCurrencyTradeURL(t *testing.T) {
t.Parallel()
_, err := e.GetCurrencyTradeURL(t.Context(), asset.Spot, currency.EMPTYPAIR)
require.ErrorIs(t, err, currency.ErrCurrencyPairEmpty)
for _, a := range e.GetAssetTypes(false) {
var pairs currency.Pairs
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)
var resp string
resp, err = e.GetCurrencyTradeURL(t.Context(), a, pairs[0])
require.NoError(t, err)
assert.NotEmpty(t, resp)
}
// specific test to ensure perps work
cp := currency.NewPair(currency.BTC, currency.NewCode("USDC-PERPETUAL"))
resp, err := e.GetCurrencyTradeURL(t.Context(), asset.Futures, cp)
require.NoError(t, err)
assert.NotEmpty(t, resp)
// specific test to ensure options with dates work
cp = currency.NewPair(currency.BTC, currency.NewCode("14JUN24-62000-C"))
resp, err = e.GetCurrencyTradeURL(t.Context(), asset.Options, cp)
require.NoError(t, err)
assert.NotEmpty(t, resp)
}
func TestFormatChannelPair(t *testing.T) {
t.Parallel()
pair := currency.NewPair(currency.BTC, currency.NewCode("USDC-PERPETUAL"))
pair.Delimiter = "-"
assert.Equal(t, "BTC_USDC-PERPETUAL", formatChannelPair(pair))
pair = currency.NewPair(currency.BTC, currency.NewCode("PERPETUAL"))
pair.Delimiter = "-"
assert.Equal(t, "BTC-PERPETUAL", formatChannelPair(pair))
}
var timeInForceList = []struct {
String string
PostOnly bool
TIF order.TimeInForce
Error error
}{
{"good_til_cancelled", false, order.GoodTillCancel, nil},
{"good_til_cancelled", true, order.GoodTillCancel | order.PostOnly, nil},
{"good_til_day", false, order.GoodTillDay, nil},
{"good_til_day", true, order.GoodTillDay | order.PostOnly, nil},
{"fill_or_kill", false, order.FillOrKill, nil},
{"immediate_or_cancel", false, order.ImmediateOrCancel, nil},
{"abcd", false, order.UnknownTIF, order.ErrInvalidTimeInForce},
{"", false, order.UnknownTIF, nil},
}
func TestTimeInForceFromString(t *testing.T) {
t.Parallel()
for i := range timeInForceList {
result, err := timeInForceFromString(timeInForceList[i].String, timeInForceList[i].PostOnly)
assert.Equalf(t, timeInForceList[i].TIF, result, "expected %s, got %s", timeInForceList[i].TIF.String(), result.String())
require.ErrorIs(t, err, timeInForceList[i].Error)
}
}
func TestOptionsComboFormatting(t *testing.T) {
t.Parallel()
availablePairs, err := e.GetAvailablePairs(asset.OptionCombo)
require.NoError(t, err, "GetAvailablePairs must not error")
require.GreaterOrEqual(t, len(availablePairs), 5, "availablePairs must be greater than or equal 5")
for _, cp := range availablePairs[:5] {
t.Run(cp.String(), func(t *testing.T) {
t.Parallel()
_, err := e.GetPublicTicker(t.Context(), optionComboPairToString(cp))
assert.NoError(t, err, "GetPublicTicker should not error")
})
}
}
func TestAppendCandles(t *testing.T) {
t.Parallel()
_, err := appendCandles(nil, time.Time{})
assert.ErrorIs(t, err, kline.ErrNoTimeSeriesDataToConvert)
candles := &TVChartData{
Ticks: []int64{1337},
}
_, err = appendCandles(candles, time.Time{})
assert.ErrorIs(t, err, kline.ErrInsufficientCandleData)
candles = &TVChartData{
Open: []float64{1337},
High: []float64{1337},
Low: []float64{1337},
Close: []float64{1337},
Volume: []float64{1337},
Ticks: []int64{1337},
}
resp, err := appendCandles(candles, time.Time{})
assert.NoError(t, err)
assert.Len(t, resp, 1)
candles = &TVChartData{
Open: []float64{1337},
High: []float64{1337},
Low: []float64{1337},
Close: []float64{1337},
Volume: []float64{1337},
Ticks: []int64{1337},
}
resp, err = appendCandles(candles, time.Unix(1338, 0))
assert.NoError(t, err)
assert.Empty(t, resp)
}