stable/tests/test_utils.py

524 lines
16 KiB
Python
Raw Normal View History

2019-06-16 08:13:24 +00:00
import re
from pathlib import Path
2019-08-16 13:28:11 +00:00
from unittest.mock import MagicMock, PropertyMock
import pytest
from freqtrade import OperationalException
from freqtrade.state import RunMode
from freqtrade.utils import (setup_utils_configuration, start_create_userdir,
2019-10-03 23:01:44 +00:00
start_download_data, start_list_exchanges,
start_list_markets, start_list_timeframes)
from tests.conftest import get_args, log_has, patch_exchange
2019-06-16 08:13:24 +00:00
2019-06-16 18:37:59 +00:00
def test_setup_utils_configuration():
2019-06-16 08:13:24 +00:00
args = [
'--config', 'config.json.example',
]
2019-06-16 18:37:59 +00:00
config = setup_utils_configuration(get_args(args), RunMode.OTHER)
2019-06-16 08:13:24 +00:00
assert "exchange" in config
2019-06-16 18:37:59 +00:00
assert config['exchange']['dry_run'] is True
2019-06-16 08:13:24 +00:00
assert config['exchange']['key'] == ''
assert config['exchange']['secret'] == ''
def test_list_exchanges(capsys):
args = [
"list-exchanges",
]
start_list_exchanges(get_args(args))
captured = capsys.readouterr()
2019-09-30 21:33:54 +00:00
assert re.match(r"Exchanges available for Freqtrade.*", captured.out)
2019-06-16 08:13:24 +00:00
assert re.match(r".*binance,.*", captured.out)
assert re.match(r".*bittrex,.*", captured.out)
# Test with --one-column
args = [
"list-exchanges",
"--one-column",
]
start_list_exchanges(get_args(args))
captured = capsys.readouterr()
assert re.search(r"^binance$", captured.out, re.MULTILINE)
assert re.search(r"^bittrex$", captured.out, re.MULTILINE)
2019-09-30 21:33:54 +00:00
# Test with --all
args = [
"list-exchanges",
"--all",
]
start_list_exchanges(get_args(args))
captured = capsys.readouterr()
assert re.match(r"All exchanges supported by the ccxt library.*", captured.out)
assert re.match(r".*binance,.*", captured.out)
assert re.match(r".*bittrex,.*", captured.out)
assert re.match(r".*bitmex,.*", captured.out)
# Test with --one-column --all
args = [
"list-exchanges",
"--one-column",
"--all",
]
start_list_exchanges(get_args(args))
captured = capsys.readouterr()
assert re.search(r"^binance$", captured.out, re.MULTILINE)
assert re.search(r"^bittrex$", captured.out, re.MULTILINE)
assert re.search(r"^bitmex$", captured.out, re.MULTILINE)
2019-10-06 08:32:19 +00:00
def test_list_timeframes(mocker, capsys):
api_mock = MagicMock()
2019-10-06 12:33:23 +00:00
api_mock.timeframes = {'1m': 'oneMin',
'5m': 'fiveMin',
'30m': 'thirtyMin',
'1h': 'hour',
'1d': 'day',
2019-10-06 08:32:19 +00:00
}
patch_exchange(mocker, api_mock=api_mock)
2019-10-03 23:01:44 +00:00
args = [
"list-timeframes",
]
pargs = get_args(args)
pargs['config'] = None
with pytest.raises(OperationalException,
match=r"This command requires a configured exchange.*"):
start_list_timeframes(pargs)
# Test with --config config.json.example
args = [
'--config', 'config.json.example',
"list-timeframes",
]
start_list_timeframes(get_args(args))
captured = capsys.readouterr()
assert re.match("Timeframes available for the exchange `bittrex`: "
"1m, 5m, 30m, 1h, 1d",
captured.out)
# Test with --exchange bittrex
args = [
"list-timeframes",
"--exchange", "bittrex",
]
start_list_timeframes(get_args(args))
captured = capsys.readouterr()
assert re.match("Timeframes available for the exchange `bittrex`: "
"1m, 5m, 30m, 1h, 1d",
captured.out)
2019-10-06 08:32:19 +00:00
api_mock.timeframes = {'1m': '1m',
'5m': '5m',
'15m': '15m',
'30m': '30m',
'1h': '1h',
'6h': '6h',
'12h': '12h',
'1d': '1d',
'3d': '3d',
}
patch_exchange(mocker, api_mock=api_mock)
2019-10-03 23:01:44 +00:00
# Test with --exchange binance
args = [
"list-timeframes",
"--exchange", "binance",
]
start_list_timeframes(get_args(args))
captured = capsys.readouterr()
assert re.match("Timeframes available for the exchange `binance`: "
2019-10-06 08:32:19 +00:00
"1m, 5m, 15m, 30m, 1h, 6h, 12h, 1d, 3d",
2019-10-03 23:01:44 +00:00
captured.out)
# Test with --one-column
args = [
'--config', 'config.json.example',
"list-timeframes",
"--one-column",
]
start_list_timeframes(get_args(args))
captured = capsys.readouterr()
assert re.search(r"^1m$", captured.out, re.MULTILINE)
assert re.search(r"^5m$", captured.out, re.MULTILINE)
assert re.search(r"^1h$", captured.out, re.MULTILINE)
assert re.search(r"^1d$", captured.out, re.MULTILINE)
# Test with --exchange binance --one-column
args = [
"list-timeframes",
"--exchange", "binance",
"--one-column",
]
start_list_timeframes(get_args(args))
captured = capsys.readouterr()
assert re.search(r"^1m$", captured.out, re.MULTILINE)
assert re.search(r"^5m$", captured.out, re.MULTILINE)
assert re.search(r"^1h$", captured.out, re.MULTILINE)
assert re.search(r"^1d$", captured.out, re.MULTILINE)
2019-10-17 21:48:40 +00:00
def test_list_markets(mocker, markets, capsys):
api_mock = MagicMock()
api_mock.markets = markets
patch_exchange(mocker, api_mock=api_mock)
# Test with no --config
args = [
"list-markets",
]
2019-10-20 19:43:00 +00:00
pargs = get_args(args)
pargs['config'] = None
2019-10-17 21:48:40 +00:00
with pytest.raises(OperationalException,
match=r"This command requires a configured exchange.*"):
2019-10-20 19:43:00 +00:00
start_list_markets(pargs, False)
2019-10-17 21:48:40 +00:00
# Test with --config config.json.example
args = [
'--config', 'config.json.example',
"list-markets",
2019-10-18 11:25:43 +00:00
"--print-list",
2019-10-17 21:48:40 +00:00
]
start_list_markets(get_args(args), False)
2019-10-17 21:48:40 +00:00
captured = capsys.readouterr()
2019-10-18 11:25:43 +00:00
assert ("Exchange Bittrex has 8 active markets: "
"BLK/BTC, BTT/BTC, ETH/BTC, ETH/USDT, LTC/USD, LTC/USDT, TKN/BTC, XLTCUSDT.\n"
in captured.out)
2019-10-17 21:48:40 +00:00
2019-10-20 23:15:37 +00:00
# Test with --exchange
args = [
"list-markets",
"--exchange", "binance"
]
pargs = get_args(args)
pargs['config'] = None
start_list_markets(pargs, False)
captured = capsys.readouterr()
assert re.match("Exchange Binance has 8 active markets:\n",
captured.out)
2019-10-17 21:48:40 +00:00
# Test with --all: all markets
args = [
'--config', 'config.json.example',
2019-10-18 11:25:43 +00:00
"list-markets", "--all",
"--print-list",
2019-10-17 21:48:40 +00:00
]
start_list_markets(get_args(args), False)
2019-10-17 21:48:40 +00:00
captured = capsys.readouterr()
2019-10-18 11:25:43 +00:00
assert ("Exchange Bittrex has 11 markets: "
"BLK/BTC, BTT/BTC, ETH/BTC, ETH/USDT, LTC/BTC, LTC/USD, LTC/USDT, NEO/BTC, "
"TKN/BTC, XLTCUSDT, XRP/BTC.\n"
in captured.out)
2019-10-17 21:48:40 +00:00
# Test list-pairs subcommand: active pairs
args = [
'--config', 'config.json.example',
"list-pairs",
2019-10-18 11:25:43 +00:00
"--print-list",
2019-10-17 21:48:40 +00:00
]
start_list_markets(get_args(args), True)
2019-10-17 21:48:40 +00:00
captured = capsys.readouterr()
2019-10-18 11:25:43 +00:00
assert ("Exchange Bittrex has 7 active pairs: "
"BLK/BTC, BTT/BTC, ETH/BTC, ETH/USDT, LTC/USD, LTC/USDT, TKN/BTC.\n"
in captured.out)
2019-10-17 21:48:40 +00:00
# Test list-pairs subcommand with --all: all pairs
args = [
'--config', 'config.json.example',
2019-10-18 11:25:43 +00:00
"list-pairs", "--all",
"--print-list",
2019-10-17 21:48:40 +00:00
]
start_list_markets(get_args(args), True)
2019-10-17 21:48:40 +00:00
captured = capsys.readouterr()
2019-10-18 11:25:43 +00:00
assert ("Exchange Bittrex has 10 pairs: "
"BLK/BTC, BTT/BTC, ETH/BTC, ETH/USDT, LTC/BTC, LTC/USD, LTC/USDT, NEO/BTC, "
"TKN/BTC, XRP/BTC.\n"
in captured.out)
2019-10-17 21:48:40 +00:00
# active markets, base=ETH, LTC
args = [
'--config', 'config.json.example',
"list-markets",
2019-10-18 11:25:43 +00:00
"--base", "ETH", "LTC",
"--print-list",
2019-10-17 21:48:40 +00:00
]
start_list_markets(get_args(args), False)
2019-10-17 21:48:40 +00:00
captured = capsys.readouterr()
2019-10-18 11:25:43 +00:00
assert ("Exchange Bittrex has 5 active markets with ETH, LTC as base currencies: "
"ETH/BTC, ETH/USDT, LTC/USD, LTC/USDT, XLTCUSDT.\n"
in captured.out)
2019-10-17 21:48:40 +00:00
# active markets, base=LTC
args = [
'--config', 'config.json.example',
"list-markets",
2019-10-18 11:25:43 +00:00
"--base", "LTC",
"--print-list",
2019-10-17 21:48:40 +00:00
]
start_list_markets(get_args(args), False)
2019-10-17 21:48:40 +00:00
captured = capsys.readouterr()
2019-10-18 11:25:43 +00:00
assert ("Exchange Bittrex has 3 active markets with LTC as base currency: "
"LTC/USD, LTC/USDT, XLTCUSDT.\n"
in captured.out)
2019-10-17 21:48:40 +00:00
# active markets, quote=USDT, USD
args = [
'--config', 'config.json.example',
"list-markets",
2019-10-18 11:25:43 +00:00
"--quote", "USDT", "USD",
"--print-list",
2019-10-17 21:48:40 +00:00
]
start_list_markets(get_args(args), False)
2019-10-17 21:48:40 +00:00
captured = capsys.readouterr()
2019-10-18 11:25:43 +00:00
assert ("Exchange Bittrex has 4 active markets with USDT, USD as quote currencies: "
"ETH/USDT, LTC/USD, LTC/USDT, XLTCUSDT.\n"
in captured.out)
2019-10-17 21:48:40 +00:00
# active markets, quote=USDT
args = [
'--config', 'config.json.example',
"list-markets",
2019-10-18 11:25:43 +00:00
"--quote", "USDT",
"--print-list",
2019-10-17 21:48:40 +00:00
]
start_list_markets(get_args(args), False)
2019-10-17 21:48:40 +00:00
captured = capsys.readouterr()
2019-10-18 11:25:43 +00:00
assert ("Exchange Bittrex has 3 active markets with USDT as quote currency: "
"ETH/USDT, LTC/USDT, XLTCUSDT.\n"
in captured.out)
2019-10-17 21:48:40 +00:00
# active markets, base=LTC, quote=USDT
args = [
'--config', 'config.json.example',
"list-markets",
2019-10-18 11:25:43 +00:00
"--base", "LTC", "--quote", "USDT",
"--print-list",
2019-10-17 21:48:40 +00:00
]
start_list_markets(get_args(args), False)
2019-10-17 21:48:40 +00:00
captured = capsys.readouterr()
2019-10-18 11:25:43 +00:00
assert ("Exchange Bittrex has 2 active markets with LTC as base currency and "
"with USDT as quote currency: LTC/USDT, XLTCUSDT.\n"
in captured.out)
2019-10-17 21:48:40 +00:00
# active pairs, base=LTC, quote=USDT
args = [
'--config', 'config.json.example',
"list-pairs",
2019-10-18 11:25:43 +00:00
"--base", "LTC", "--quote", "USDT",
"--print-list",
2019-10-17 21:48:40 +00:00
]
start_list_markets(get_args(args), True)
2019-10-17 21:48:40 +00:00
captured = capsys.readouterr()
2019-10-18 11:25:43 +00:00
assert ("Exchange Bittrex has 1 active pair with LTC as base currency and "
"with USDT as quote currency: LTC/USDT.\n"
in captured.out)
2019-10-17 21:48:40 +00:00
# active markets, base=LTC, quote=USDT, NONEXISTENT
args = [
'--config', 'config.json.example',
"list-markets",
2019-10-18 11:25:43 +00:00
"--base", "LTC", "--quote", "USDT", "NONEXISTENT",
"--print-list",
2019-10-17 21:48:40 +00:00
]
start_list_markets(get_args(args), False)
2019-10-17 21:48:40 +00:00
captured = capsys.readouterr()
2019-10-18 11:25:43 +00:00
assert ("Exchange Bittrex has 2 active markets with LTC as base currency and "
"with USDT, NONEXISTENT as quote currencies: LTC/USDT, XLTCUSDT.\n"
in captured.out)
2019-10-17 21:48:40 +00:00
# active markets, base=LTC, quote=NONEXISTENT
args = [
'--config', 'config.json.example',
"list-markets",
2019-10-18 11:25:43 +00:00
"--base", "LTC", "--quote", "NONEXISTENT",
"--print-list",
2019-10-17 21:48:40 +00:00
]
start_list_markets(get_args(args), False)
2019-10-17 21:48:40 +00:00
captured = capsys.readouterr()
2019-10-18 11:25:43 +00:00
assert ("Exchange Bittrex has 0 active markets with LTC as base currency and "
"with NONEXISTENT as quote currency.\n"
in captured.out)
# Test tabular output
args = [
'--config', 'config.json.example',
"list-markets",
]
start_list_markets(get_args(args), False)
captured = capsys.readouterr()
assert ("Exchange Bittrex has 8 active markets:\n"
in captured.out)
# Test tabular output, no markets found
args = [
'--config', 'config.json.example',
"list-markets",
"--base", "LTC", "--quote", "NONEXISTENT",
]
start_list_markets(get_args(args), False)
captured = capsys.readouterr()
assert ("Exchange Bittrex has 0 active markets with LTC as base currency and "
"with NONEXISTENT as quote currency.\n"
in captured.out)
# Test --print-json
args = [
'--config', 'config.json.example',
"list-markets",
"--print-json"
]
start_list_markets(get_args(args), False)
captured = capsys.readouterr()
assert ('["BLK/BTC","BTT/BTC","ETH/BTC","ETH/USDT","LTC/USD","LTC/USDT","TKN/BTC","XLTCUSDT"]'
in captured.out)
# Test --print-csv
args = [
'--config', 'config.json.example',
"list-markets",
"--print-csv"
]
start_list_markets(get_args(args), False)
captured = capsys.readouterr()
assert ("Id,Symbol,Base,Quote,Active,Is pair" in captured.out)
assert ("blkbtc,BLK/BTC,BLK,BTC,True,True" in captured.out)
assert ("BTTBTC,BTT/BTC,BTT,BTC,True,True" in captured.out)
# Test --one-column
args = [
'--config', 'config.json.example',
"list-markets",
"--one-column"
]
start_list_markets(get_args(args), False)
captured = capsys.readouterr()
assert re.search(r"^BLK/BTC$", captured.out, re.MULTILINE)
assert re.search(r"^BTT/BTC$", captured.out, re.MULTILINE)
2019-10-17 21:48:40 +00:00
def test_create_datadir_failed(caplog):
args = [
"create-userdir",
]
with pytest.raises(SystemExit):
start_create_userdir(get_args(args))
2019-08-18 13:09:44 +00:00
assert log_has("`create-userdir` requires --userdir to be set.", caplog)
def test_create_datadir(caplog, mocker):
cud = mocker.patch("freqtrade.utils.create_userdata_dir", MagicMock())
args = [
"create-userdir",
"--userdir",
"/temp/freqtrade/test"
]
start_create_userdir(get_args(args))
assert cud.call_count == 1
assert len(caplog.record_tuples) == 0
2019-08-25 13:02:40 +00:00
def test_download_data_keyboardInterrupt(mocker, caplog, markets):
dl_mock = mocker.patch('freqtrade.utils.refresh_backtest_ohlcv_data',
MagicMock(side_effect=KeyboardInterrupt))
2019-08-17 04:58:38 +00:00
patch_exchange(mocker)
mocker.patch(
'freqtrade.exchange.Exchange.markets', PropertyMock(return_value=markets)
)
args = [
"download-data",
"--exchange", "binance",
"--pairs", "ETH/BTC", "XRP/BTC",
]
2019-08-25 13:02:40 +00:00
with pytest.raises(SystemExit):
start_download_data(get_args(args))
2019-08-17 04:58:38 +00:00
2019-08-25 13:02:40 +00:00
assert dl_mock.call_count == 1
2019-08-17 04:58:38 +00:00
2019-08-16 13:28:11 +00:00
def test_download_data_no_markets(mocker, caplog):
2019-08-25 13:02:40 +00:00
dl_mock = mocker.patch('freqtrade.utils.refresh_backtest_ohlcv_data',
MagicMock(return_value=["ETH/BTC", "XRP/BTC"]))
2019-08-16 13:28:11 +00:00
patch_exchange(mocker)
mocker.patch(
'freqtrade.exchange.Exchange.markets', PropertyMock(return_value={})
)
args = [
"download-data",
"--exchange", "binance",
2019-08-17 04:58:38 +00:00
"--pairs", "ETH/BTC", "XRP/BTC",
2019-08-25 13:02:40 +00:00
"--days", "20"
2019-08-16 13:28:11 +00:00
]
start_download_data(get_args(args))
2019-08-25 13:02:40 +00:00
assert dl_mock.call_args[1]['timerange'].starttype == "date"
2019-08-16 13:28:11 +00:00
assert log_has("Pairs [ETH/BTC,XRP/BTC] not available on exchange binance.", caplog)
def test_download_data_no_exchange(mocker, caplog):
mocker.patch('freqtrade.utils.refresh_backtest_ohlcv_data',
MagicMock(return_value=["ETH/BTC", "XRP/BTC"]))
patch_exchange(mocker)
mocker.patch(
'freqtrade.exchange.Exchange.markets', PropertyMock(return_value={})
)
args = [
"download-data",
]
pargs = get_args(args)
pargs['config'] = None
with pytest.raises(OperationalException,
match=r"This command requires a configured exchange.*"):
start_download_data(pargs)
def test_download_data_no_pairs(mocker, caplog):
mocker.patch.object(Path, "exists", MagicMock(return_value=False))
mocker.patch('freqtrade.utils.refresh_backtest_ohlcv_data',
MagicMock(return_value=["ETH/BTC", "XRP/BTC"]))
patch_exchange(mocker)
mocker.patch(
'freqtrade.exchange.Exchange.markets', PropertyMock(return_value={})
)
args = [
"download-data",
"--exchange",
"binance",
]
pargs = get_args(args)
pargs['config'] = None
with pytest.raises(OperationalException,
match=r"Downloading data requires a list of pairs\..*"):
start_download_data(pargs)
2019-10-08 18:31:14 +00:00
def test_download_data_trades(mocker, caplog):
dl_mock = mocker.patch('freqtrade.utils.refresh_backtest_trades_data',
MagicMock(return_value=[]))
convert_mock = mocker.patch('freqtrade.utils.convert_trades_to_ohlcv',
MagicMock(return_value=[]))
patch_exchange(mocker)
mocker.patch(
'freqtrade.exchange.Exchange.markets', PropertyMock(return_value={})
)
args = [
"download-data",
"--exchange", "kraken",
"--pairs", "ETH/BTC", "XRP/BTC",
"--days", "20",
"--dl-trades"
]
start_download_data(get_args(args))
assert dl_mock.call_args[1]['timerange'].starttype == "date"
assert dl_mock.call_count == 1
assert convert_mock.call_count == 1