Source code for CryptoPrice.retrievers.BinanceRetriever

import datetime
import traceback
from typing import List

from binance.client import Client
from binance.exceptions import BinanceAPIException

from CryptoPrice.common.trade import TradingPair
from CryptoPrice.exceptions import RateAPIException
from CryptoPrice.retrievers.KlineRetriever import KlineRetriever
from CryptoPrice.common.prices import Kline
from CryptoPrice.utils.time import TIMEFRAME


[docs]class BinanceRetriever(KlineRetriever): """ This class is in charge of fetching klines from the Binance API docs: https://python-binance.readthedocs.io/en/latest/binance.html """ kline_translation = { TIMEFRAME.m1: Client.KLINE_INTERVAL_1MINUTE, TIMEFRAME.m3: Client.KLINE_INTERVAL_3MINUTE, TIMEFRAME.m5: Client.KLINE_INTERVAL_5MINUTE, TIMEFRAME.m15: Client.KLINE_INTERVAL_15MINUTE, TIMEFRAME.m30: Client.KLINE_INTERVAL_30MINUTE, TIMEFRAME.h1: Client.KLINE_INTERVAL_1HOUR, TIMEFRAME.h2: Client.KLINE_INTERVAL_2HOUR, TIMEFRAME.h4: Client.KLINE_INTERVAL_4HOUR, TIMEFRAME.h6: Client.KLINE_INTERVAL_6HOUR, TIMEFRAME.h8: Client.KLINE_INTERVAL_8HOUR, TIMEFRAME.h12: Client.KLINE_INTERVAL_12HOUR, TIMEFRAME.d1: Client.KLINE_INTERVAL_1DAY, TIMEFRAME.d3: Client.KLINE_INTERVAL_3DAY, TIMEFRAME.w1: Client.KLINE_INTERVAL_1WEEK }
[docs] def __init__(self, kline_timeframe: TIMEFRAME = TIMEFRAME.m1, closest_window: int = 310): self.client = Client() super(BinanceRetriever, self).__init__('binance', kline_timeframe, closest_window)
[docs] def get_supported_pairs(self) -> List[TradingPair]: """ Return the list of trading pair supported by this retriever :return: list of trading pairs :rtype: List[TradingPair] """ binance_symbols = self.client.get_exchange_info()['symbols'] return [TradingPair(s['symbol'], s['baseAsset'], s['quoteAsset'], source=self.name) for s in binance_symbols]
def _get_klines_online(self, asset: str, ref_asset: str, timeframe: TIMEFRAME, start_time: int, end_time: int) -> List[Kline]: """ Fetch klines online by asking the Binance API :param asset: asset of the trading pair :type asset: str :param ref_asset: reference asset of the trading pair :type ref_asset: str :param timeframe: timeframe for the kline :type timeframe: TIMEFRAME :param start_time: fetch only klines with an open time greater or equal than start_time :type start_time: Optional[int] :param end_time: fetch only klines with an open time lower than end_time :type end_time: Optional[int] :return: list of klines :rtype: List[Kline] """ pair_name = asset + ref_asset batch_size = 1000 interval_trad = self.kline_translation[timeframe] try: result = self.client.get_klines(symbol=pair_name, interval=interval_trad, startTime=start_time * 1000, endTime=end_time * 1000, limit=batch_size) except BinanceAPIException as err: if err.code == -1121: self.logger.info(f"The trading pair {asset} {ref_asset} is not supported") return [] elif err.code == -1003: retry_after = 1 + 60 - datetime.datetime.now().timestamp() % 60 raise RateAPIException(retry_after, err.response) self.logger.error(str(traceback.format_exc())) raise err klines = [] for row in result: open_timestamp = int(row[0] / 1000) open = float(row[1]) high = float(row[2]) low = float(row[3]) close = float(row[4]) klines.append(Kline(open_timestamp, open, high, low, close, asset, ref_asset, timeframe, source=self.name)) return klines