2020-06-28 09:17:06 +00:00
|
|
|
import asyncio
|
2019-10-31 09:39:24 +00:00
|
|
|
import logging
|
2020-06-28 09:17:06 +00:00
|
|
|
import time
|
2020-06-28 09:56:29 +00:00
|
|
|
from functools import wraps
|
2019-10-31 09:39:24 +00:00
|
|
|
|
2020-09-28 17:39:41 +00:00
|
|
|
from freqtrade.exceptions import DDosProtection, RetryableOrderError, TemporaryError
|
|
|
|
|
2019-10-31 09:39:24 +00:00
|
|
|
|
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
|
2020-08-22 15:35:42 +00:00
|
|
|
# Maximum default retry count.
|
|
|
|
# Functions are always called RETRY_COUNT + 1 times (for the original call)
|
2019-10-31 09:39:24 +00:00
|
|
|
API_RETRY_COUNT = 4
|
2020-09-19 06:42:37 +00:00
|
|
|
API_FETCH_ORDER_RETRY_COUNT = 5
|
2020-08-22 15:35:42 +00:00
|
|
|
|
2019-10-31 09:39:24 +00:00
|
|
|
BAD_EXCHANGES = {
|
|
|
|
"bitmex": "Various reasons.",
|
2020-08-23 07:11:34 +00:00
|
|
|
"phemex": "Does not provide history. ",
|
2021-01-20 18:30:43 +00:00
|
|
|
"poloniex": "Does not provide fetch_order endpoint to fetch both open and closed orders.",
|
2019-10-31 09:39:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MAP_EXCHANGE_CHILDCLASS = {
|
|
|
|
'binanceus': 'binance',
|
|
|
|
'binanceje': 'binance',
|
2021-11-27 15:46:17 +00:00
|
|
|
'binanceusdm': 'binance',
|
2019-10-31 09:39:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-04-06 05:47:44 +00:00
|
|
|
EXCHANGE_HAS_REQUIRED = [
|
|
|
|
# Required / private
|
|
|
|
'fetchOrder',
|
|
|
|
'cancelOrder',
|
|
|
|
'createOrder',
|
|
|
|
# 'createLimitOrder', 'createMarketOrder',
|
|
|
|
'fetchBalance',
|
|
|
|
|
|
|
|
# Public endpoints
|
|
|
|
'loadMarkets',
|
|
|
|
'fetchOHLCV',
|
|
|
|
]
|
|
|
|
|
|
|
|
EXCHANGE_HAS_OPTIONAL = [
|
|
|
|
# Private
|
|
|
|
'fetchMyTrades', # Trades for order - fee detection
|
|
|
|
# Public
|
|
|
|
'fetchOrderBook', 'fetchL2OrderBook', 'fetchTicker', # OR for pricing
|
|
|
|
'fetchTickers', # For volumepairlist?
|
|
|
|
'fetchTrades', # Downloading trades data
|
|
|
|
]
|
|
|
|
|
|
|
|
|
2021-09-13 18:00:22 +00:00
|
|
|
def remove_credentials(config) -> None:
|
|
|
|
"""
|
|
|
|
Removes exchange keys from the configuration and specifies dry-run
|
|
|
|
Used for backtesting / hyperopt / edge and utils.
|
|
|
|
Modifies the input dict!
|
|
|
|
"""
|
|
|
|
if config.get('dry_run', False):
|
|
|
|
config['exchange']['key'] = ''
|
|
|
|
config['exchange']['secret'] = ''
|
|
|
|
config['exchange']['password'] = ''
|
|
|
|
config['exchange']['uid'] = ''
|
|
|
|
|
|
|
|
|
2020-06-28 17:40:33 +00:00
|
|
|
def calculate_backoff(retrycount, max_retries):
|
2020-06-28 14:18:39 +00:00
|
|
|
"""
|
|
|
|
Calculate backoff
|
|
|
|
"""
|
2020-06-28 17:40:33 +00:00
|
|
|
return (max_retries - retrycount) ** 2 + 1
|
2020-06-28 14:18:39 +00:00
|
|
|
|
|
|
|
|
2019-10-31 09:39:24 +00:00
|
|
|
def retrier_async(f):
|
|
|
|
async def wrapper(*args, **kwargs):
|
|
|
|
count = kwargs.pop('count', API_RETRY_COUNT)
|
|
|
|
try:
|
|
|
|
return await f(*args, **kwargs)
|
2020-05-18 14:31:34 +00:00
|
|
|
except TemporaryError as ex:
|
2019-10-31 09:39:24 +00:00
|
|
|
logger.warning('%s() returned exception: "%s"', f.__name__, ex)
|
|
|
|
if count > 0:
|
2020-08-11 13:27:41 +00:00
|
|
|
logger.warning('retrying %s() still for %s times', f.__name__, count)
|
2019-10-31 09:39:24 +00:00
|
|
|
count -= 1
|
|
|
|
kwargs.update({'count': count})
|
2020-06-28 18:17:03 +00:00
|
|
|
if isinstance(ex, DDosProtection):
|
2021-06-14 05:39:12 +00:00
|
|
|
if "kucoin" in str(ex) and "429000" in str(ex):
|
|
|
|
# Temporary fix for 429000 error on kucoin
|
|
|
|
# see https://github.com/freqtrade/freqtrade/issues/5700 for details.
|
|
|
|
logger.warning(
|
|
|
|
f"Kucoin 429 error, avoid triggering DDosProtection backoff delay. "
|
|
|
|
f"{count} tries left before giving up")
|
|
|
|
else:
|
|
|
|
backoff_delay = calculate_backoff(count + 1, API_RETRY_COUNT)
|
|
|
|
logger.info(f"Applying DDosProtection backoff delay: {backoff_delay}")
|
|
|
|
await asyncio.sleep(backoff_delay)
|
2019-10-31 09:39:24 +00:00
|
|
|
return await wrapper(*args, **kwargs)
|
|
|
|
else:
|
|
|
|
logger.warning('Giving up retrying: %s()', f.__name__)
|
|
|
|
raise ex
|
|
|
|
return wrapper
|
|
|
|
|
|
|
|
|
2020-06-28 09:56:29 +00:00
|
|
|
def retrier(_func=None, retries=API_RETRY_COUNT):
|
|
|
|
def decorator(f):
|
|
|
|
@wraps(f)
|
|
|
|
def wrapper(*args, **kwargs):
|
|
|
|
count = kwargs.pop('count', retries)
|
|
|
|
try:
|
|
|
|
return f(*args, **kwargs)
|
2020-06-28 17:45:42 +00:00
|
|
|
except (TemporaryError, RetryableOrderError) as ex:
|
2020-06-28 09:56:29 +00:00
|
|
|
logger.warning('%s() returned exception: "%s"', f.__name__, ex)
|
|
|
|
if count > 0:
|
2020-08-11 13:27:41 +00:00
|
|
|
logger.warning('retrying %s() still for %s times', f.__name__, count)
|
2020-06-28 09:56:29 +00:00
|
|
|
count -= 1
|
|
|
|
kwargs.update({'count': count})
|
2021-03-21 11:44:34 +00:00
|
|
|
if isinstance(ex, (DDosProtection, RetryableOrderError)):
|
2020-06-28 17:45:42 +00:00
|
|
|
# increasing backoff
|
2020-06-29 18:00:42 +00:00
|
|
|
backoff_delay = calculate_backoff(count + 1, retries)
|
2020-08-11 17:27:25 +00:00
|
|
|
logger.info(f"Applying DDosProtection backoff delay: {backoff_delay}")
|
2020-06-29 18:00:42 +00:00
|
|
|
time.sleep(backoff_delay)
|
2020-06-28 09:56:29 +00:00
|
|
|
return wrapper(*args, **kwargs)
|
|
|
|
else:
|
|
|
|
logger.warning('Giving up retrying: %s()', f.__name__)
|
|
|
|
raise ex
|
|
|
|
return wrapper
|
2020-06-28 14:18:39 +00:00
|
|
|
# Support both @retrier and @retrier(retries=2) syntax
|
2020-06-28 09:56:29 +00:00
|
|
|
if _func is None:
|
|
|
|
return decorator
|
|
|
|
else:
|
|
|
|
return decorator(_func)
|