freqtrade_origin/tests/exchange_online/test_ccxt_compat.py

490 lines
22 KiB
Python
Raw Normal View History

2020-10-13 17:54:27 +00:00
"""
Tests in this file do NOT mock network calls, so they are expected to be fluky at times.
However, these tests should give a good idea to determine if a new exchange is
suitable to run with freqtrade.
"""
from datetime import datetime, timedelta, timezone
2020-12-23 14:55:46 +00:00
import pytest
from freqtrade.enums import CandleType
from freqtrade.exchange import timeframe_to_minutes, timeframe_to_prev_date
from freqtrade.exchange.exchange import timeframe_to_msecs
from freqtrade.util import dt_floor_day, dt_now, dt_ts
from tests.exchange_online.conftest import EXCHANGE_FIXTURE_TYPE, EXCHANGES
2021-11-12 06:26:59 +00:00
2023-02-11 07:13:46 +00:00
@pytest.mark.longrun
class TestCCXTExchange:
2022-12-31 09:59:26 +00:00
def test_load_markets(self, exchange: EXCHANGE_FIXTURE_TYPE):
exch, exchangename = exchange
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename]["pair"]
2022-12-31 09:59:26 +00:00
markets = exch.markets
2020-10-23 18:46:01 +00:00
assert pair in markets
assert isinstance(markets[pair], dict)
2022-12-31 09:59:26 +00:00
assert exch.market_is_spot(markets[pair])
2022-12-31 09:59:26 +00:00
def test_has_validations(self, exchange: EXCHANGE_FIXTURE_TYPE):
exch, exchangename = exchange
2022-07-11 08:43:21 +00:00
2024-05-12 13:38:09 +00:00
exch.validate_ordertypes(
{
"entry": "limit",
"exit": "limit",
"stoploss": "limit",
}
)
2022-07-11 08:43:21 +00:00
2024-05-12 13:38:09 +00:00
if exchangename == "gate":
2023-02-10 19:58:02 +00:00
# gate doesn't have market orders on spot
2022-07-11 08:43:21 +00:00
return
2024-05-12 13:38:09 +00:00
exch.validate_ordertypes(
{
"entry": "market",
"exit": "market",
"stoploss": "market",
}
)
2022-07-11 08:43:21 +00:00
2022-12-31 09:59:26 +00:00
def test_load_markets_futures(self, exchange_futures: EXCHANGE_FIXTURE_TYPE):
exchange, exchangename = exchange_futures
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename]["pair"]
pair = EXCHANGES[exchangename].get("futures_pair", pair)
markets = exchange.markets
assert pair in markets
assert isinstance(markets[pair], dict)
assert exchange.market_is_future(markets[pair])
2020-10-13 17:54:27 +00:00
def test_ccxt_order_parse(self, exchange: EXCHANGE_FIXTURE_TYPE):
exch, exchange_name = exchange
2024-05-12 13:38:09 +00:00
if orders := EXCHANGES[exchange_name].get("sample_order"):
pair = "SOL/USDT"
for order in orders:
2023-09-18 17:09:12 +00:00
market = exch._api.markets[pair]
po = exch._api.parse_order(order, market)
2024-05-12 13:38:09 +00:00
assert isinstance(po["id"], str)
assert po["id"] is not None
2023-01-31 19:59:55 +00:00
if len(order.keys()) < 5:
# Kucoin case
2024-05-12 13:38:09 +00:00
assert po["status"] is None
2023-01-31 19:59:55 +00:00
continue
2024-05-12 13:38:09 +00:00
assert po["timestamp"] == 1674493798550
assert isinstance(po["datetime"], str)
assert isinstance(po["timestamp"], int)
assert isinstance(po["price"], float)
assert po["price"] == 15.5
if po["status"] == "closed":
# Filled orders should have average assigned.
2024-05-12 13:38:09 +00:00
assert isinstance(po["average"], float)
assert po["average"] == 15.5
assert po["symbol"] == pair
assert isinstance(po["amount"], float)
assert po["amount"] == 1.1
assert isinstance(po["status"], str)
else:
pytest.skip(f"No sample order available for exchange {exchange_name}")
def test_ccxt_my_trades_parse(self, exchange: EXCHANGE_FIXTURE_TYPE):
exch, exchange_name = exchange
if trades := EXCHANGES[exchange_name].get("sample_my_trades"):
pair = "SOL/USDT"
for trade in trades:
market = exch._api.markets[pair]
po = exch._api.parse_trade(trade)
(trade, market)
assert isinstance(po["id"], str)
assert isinstance(po["side"], str)
assert isinstance(po["amount"], float)
assert isinstance(po["price"], float)
assert isinstance(po["datetime"], str)
assert isinstance(po["timestamp"], int)
if fees := po.get("fees"):
assert isinstance(fees, list)
for fee in fees:
assert isinstance(fee, dict)
assert isinstance(fee["cost"], str)
# TODO: this should be a float!
# assert isinstance(fee["cost"], float)
assert isinstance(fee["currency"], str)
else:
pytest.skip(f"No sample Trades available for exchange {exchange_name}")
2022-12-31 09:59:26 +00:00
def test_ccxt_fetch_tickers(self, exchange: EXCHANGE_FIXTURE_TYPE):
exch, exchangename = exchange
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename]["pair"]
2020-10-13 17:54:27 +00:00
2022-12-31 09:59:26 +00:00
tickers = exch.get_tickers()
2020-10-23 18:46:01 +00:00
assert pair in tickers
2024-05-12 13:38:09 +00:00
assert "ask" in tickers[pair]
assert tickers[pair]["ask"] is not None
assert "bid" in tickers[pair]
assert tickers[pair]["bid"] is not None
assert "quoteVolume" in tickers[pair]
if EXCHANGES[exchangename].get("hasQuoteVolume"):
assert tickers[pair]["quoteVolume"] is not None
2020-10-13 17:54:27 +00:00
2023-01-16 22:06:13 +00:00
def test_ccxt_fetch_tickers_futures(self, exchange_futures: EXCHANGE_FIXTURE_TYPE):
exch, exchangename = exchange_futures
2024-05-12 13:38:09 +00:00
if not exch or exchangename in ("gate"):
2023-01-16 22:06:13 +00:00
# exchange_futures only returns values for supported exchanges
return
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename]["pair"]
pair = EXCHANGES[exchangename].get("futures_pair", pair)
2023-01-16 22:06:13 +00:00
tickers = exch.get_tickers()
assert pair in tickers
2024-05-12 13:38:09 +00:00
assert "ask" in tickers[pair]
assert tickers[pair]["ask"] is not None
assert "bid" in tickers[pair]
assert tickers[pair]["bid"] is not None
assert "quoteVolume" in tickers[pair]
if EXCHANGES[exchangename].get("hasQuoteVolumeFutures"):
assert tickers[pair]["quoteVolume"] is not None
2023-01-16 22:06:13 +00:00
2022-12-31 09:59:26 +00:00
def test_ccxt_fetch_ticker(self, exchange: EXCHANGE_FIXTURE_TYPE):
exch, exchangename = exchange
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename]["pair"]
2020-10-13 17:54:27 +00:00
2022-12-31 09:59:26 +00:00
ticker = exch.fetch_ticker(pair)
2024-05-12 13:38:09 +00:00
assert "ask" in ticker
assert ticker["ask"] is not None
assert "bid" in ticker
assert ticker["bid"] is not None
assert "quoteVolume" in ticker
if EXCHANGES[exchangename].get("hasQuoteVolume"):
assert ticker["quoteVolume"] is not None
2020-10-13 17:54:27 +00:00
2022-12-31 09:59:26 +00:00
def test_ccxt_fetch_l2_orderbook(self, exchange: EXCHANGE_FIXTURE_TYPE):
exch, exchangename = exchange
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename]["pair"]
2022-12-31 09:59:26 +00:00
l2 = exch.fetch_l2_order_book(pair)
2024-05-12 13:38:09 +00:00
orderbook_max_entries = EXCHANGES[exchangename].get("orderbook_max_entries")
assert "asks" in l2
assert "bids" in l2
assert len(l2["asks"]) >= 1
assert len(l2["bids"]) >= 1
l2_limit_range = exch._ft_has["l2_limit_range"]
l2_limit_range_required = exch._ft_has["l2_limit_range_required"]
if exchangename == "gate":
2023-02-10 19:58:02 +00:00
# TODO: Gate is unstable here at the moment, ignoring the limit partially.
2022-06-20 05:05:51 +00:00
return
2023-03-04 18:49:59 +00:00
for val in [1, 2, 5, 25, 50, 100]:
if orderbook_max_entries and val > orderbook_max_entries:
2023-03-04 18:49:59 +00:00
continue
2022-12-31 09:59:26 +00:00
l2 = exch.fetch_l2_order_book(pair, val)
if not l2_limit_range or val in l2_limit_range:
2022-12-08 07:33:07 +00:00
if val > 50:
# Orderbooks are not always this deep.
2024-05-12 13:38:09 +00:00
assert val - 5 < len(l2["asks"]) <= val
assert val - 5 < len(l2["bids"]) <= val
2022-12-08 07:33:07 +00:00
else:
2024-05-12 13:38:09 +00:00
assert len(l2["asks"]) == val
assert len(l2["bids"]) == val
2020-10-23 18:46:01 +00:00
else:
2022-12-31 09:59:26 +00:00
next_limit = exch.get_next_limit_in_list(
2024-05-12 13:38:09 +00:00
val, l2_limit_range, l2_limit_range_required
)
2022-02-27 10:56:22 +00:00
if next_limit is None:
2024-05-12 13:38:09 +00:00
assert len(l2["asks"]) > 100
assert len(l2["asks"]) > 100
2022-02-27 10:56:22 +00:00
elif next_limit > 200:
# Large orderbook sizes can be a problem for some exchanges (bitrex ...)
2024-05-12 13:38:09 +00:00
assert len(l2["asks"]) > 200
assert len(l2["asks"]) > 200
2020-12-23 14:29:39 +00:00
else:
2024-05-12 13:38:09 +00:00
assert len(l2["asks"]) == next_limit
assert len(l2["asks"]) == next_limit
2020-10-23 18:49:46 +00:00
2022-12-31 09:59:26 +00:00
def test_ccxt_fetch_ohlcv(self, exchange: EXCHANGE_FIXTURE_TYPE):
exch, exchangename = exchange
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename]["pair"]
timeframe = EXCHANGES[exchangename]["timeframe"]
pair_tf = (pair, timeframe, CandleType.SPOT)
2022-12-31 09:59:26 +00:00
ohlcv = exch.refresh_latest_ohlcv([pair_tf])
2020-12-23 14:50:24 +00:00
assert isinstance(ohlcv, dict)
2022-12-31 09:59:26 +00:00
assert len(ohlcv[pair_tf]) == len(exch.klines(pair_tf))
# assert len(exch.klines(pair_tf)) > 200
2021-02-05 19:02:55 +00:00
# Assume 90% uptime ...
2024-05-12 13:38:09 +00:00
assert (
len(exch.klines(pair_tf)) > exch.ohlcv_candle_limit(timeframe, CandleType.SPOT) * 0.90
)
# Check if last-timeframe is within the last 2 intervals
now = datetime.now(timezone.utc) - timedelta(minutes=(timeframe_to_minutes(timeframe) * 2))
2024-05-12 13:38:09 +00:00
assert exch.klines(pair_tf).iloc[-1]["date"] >= timeframe_to_prev_date(timeframe, now)
2020-10-23 18:50:31 +00:00
def test_ccxt_fetch_ohlcv_startdate(self, exchange: EXCHANGE_FIXTURE_TYPE):
"""
Test that pair data starts at the provided startdate
"""
exch, exchangename = exchange
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename]["pair"]
timeframe = "1d"
pair_tf = (pair, timeframe, CandleType.SPOT)
# last 5 days ...
since_ms = dt_ts(dt_floor_day(dt_now()) - timedelta(days=6))
ohlcv = exch.refresh_latest_ohlcv([pair_tf], since_ms=since_ms)
assert isinstance(ohlcv, dict)
assert len(ohlcv[pair_tf]) == len(exch.klines(pair_tf))
# Check if last-timeframe is within the last 2 intervals
now = datetime.now(timezone.utc) - timedelta(minutes=(timeframe_to_minutes(timeframe) * 2))
2024-05-12 13:38:09 +00:00
assert exch.klines(pair_tf).iloc[-1]["date"] >= timeframe_to_prev_date(timeframe, now)
assert exch.klines(pair_tf)["date"].astype(int).iloc[0] // 1e6 == since_ms
def ccxt__async_get_candle_history(
2024-05-12 13:38:09 +00:00
self, exchange, exchangename, pair, timeframe, candle_type, factor=0.9
):
2022-05-14 07:51:44 +00:00
timeframe_ms = timeframe_to_msecs(timeframe)
2024-05-12 13:38:09 +00:00
now = timeframe_to_prev_date(timeframe, datetime.now(timezone.utc))
2022-05-14 07:51:44 +00:00
for offset in (360, 120, 30, 10, 5, 2):
since = now - timedelta(days=offset)
since_ms = int(since.timestamp() * 1000)
2024-05-12 13:38:09 +00:00
res = exchange.loop.run_until_complete(
exchange._async_get_candle_history(
pair=pair, timeframe=timeframe, since_ms=since_ms, candle_type=candle_type
)
2022-05-14 07:51:44 +00:00
)
assert res
assert res[0] == pair
assert res[1] == timeframe
assert res[2] == candle_type
candles = res[3]
2023-04-17 05:22:26 +00:00
candle_count = exchange.ohlcv_candle_limit(timeframe, candle_type, since_ms) * factor
candle_count1 = (now.timestamp() * 1000 - since_ms) // timeframe_ms * factor
2024-05-12 13:38:09 +00:00
assert len(candles) >= min(
candle_count, candle_count1
), f"{len(candles)} < {candle_count} in {timeframe}, Offset: {offset} {factor}"
# Check if first-timeframe is either the start, or start + 1
2022-05-14 07:51:44 +00:00
assert candles[0][0] == since_ms or (since_ms + timeframe_ms)
2022-12-31 09:59:26 +00:00
def test_ccxt__async_get_candle_history(self, exchange: EXCHANGE_FIXTURE_TYPE):
exc, exchangename = exchange
2023-02-07 18:45:37 +00:00
2024-05-12 13:38:09 +00:00
if not exc._ft_has["ohlcv_has_history"]:
2023-02-07 18:45:37 +00:00
pytest.skip("Exchange does not support candle history")
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename]["pair"]
timeframe = EXCHANGES[exchangename]["timeframe"]
self.ccxt__async_get_candle_history(exc, exchangename, pair, timeframe, CandleType.SPOT)
@pytest.mark.parametrize(
"candle_type",
[
CandleType.FUTURES,
CandleType.FUNDING_RATE,
CandleType.MARK,
],
)
def test_ccxt__async_get_candle_history_futures(
2024-05-12 13:38:09 +00:00
self, exchange_futures: EXCHANGE_FIXTURE_TYPE, candle_type
):
exchange, exchangename = exchange_futures
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename].get("futures_pair", EXCHANGES[exchangename]["pair"])
timeframe = EXCHANGES[exchangename]["timeframe"]
if candle_type == CandleType.FUNDING_RATE:
2024-05-12 13:38:09 +00:00
timeframe = exchange._ft_has.get(
"funding_fee_timeframe", exchange._ft_has["mark_ohlcv_timeframe"]
)
2022-10-09 08:52:01 +00:00
self.ccxt__async_get_candle_history(
exchange,
exchangename,
pair=pair,
timeframe=timeframe,
candle_type=candle_type,
)
2022-12-31 09:59:26 +00:00
def test_ccxt_fetch_funding_rate_history(self, exchange_futures: EXCHANGE_FIXTURE_TYPE):
2021-11-12 06:26:59 +00:00
exchange, exchangename = exchange_futures
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename].get("futures_pair", EXCHANGES[exchangename]["pair"])
2021-11-12 06:26:59 +00:00
since = int((datetime.now(timezone.utc) - timedelta(days=5)).timestamp() * 1000)
2024-05-12 13:38:09 +00:00
timeframe_ff = exchange._ft_has.get(
"funding_fee_timeframe", exchange._ft_has["mark_ohlcv_timeframe"]
)
2021-12-19 13:48:59 +00:00
pair_tf = (pair, timeframe_ff, CandleType.FUNDING_RATE)
2021-11-12 06:26:59 +00:00
funding_ohlcv = exchange.refresh_latest_ohlcv(
2024-05-12 13:38:09 +00:00
[pair_tf], since_ms=since, drop_incomplete=False
)
assert isinstance(funding_ohlcv, dict)
rate = funding_ohlcv[pair_tf]
2021-12-19 13:48:59 +00:00
this_hour = timeframe_to_prev_date(timeframe_ff)
hour1 = timeframe_to_prev_date(timeframe_ff, this_hour - timedelta(minutes=1))
hour2 = timeframe_to_prev_date(timeframe_ff, hour1 - timedelta(minutes=1))
hour3 = timeframe_to_prev_date(timeframe_ff, hour2 - timedelta(minutes=1))
2024-05-12 13:38:09 +00:00
val0 = rate[rate["date"] == this_hour].iloc[0]["open"]
val1 = rate[rate["date"] == hour1].iloc[0]["open"]
val2 = rate[rate["date"] == hour2].iloc[0]["open"]
val3 = rate[rate["date"] == hour3].iloc[0]["open"]
# Test For last 4 hours
# Avoids random test-failure when funding-fees are 0 for a few hours.
assert val0 != 0.0 or val1 != 0.0 or val2 != 0.0 or val3 != 0.0
2022-01-05 19:37:15 +00:00
# We expect funding rates to be different from 0.0 - or moving around.
assert (
2024-05-12 13:38:09 +00:00
rate["open"].max() != 0.0
or rate["open"].min() != 0.0
or (rate["open"].min() != rate["open"].max())
2022-01-05 19:37:15 +00:00
)
2022-12-31 09:59:26 +00:00
def test_ccxt_fetch_mark_price_history(self, exchange_futures: EXCHANGE_FIXTURE_TYPE):
exchange, exchangename = exchange_futures
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename].get("futures_pair", EXCHANGES[exchangename]["pair"])
since = int((datetime.now(timezone.utc) - timedelta(days=5)).timestamp() * 1000)
2024-05-12 13:38:09 +00:00
pair_tf = (pair, "1h", CandleType.MARK)
2024-05-12 13:38:09 +00:00
mark_ohlcv = exchange.refresh_latest_ohlcv([pair_tf], since_ms=since, drop_incomplete=False)
2021-12-10 05:46:35 +00:00
assert isinstance(mark_ohlcv, dict)
2024-05-12 13:38:09 +00:00
expected_tf = "1h"
2021-12-10 05:46:35 +00:00
mark_candles = mark_ohlcv[pair_tf]
this_hour = timeframe_to_prev_date(expected_tf)
2021-12-10 05:46:35 +00:00
prev_hour = timeframe_to_prev_date(expected_tf, this_hour - timedelta(minutes=1))
2024-05-12 13:38:09 +00:00
assert mark_candles[mark_candles["date"] == prev_hour].iloc[0]["open"] != 0.0
assert mark_candles[mark_candles["date"] == this_hour].iloc[0]["open"] != 0.0
2022-12-31 09:59:26 +00:00
def test_ccxt__calculate_funding_fees(self, exchange_futures: EXCHANGE_FIXTURE_TYPE):
exchange, exchangename = exchange_futures
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename].get("futures_pair", EXCHANGES[exchangename]["pair"])
since = datetime.now(timezone.utc) - timedelta(days=5)
2022-01-18 06:40:09 +00:00
funding_fee = exchange._fetch_and_calculate_funding_fees(
2024-05-12 13:38:09 +00:00
pair, 20, is_short=False, open_date=since
)
assert isinstance(funding_fee, float)
# assert funding_fee > 0
def test_ccxt__async_get_trade_history(self, exchange: EXCHANGE_FIXTURE_TYPE):
exch, exchangename = exchange
2024-05-12 13:38:09 +00:00
if not (lookback := EXCHANGES[exchangename].get("trades_lookback_hours")):
pytest.skip("test_fetch_trades not enabled for this exchange")
pair = EXCHANGES[exchangename]["pair"]
since = int((datetime.now(timezone.utc) - timedelta(hours=lookback)).timestamp() * 1000)
2024-05-12 13:38:09 +00:00
res = exch.loop.run_until_complete(exch._async_get_trade_history(pair, since, None, None))
assert len(res) == 2
res_pair, res_trades = res
assert res_pair == pair
assert isinstance(res_trades, list)
assert res_trades[0][0] >= since
assert len(res_trades) > 1200
2022-12-31 09:59:26 +00:00
def test_ccxt_get_fee(self, exchange: EXCHANGE_FIXTURE_TYPE):
exch, exchangename = exchange
2024-05-12 13:38:09 +00:00
pair = EXCHANGES[exchangename]["pair"]
threshold = 0.01
2024-05-12 13:38:09 +00:00
assert 0 < exch.get_fee(pair, "limit", "buy") < threshold
assert 0 < exch.get_fee(pair, "limit", "sell") < threshold
assert 0 < exch.get_fee(pair, "market", "buy") < threshold
assert 0 < exch.get_fee(pair, "market", "sell") < threshold
2022-12-31 09:59:26 +00:00
def test_ccxt_get_max_leverage_spot(self, exchange: EXCHANGE_FIXTURE_TYPE):
spot, spot_name = exchange
if spot:
2024-05-12 13:38:09 +00:00
leverage_in_market_spot = EXCHANGES[spot_name].get("leverage_in_spot_market")
if leverage_in_market_spot:
2024-05-12 13:38:09 +00:00
spot_pair = EXCHANGES[spot_name].get("pair", EXCHANGES[spot_name]["pair"])
spot_leverage = spot.get_max_leverage(spot_pair, 20)
2024-05-12 13:38:09 +00:00
assert isinstance(spot_leverage, float) or isinstance(spot_leverage, int)
assert spot_leverage >= 1.0
2022-12-31 09:59:26 +00:00
def test_ccxt_get_max_leverage_futures(self, exchange_futures: EXCHANGE_FIXTURE_TYPE):
futures, futures_name = exchange_futures
2024-05-12 13:38:09 +00:00
leverage_tiers_public = EXCHANGES[futures_name].get("leverage_tiers_public")
if leverage_tiers_public:
2022-02-10 12:59:43 +00:00
futures_pair = EXCHANGES[futures_name].get(
2024-05-12 13:38:09 +00:00
"futures_pair", EXCHANGES[futures_name]["pair"]
2022-02-10 12:59:43 +00:00
)
futures_leverage = futures.get_max_leverage(futures_pair, 20)
2024-05-12 13:38:09 +00:00
assert isinstance(futures_leverage, float) or isinstance(futures_leverage, int)
assert futures_leverage >= 1.0
def test_ccxt_get_contract_size(self, exchange_futures: EXCHANGE_FIXTURE_TYPE):
futures, futures_name = exchange_futures
2024-05-12 13:38:09 +00:00
futures_pair = EXCHANGES[futures_name].get("futures_pair", EXCHANGES[futures_name]["pair"])
contract_size = futures.get_contract_size(futures_pair)
2024-05-12 13:38:09 +00:00
assert isinstance(contract_size, float) or isinstance(contract_size, int)
assert contract_size >= 0.0
2022-02-07 11:03:10 +00:00
2022-12-31 09:59:26 +00:00
def test_ccxt_load_leverage_tiers(self, exchange_futures: EXCHANGE_FIXTURE_TYPE):
2022-02-16 14:08:10 +00:00
futures, futures_name = exchange_futures
2024-05-12 13:38:09 +00:00
if EXCHANGES[futures_name].get("leverage_tiers_public"):
2022-02-16 14:08:10 +00:00
leverage_tiers = futures.load_leverage_tiers()
futures_pair = EXCHANGES[futures_name].get(
2024-05-12 13:38:09 +00:00
"futures_pair", EXCHANGES[futures_name]["pair"]
2022-02-16 14:08:10 +00:00
)
2024-05-12 13:38:09 +00:00
assert isinstance(leverage_tiers, dict)
2022-02-16 14:08:10 +00:00
assert futures_pair in leverage_tiers
pair_tiers = leverage_tiers[futures_pair]
assert len(pair_tiers) > 0
2024-05-12 13:38:09 +00:00
oldLeverage = float("inf")
oldMaintenanceMarginRate = oldminNotional = oldmaxNotional = -1
2022-02-16 14:08:10 +00:00
for tier in pair_tiers:
2024-05-12 13:38:09 +00:00
for key in ["maintenanceMarginRate", "minNotional", "maxNotional", "maxLeverage"]:
2022-02-16 14:08:10 +00:00
assert key in tier
2022-02-16 14:48:53 +00:00
assert tier[key] >= 0.0
2024-05-12 13:38:09 +00:00
assert tier["maxNotional"] > tier["minNotional"]
assert tier["maxLeverage"] <= oldLeverage
assert tier["maintenanceMarginRate"] >= oldMaintenanceMarginRate
assert tier["minNotional"] > oldminNotional
assert tier["maxNotional"] > oldmaxNotional
oldLeverage = tier["maxLeverage"]
oldMaintenanceMarginRate = tier["maintenanceMarginRate"]
oldminNotional = tier["minNotional"]
oldmaxNotional = tier["maxNotional"]
2022-02-16 14:08:10 +00:00
2022-12-31 09:59:26 +00:00
def test_ccxt_dry_run_liquidation_price(self, exchange_futures: EXCHANGE_FIXTURE_TYPE):
2022-02-16 15:20:22 +00:00
futures, futures_name = exchange_futures
2024-05-12 13:38:09 +00:00
if EXCHANGES[futures_name].get("leverage_tiers_public"):
2022-02-16 15:20:22 +00:00
futures_pair = EXCHANGES[futures_name].get(
2024-05-12 13:38:09 +00:00
"futures_pair", EXCHANGES[futures_name]["pair"]
2022-02-16 15:20:22 +00:00
)
liquidation_price = futures.dry_run_liquidation_price(
pair=futures_pair,
open_rate=40000,
is_short=False,
amount=100,
stake_amount=100,
leverage=5,
wallet_balance=100,
2022-02-16 15:20:22 +00:00
)
2024-05-12 13:38:09 +00:00
assert isinstance(liquidation_price, float)
2022-02-16 15:20:22 +00:00
assert liquidation_price >= 0.0
liquidation_price = futures.dry_run_liquidation_price(
pair=futures_pair,
open_rate=40000,
is_short=False,
amount=100,
stake_amount=100,
leverage=5,
wallet_balance=100,
2022-02-16 15:20:22 +00:00
)
2024-05-12 13:38:09 +00:00
assert isinstance(liquidation_price, float)
2022-02-16 15:20:22 +00:00
assert liquidation_price >= 0.0
2022-02-07 11:03:10 +00:00
2022-12-31 09:59:26 +00:00
def test_ccxt_get_max_pair_stake_amount(self, exchange_futures: EXCHANGE_FIXTURE_TYPE):
futures, futures_name = exchange_futures
2024-05-12 13:38:09 +00:00
futures_pair = EXCHANGES[futures_name].get("futures_pair", EXCHANGES[futures_name]["pair"])
max_stake_amount = futures.get_max_pair_stake_amount(futures_pair, 40000)
2024-05-12 13:38:09 +00:00
assert isinstance(max_stake_amount, float)
assert max_stake_amount >= 0.0
def test_private_method_presence(self, exchange: EXCHANGE_FIXTURE_TYPE):
exch, exchangename = exchange
2024-05-12 13:38:09 +00:00
for method in EXCHANGES[exchangename].get("private_methods", []):
assert hasattr(exch._api, method)