2020-05-11 15:32:28 +00:00
|
|
|
from unittest.mock import MagicMock, patch
|
2018-12-25 12:56:30 +00:00
|
|
|
|
|
|
|
from pandas import DataFrame
|
2020-05-11 15:32:28 +00:00
|
|
|
import pytest
|
2018-12-25 12:56:30 +00:00
|
|
|
|
|
|
|
from freqtrade.data.dataprovider import DataProvider
|
2020-05-11 15:32:28 +00:00
|
|
|
from freqtrade.exceptions import OperationalException
|
2018-12-29 08:13:20 +00:00
|
|
|
from freqtrade.state import RunMode
|
2020-05-11 15:32:28 +00:00
|
|
|
from tests.conftest import get_patched_exchange
|
2018-12-25 12:56:30 +00:00
|
|
|
|
|
|
|
|
2020-03-08 10:35:31 +00:00
|
|
|
def test_ohlcv(mocker, default_conf, ohlcv_history):
|
2019-01-22 05:56:54 +00:00
|
|
|
default_conf["runmode"] = RunMode.DRY_RUN
|
2019-11-02 19:26:26 +00:00
|
|
|
timeframe = default_conf["ticker_interval"]
|
2018-12-25 12:56:30 +00:00
|
|
|
exchange = get_patched_exchange(mocker, default_conf)
|
2020-03-08 10:35:31 +00:00
|
|
|
exchange._klines[("XRP/BTC", timeframe)] = ohlcv_history
|
|
|
|
exchange._klines[("UNITTEST/BTC", timeframe)] = ohlcv_history
|
2019-08-18 10:06:21 +00:00
|
|
|
|
2018-12-25 12:56:30 +00:00
|
|
|
dp = DataProvider(default_conf, exchange)
|
2018-12-29 08:13:20 +00:00
|
|
|
assert dp.runmode == RunMode.DRY_RUN
|
2020-03-08 10:35:31 +00:00
|
|
|
assert ohlcv_history.equals(dp.ohlcv("UNITTEST/BTC", timeframe))
|
2019-11-02 19:26:26 +00:00
|
|
|
assert isinstance(dp.ohlcv("UNITTEST/BTC", timeframe), DataFrame)
|
2020-03-08 10:35:31 +00:00
|
|
|
assert dp.ohlcv("UNITTEST/BTC", timeframe) is not ohlcv_history
|
|
|
|
assert dp.ohlcv("UNITTEST/BTC", timeframe, copy=False) is ohlcv_history
|
2019-11-02 19:26:26 +00:00
|
|
|
assert not dp.ohlcv("UNITTEST/BTC", timeframe).empty
|
|
|
|
assert dp.ohlcv("NONESENSE/AAA", timeframe).empty
|
2018-12-30 06:15:49 +00:00
|
|
|
|
|
|
|
# Test with and without parameter
|
2019-11-02 19:26:26 +00:00
|
|
|
assert dp.ohlcv("UNITTEST/BTC", timeframe).equals(dp.ohlcv("UNITTEST/BTC"))
|
2018-12-29 08:13:20 +00:00
|
|
|
|
2019-01-22 05:56:54 +00:00
|
|
|
default_conf["runmode"] = RunMode.LIVE
|
2018-12-29 08:13:20 +00:00
|
|
|
dp = DataProvider(default_conf, exchange)
|
|
|
|
assert dp.runmode == RunMode.LIVE
|
2019-11-02 19:26:26 +00:00
|
|
|
assert isinstance(dp.ohlcv("UNITTEST/BTC", timeframe), DataFrame)
|
2018-12-29 08:13:20 +00:00
|
|
|
|
2019-01-22 05:56:54 +00:00
|
|
|
default_conf["runmode"] = RunMode.BACKTEST
|
2018-12-29 08:13:20 +00:00
|
|
|
dp = DataProvider(default_conf, exchange)
|
|
|
|
assert dp.runmode == RunMode.BACKTEST
|
2019-11-02 19:26:26 +00:00
|
|
|
assert dp.ohlcv("UNITTEST/BTC", timeframe).empty
|
2018-12-25 12:56:30 +00:00
|
|
|
|
|
|
|
|
2020-03-08 10:35:31 +00:00
|
|
|
def test_historic_ohlcv(mocker, default_conf, ohlcv_history):
|
|
|
|
historymock = MagicMock(return_value=ohlcv_history)
|
2019-01-22 05:56:54 +00:00
|
|
|
mocker.patch("freqtrade.data.dataprovider.load_pair_history", historymock)
|
2018-12-25 12:56:30 +00:00
|
|
|
|
|
|
|
dp = DataProvider(default_conf, None)
|
2019-01-22 05:56:54 +00:00
|
|
|
data = dp.historic_ohlcv("UNITTEST/BTC", "5m")
|
2018-12-25 12:56:30 +00:00
|
|
|
assert isinstance(data, DataFrame)
|
|
|
|
assert historymock.call_count == 1
|
2019-11-02 19:19:13 +00:00
|
|
|
assert historymock.call_args_list[0][1]["timeframe"] == "5m"
|
2018-12-26 13:23:21 +00:00
|
|
|
|
|
|
|
|
2020-03-08 10:35:31 +00:00
|
|
|
def test_get_pair_dataframe(mocker, default_conf, ohlcv_history):
|
2019-08-17 09:59:27 +00:00
|
|
|
default_conf["runmode"] = RunMode.DRY_RUN
|
|
|
|
ticker_interval = default_conf["ticker_interval"]
|
|
|
|
exchange = get_patched_exchange(mocker, default_conf)
|
2020-03-08 10:35:31 +00:00
|
|
|
exchange._klines[("XRP/BTC", ticker_interval)] = ohlcv_history
|
|
|
|
exchange._klines[("UNITTEST/BTC", ticker_interval)] = ohlcv_history
|
2019-08-18 10:06:21 +00:00
|
|
|
|
2019-08-17 09:59:27 +00:00
|
|
|
dp = DataProvider(default_conf, exchange)
|
|
|
|
assert dp.runmode == RunMode.DRY_RUN
|
2020-03-08 10:35:31 +00:00
|
|
|
assert ohlcv_history.equals(dp.get_pair_dataframe("UNITTEST/BTC", ticker_interval))
|
2019-08-17 09:59:27 +00:00
|
|
|
assert isinstance(dp.get_pair_dataframe("UNITTEST/BTC", ticker_interval), DataFrame)
|
2020-03-08 10:35:31 +00:00
|
|
|
assert dp.get_pair_dataframe("UNITTEST/BTC", ticker_interval) is not ohlcv_history
|
2019-08-17 09:59:27 +00:00
|
|
|
assert not dp.get_pair_dataframe("UNITTEST/BTC", ticker_interval).empty
|
|
|
|
assert dp.get_pair_dataframe("NONESENSE/AAA", ticker_interval).empty
|
|
|
|
|
|
|
|
# Test with and without parameter
|
2020-05-06 15:48:57 +00:00
|
|
|
assert dp.get_pair_dataframe("UNITTEST/BTC", ticker_interval)\
|
|
|
|
.equals(dp.get_pair_dataframe("UNITTEST/BTC"))
|
2019-08-17 09:59:27 +00:00
|
|
|
|
|
|
|
default_conf["runmode"] = RunMode.LIVE
|
|
|
|
dp = DataProvider(default_conf, exchange)
|
|
|
|
assert dp.runmode == RunMode.LIVE
|
|
|
|
assert isinstance(dp.get_pair_dataframe("UNITTEST/BTC", ticker_interval), DataFrame)
|
|
|
|
assert dp.get_pair_dataframe("NONESENSE/AAA", ticker_interval).empty
|
|
|
|
|
2020-03-08 10:35:31 +00:00
|
|
|
historymock = MagicMock(return_value=ohlcv_history)
|
2019-08-17 09:59:27 +00:00
|
|
|
mocker.patch("freqtrade.data.dataprovider.load_pair_history", historymock)
|
|
|
|
default_conf["runmode"] = RunMode.BACKTEST
|
|
|
|
dp = DataProvider(default_conf, exchange)
|
|
|
|
assert dp.runmode == RunMode.BACKTEST
|
|
|
|
assert isinstance(dp.get_pair_dataframe("UNITTEST/BTC", ticker_interval), DataFrame)
|
|
|
|
# assert dp.get_pair_dataframe("NONESENSE/AAA", ticker_interval).empty
|
|
|
|
|
|
|
|
|
2020-03-08 10:35:31 +00:00
|
|
|
def test_available_pairs(mocker, default_conf, ohlcv_history):
|
2018-12-26 13:23:21 +00:00
|
|
|
exchange = get_patched_exchange(mocker, default_conf)
|
2019-04-07 13:14:40 +00:00
|
|
|
ticker_interval = default_conf["ticker_interval"]
|
2020-03-08 10:35:31 +00:00
|
|
|
exchange._klines[("XRP/BTC", ticker_interval)] = ohlcv_history
|
|
|
|
exchange._klines[("UNITTEST/BTC", ticker_interval)] = ohlcv_history
|
2018-12-26 13:23:21 +00:00
|
|
|
|
2019-08-18 10:06:21 +00:00
|
|
|
dp = DataProvider(default_conf, exchange)
|
2018-12-26 13:23:21 +00:00
|
|
|
assert len(dp.available_pairs) == 2
|
2020-05-06 15:48:57 +00:00
|
|
|
assert dp.available_pairs == [("XRP/BTC", ticker_interval), ("UNITTEST/BTC", ticker_interval), ]
|
2019-01-22 05:55:40 +00:00
|
|
|
|
|
|
|
|
2020-03-08 10:35:31 +00:00
|
|
|
def test_refresh(mocker, default_conf, ohlcv_history):
|
2019-01-22 05:55:40 +00:00
|
|
|
refresh_mock = MagicMock()
|
2019-01-22 05:56:54 +00:00
|
|
|
mocker.patch("freqtrade.exchange.Exchange.refresh_latest_ohlcv", refresh_mock)
|
2019-01-22 05:55:40 +00:00
|
|
|
|
2019-01-22 05:56:54 +00:00
|
|
|
exchange = get_patched_exchange(mocker, default_conf, id="binance")
|
2019-04-07 13:14:40 +00:00
|
|
|
ticker_interval = default_conf["ticker_interval"]
|
|
|
|
pairs = [("XRP/BTC", ticker_interval), ("UNITTEST/BTC", ticker_interval)]
|
2019-01-22 05:55:40 +00:00
|
|
|
|
2019-04-07 13:14:40 +00:00
|
|
|
pairs_non_trad = [("ETH/USDT", ticker_interval), ("BTC/TUSD", "1h")]
|
2019-01-22 05:55:40 +00:00
|
|
|
|
|
|
|
dp = DataProvider(default_conf, exchange)
|
|
|
|
dp.refresh(pairs)
|
|
|
|
|
|
|
|
assert refresh_mock.call_count == 1
|
|
|
|
assert len(refresh_mock.call_args[0]) == 1
|
|
|
|
assert len(refresh_mock.call_args[0][0]) == len(pairs)
|
|
|
|
assert refresh_mock.call_args[0][0] == pairs
|
|
|
|
|
|
|
|
refresh_mock.reset_mock()
|
|
|
|
dp.refresh(pairs, pairs_non_trad)
|
|
|
|
assert refresh_mock.call_count == 1
|
|
|
|
assert len(refresh_mock.call_args[0]) == 1
|
|
|
|
assert len(refresh_mock.call_args[0][0]) == len(pairs) + len(pairs_non_trad)
|
|
|
|
assert refresh_mock.call_args[0][0] == pairs + pairs_non_trad
|
2019-10-10 15:03:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
def test_orderbook(mocker, default_conf, order_book_l2):
|
|
|
|
api_mock = MagicMock()
|
|
|
|
api_mock.fetch_l2_order_book = order_book_l2
|
|
|
|
exchange = get_patched_exchange(mocker, default_conf, api_mock=api_mock)
|
|
|
|
|
|
|
|
dp = DataProvider(default_conf, exchange)
|
|
|
|
res = dp.orderbook('ETH/BTC', 5)
|
|
|
|
assert order_book_l2.call_count == 1
|
|
|
|
assert order_book_l2.call_args_list[0][0][0] == 'ETH/BTC'
|
|
|
|
assert order_book_l2.call_args_list[0][0][1] == 5
|
|
|
|
|
|
|
|
assert type(res) is dict
|
|
|
|
assert 'bids' in res
|
|
|
|
assert 'asks' in res
|
|
|
|
|
|
|
|
|
|
|
|
def test_market(mocker, default_conf, markets):
|
|
|
|
api_mock = MagicMock()
|
|
|
|
api_mock.markets = markets
|
|
|
|
exchange = get_patched_exchange(mocker, default_conf, api_mock=api_mock)
|
|
|
|
|
|
|
|
dp = DataProvider(default_conf, exchange)
|
|
|
|
res = dp.market('ETH/BTC')
|
|
|
|
|
|
|
|
assert type(res) is dict
|
|
|
|
assert 'symbol' in res
|
|
|
|
assert res['symbol'] == 'ETH/BTC'
|
|
|
|
|
|
|
|
res = dp.market('UNITTEST/BTC')
|
|
|
|
assert res is None
|
2020-05-06 15:48:57 +00:00
|
|
|
|
|
|
|
|
2020-05-11 15:32:28 +00:00
|
|
|
@patch('freqtrade.pairlist.pairlistmanager.PairListManager')
|
|
|
|
@patch('freqtrade.exchange.Exchange')
|
|
|
|
def test_current_whitelist(exchange, PairListManager, default_conf):
|
|
|
|
# patch default conf to volumepairlist
|
|
|
|
default_conf['pairlists'][0] = {'method': 'VolumePairList', "number_assets": 5}
|
|
|
|
|
|
|
|
pairlist = PairListManager(exchange, default_conf)
|
|
|
|
dp = DataProvider(default_conf, exchange, pairlist)
|
|
|
|
|
|
|
|
# Simulate volumepairs from exchange.
|
|
|
|
# pairlist.refresh_pairlist()
|
|
|
|
|
|
|
|
# Set the pairs manually... this would be done in refresh pairlist
|
|
|
|
# default whitelist + volumePL - blacklist
|
|
|
|
default_whitelist = default_conf['exchange']['pair_whitelist']
|
|
|
|
default_blacklist = default_conf['exchange']['pair_blacklist']
|
|
|
|
volume_pairlist = ['ETH/BTC', 'LINK/BTC', 'ZRX/BTC', 'BCH/BTC', 'XRP/BTC']
|
|
|
|
current_whitelist = list(set(volume_pairlist + default_whitelist))
|
|
|
|
for pair in default_blacklist:
|
|
|
|
if pair in current_whitelist:
|
|
|
|
current_whitelist.remove(pair)
|
|
|
|
pairlist._whitelist = current_whitelist
|
|
|
|
|
|
|
|
assert dp.current_whitelist() == pairlist._whitelist
|
|
|
|
|
|
|
|
with pytest.raises(OperationalException):
|
|
|
|
dp = DataProvider(default_conf, exchange)
|
|
|
|
dp.current_whitelist()
|