mirror of
https://github.com/microsoft/qlib.git
synced 2026-06-30 17:41:18 +08:00
* Refine several todos * CI issues * Remove Dropna limitation of `quote_df` in Exchange (#1334) * Remove Dropna limitation of `quote_df` of Exchange * Impreove docstring * Fix type error when expression is specified (#1335) * Refine fill_missing_data() * Remove several TODO comments * Add back env for interpreters * Change Literal import * Resolve PR comments * Move to SAOEState * Add Trainer.get_policy_state_dict() * Mypy issue Co-authored-by: you-n-g <you-n-g@users.noreply.github.com>
111 lines
3.9 KiB
Python
111 lines
3.9 KiB
Python
# Copyright (c) Microsoft Corporation.
|
|
# Licensed under the MIT License.
|
|
|
|
from __future__ import annotations
|
|
|
|
from typing import Dict, TYPE_CHECKING, Generator, Optional, Tuple, Union, cast
|
|
|
|
import pandas as pd
|
|
|
|
from qlib.backtest.decision import BaseTradeDecision
|
|
from qlib.backtest.report import Indicator
|
|
|
|
if TYPE_CHECKING:
|
|
from qlib.strategy.base import BaseStrategy
|
|
from qlib.backtest.executor import BaseExecutor
|
|
|
|
from tqdm.auto import tqdm
|
|
|
|
from ..utils.time import Freq
|
|
|
|
|
|
PORT_METRIC = Dict[str, Tuple[pd.DataFrame, dict]]
|
|
INDICATOR_METRIC = Dict[str, Tuple[pd.DataFrame, Indicator]]
|
|
|
|
|
|
def backtest_loop(
|
|
start_time: Union[pd.Timestamp, str],
|
|
end_time: Union[pd.Timestamp, str],
|
|
trade_strategy: BaseStrategy,
|
|
trade_executor: BaseExecutor,
|
|
) -> Tuple[PORT_METRIC, INDICATOR_METRIC]:
|
|
"""backtest function for the interaction of the outermost strategy and executor in the nested decision execution
|
|
|
|
please refer to the docs of `collect_data_loop`
|
|
|
|
Returns
|
|
-------
|
|
portfolio_dict: PORT_METRIC
|
|
it records the trading portfolio_metrics information
|
|
indicator_dict: INDICATOR_METRIC
|
|
it computes the trading indicator
|
|
"""
|
|
return_value: dict = {}
|
|
for _decision in collect_data_loop(start_time, end_time, trade_strategy, trade_executor, return_value):
|
|
pass
|
|
|
|
portfolio_dict = cast(PORT_METRIC, return_value.get("portfolio_dict"))
|
|
indicator_dict = cast(INDICATOR_METRIC, return_value.get("indicator_dict"))
|
|
|
|
return portfolio_dict, indicator_dict
|
|
|
|
|
|
def collect_data_loop(
|
|
start_time: Union[pd.Timestamp, str],
|
|
end_time: Union[pd.Timestamp, str],
|
|
trade_strategy: BaseStrategy,
|
|
trade_executor: BaseExecutor,
|
|
return_value: dict = None,
|
|
) -> Generator[BaseTradeDecision, Optional[BaseTradeDecision], None]:
|
|
"""Generator for collecting the trade decision data for rl training
|
|
|
|
Parameters
|
|
----------
|
|
start_time : Union[pd.Timestamp, str]
|
|
closed start time for backtest
|
|
**NOTE**: This will be applied to the outmost executor's calendar.
|
|
end_time : Union[pd.Timestamp, str]
|
|
closed end time for backtest
|
|
**NOTE**: This will be applied to the outmost executor's calendar.
|
|
E.g. Executor[day](Executor[1min]), setting `end_time == 20XX0301` will include all the minutes on 20XX0301
|
|
trade_strategy : BaseStrategy
|
|
the outermost portfolio strategy
|
|
trade_executor : BaseExecutor
|
|
the outermost executor
|
|
return_value : dict
|
|
used for backtest_loop
|
|
|
|
Yields
|
|
-------
|
|
object
|
|
trade decision
|
|
"""
|
|
trade_executor.reset(start_time=start_time, end_time=end_time)
|
|
trade_strategy.reset(level_infra=trade_executor.get_level_infra())
|
|
|
|
with tqdm(total=trade_executor.trade_calendar.get_trade_len(), desc="backtest loop") as bar:
|
|
_execute_result = None
|
|
while not trade_executor.finished():
|
|
_trade_decision: BaseTradeDecision = trade_strategy.generate_trade_decision(_execute_result)
|
|
_execute_result = yield from trade_executor.collect_data(_trade_decision, level=0)
|
|
trade_strategy.post_exe_step(_execute_result)
|
|
bar.update(1)
|
|
trade_strategy.post_upper_level_exe_step()
|
|
|
|
if return_value is not None:
|
|
all_executors = trade_executor.get_all_executors()
|
|
|
|
portfolio_dict: PORT_METRIC = {}
|
|
indicator_dict: INDICATOR_METRIC = {}
|
|
|
|
for executor in all_executors:
|
|
key = "{}{}".format(*Freq.parse(executor.time_per_step))
|
|
if executor.trade_account.is_port_metr_enabled():
|
|
portfolio_dict[key] = executor.trade_account.get_portfolio_metrics()
|
|
|
|
indicator_df = executor.trade_account.get_trade_indicator().generate_trade_indicators_dataframe()
|
|
indicator_obj = executor.trade_account.get_trade_indicator()
|
|
indicator_dict[key] = (indicator_df, indicator_obj)
|
|
|
|
return_value.update({"portfolio_dict": portfolio_dict, "indicator_dict": indicator_dict})
|