Add test for detail backtesting
This commit is contained in:
parent
0888b53b5a
commit
5bd3e54b17
@ -787,17 +787,98 @@ def test_backtest_one(default_conf, fee, mocker, testdatadir) -> None:
|
|||||||
for _, t in results.iterrows():
|
for _, t in results.iterrows():
|
||||||
assert len(t['orders']) == 2
|
assert len(t['orders']) == 2
|
||||||
ln = data_pair.loc[data_pair["date"] == t["open_date"]]
|
ln = data_pair.loc[data_pair["date"] == t["open_date"]]
|
||||||
# Check open trade rate alignes to open rate
|
# Check open trade rate aligns to open rate
|
||||||
assert not ln.empty
|
assert not ln.empty
|
||||||
assert round(ln.iloc[0]["open"], 6) == round(t["open_rate"], 6)
|
assert round(ln.iloc[0]["open"], 6) == round(t["open_rate"], 6)
|
||||||
# check close trade rate alignes to close rate or is between high and low
|
# check close trade rate aligns to close rate or is between high and low
|
||||||
ln1 = data_pair.loc[data_pair["date"] == t["close_date"]]
|
ln1 = data_pair.loc[data_pair["date"] == t["close_date"]]
|
||||||
assert not ln1.empty
|
|
||||||
assert (round(ln1.iloc[0]["open"], 6) == round(t["close_rate"], 6) or
|
assert (round(ln1.iloc[0]["open"], 6) == round(t["close_rate"], 6) or
|
||||||
round(ln1.iloc[0]["low"], 6) < round(
|
round(ln1.iloc[0]["low"], 6) < round(
|
||||||
t["close_rate"], 6) < round(ln1.iloc[0]["high"], 6))
|
t["close_rate"], 6) < round(ln1.iloc[0]["high"], 6))
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.parametrize('use_detail', [True, False])
|
||||||
|
def test_backtest_one_detail(default_conf_usdt, fee, mocker, testdatadir, use_detail) -> None:
|
||||||
|
default_conf_usdt['use_exit_signal'] = False
|
||||||
|
mocker.patch('freqtrade.exchange.Exchange.get_fee', fee)
|
||||||
|
mocker.patch("freqtrade.exchange.Exchange.get_min_pair_stake_amount", return_value=0.00001)
|
||||||
|
mocker.patch("freqtrade.exchange.Exchange.get_max_pair_stake_amount", return_value=float('inf'))
|
||||||
|
if use_detail:
|
||||||
|
default_conf_usdt['timeframe_detail'] = '1m'
|
||||||
|
patch_exchange(mocker)
|
||||||
|
|
||||||
|
def advise_entry(df, *args, **kwargs):
|
||||||
|
# Mock function to force several entries
|
||||||
|
df.loc[(df['rsi'] < 40), 'enter_long'] = 1
|
||||||
|
return df
|
||||||
|
|
||||||
|
def custom_entry_price(proposed_rate, **kwargs):
|
||||||
|
return proposed_rate * 0.997
|
||||||
|
|
||||||
|
backtesting = Backtesting(default_conf_usdt)
|
||||||
|
backtesting._set_strategy(backtesting.strategylist[0])
|
||||||
|
backtesting.strategy.populate_entry_trend = advise_entry
|
||||||
|
backtesting.strategy.custom_entry_price = custom_entry_price
|
||||||
|
pair = 'XRP/ETH'
|
||||||
|
# Pick a timerange adapted to the pair we use to test
|
||||||
|
timerange = TimeRange.parse_timerange('20191010-20191013')
|
||||||
|
data = history.load_data(datadir=testdatadir, timeframe='5m', pairs=['XRP/ETH'],
|
||||||
|
timerange=timerange)
|
||||||
|
if use_detail:
|
||||||
|
data_1m = history.load_data(datadir=testdatadir, timeframe='1m', pairs=['XRP/ETH'],
|
||||||
|
timerange=timerange)
|
||||||
|
backtesting.detail_data = data_1m
|
||||||
|
processed = backtesting.strategy.advise_all_indicators(data)
|
||||||
|
min_date, max_date = get_timerange(processed)
|
||||||
|
|
||||||
|
result = backtesting.backtest(
|
||||||
|
processed=deepcopy(processed),
|
||||||
|
start_date=min_date,
|
||||||
|
end_date=max_date,
|
||||||
|
max_open_trades=10,
|
||||||
|
)
|
||||||
|
results = result['results']
|
||||||
|
assert not results.empty
|
||||||
|
# Timeout settings from default_conf = entry: 10, exit: 30
|
||||||
|
assert len(results) == (2 if use_detail else 3)
|
||||||
|
|
||||||
|
assert 'orders' in results.columns
|
||||||
|
data_pair = processed[pair]
|
||||||
|
|
||||||
|
data_1m_pair = data_1m[pair] if use_detail else pd.DataFrame()
|
||||||
|
late_entry = 0
|
||||||
|
for _, t in results.iterrows():
|
||||||
|
assert len(t['orders']) == 2
|
||||||
|
|
||||||
|
entryo = t['orders'][0]
|
||||||
|
entry_ts = datetime.fromtimestamp(entryo['order_filled_timestamp'] // 1000, tz=timezone.utc)
|
||||||
|
if entry_ts > t['open_date']:
|
||||||
|
late_entry += 1
|
||||||
|
|
||||||
|
# Get "entry fill" candle
|
||||||
|
ln = (data_1m_pair.loc[data_1m_pair["date"] == entry_ts]
|
||||||
|
if use_detail else data_pair.loc[data_pair["date"] == entry_ts])
|
||||||
|
# Check open trade rate aligns to open rate
|
||||||
|
assert not ln.empty
|
||||||
|
|
||||||
|
# assert round(ln.iloc[0]["open"], 6) == round(t["open_rate"], 6)
|
||||||
|
assert round(ln.iloc[0]["low"], 6) <= round(
|
||||||
|
t["open_rate"], 6) <= round(ln.iloc[0]["high"], 6)
|
||||||
|
# check close trade rate aligns to close rate or is between high and low
|
||||||
|
ln1 = data_pair.loc[data_pair["date"] == t["close_date"]]
|
||||||
|
if use_detail:
|
||||||
|
ln1_1m = data_1m_pair.loc[data_1m_pair["date"] == t["close_date"]]
|
||||||
|
assert not ln1.empty or not ln1_1m.empty
|
||||||
|
else:
|
||||||
|
assert not ln1.empty
|
||||||
|
ln2 = ln1_1m if ln1.empty else ln1
|
||||||
|
|
||||||
|
assert (round(ln2.iloc[0]["low"], 6) <= round(
|
||||||
|
t["close_rate"], 6) <= round(ln2.iloc[0]["high"], 6))
|
||||||
|
|
||||||
|
assert late_entry > 0
|
||||||
|
|
||||||
|
|
||||||
def test_backtest_timedout_entry_orders(default_conf, fee, mocker, testdatadir) -> None:
|
def test_backtest_timedout_entry_orders(default_conf, fee, mocker, testdatadir) -> None:
|
||||||
# This strategy intentionally places unfillable orders.
|
# This strategy intentionally places unfillable orders.
|
||||||
default_conf['strategy'] = 'StrategyTestV3CustomEntryPrice'
|
default_conf['strategy'] = 'StrategyTestV3CustomEntryPrice'
|
||||||
|
Loading…
Reference in New Issue
Block a user