investing-algorithm-framework 3.7.0__py3-none-any.whl → 7.19.15__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of investing-algorithm-framework might be problematic. Click here for more details.

Files changed (256) hide show
  1. investing_algorithm_framework/__init__.py +168 -45
  2. investing_algorithm_framework/app/__init__.py +32 -1
  3. investing_algorithm_framework/app/algorithm/__init__.py +7 -0
  4. investing_algorithm_framework/app/algorithm/algorithm.py +239 -0
  5. investing_algorithm_framework/app/algorithm/algorithm_factory.py +114 -0
  6. investing_algorithm_framework/app/analysis/__init__.py +15 -0
  7. investing_algorithm_framework/app/analysis/backtest_data_ranges.py +121 -0
  8. investing_algorithm_framework/app/analysis/backtest_utils.py +107 -0
  9. investing_algorithm_framework/app/analysis/permutation.py +116 -0
  10. investing_algorithm_framework/app/analysis/ranking.py +297 -0
  11. investing_algorithm_framework/app/app.py +1933 -589
  12. investing_algorithm_framework/app/app_hook.py +28 -0
  13. investing_algorithm_framework/app/context.py +1725 -0
  14. investing_algorithm_framework/app/eventloop.py +590 -0
  15. investing_algorithm_framework/app/reporting/__init__.py +27 -0
  16. investing_algorithm_framework/app/reporting/ascii.py +921 -0
  17. investing_algorithm_framework/app/reporting/backtest_report.py +349 -0
  18. investing_algorithm_framework/app/reporting/charts/__init__.py +19 -0
  19. investing_algorithm_framework/app/reporting/charts/entry_exist_signals.py +66 -0
  20. investing_algorithm_framework/app/reporting/charts/equity_curve.py +37 -0
  21. investing_algorithm_framework/app/reporting/charts/equity_curve_drawdown.py +74 -0
  22. investing_algorithm_framework/app/reporting/charts/line_chart.py +11 -0
  23. investing_algorithm_framework/app/reporting/charts/monthly_returns_heatmap.py +70 -0
  24. investing_algorithm_framework/app/reporting/charts/ohlcv_data_completeness.py +51 -0
  25. investing_algorithm_framework/app/reporting/charts/rolling_sharp_ratio.py +79 -0
  26. investing_algorithm_framework/app/reporting/charts/yearly_returns_barchart.py +55 -0
  27. investing_algorithm_framework/app/reporting/generate.py +185 -0
  28. investing_algorithm_framework/app/reporting/tables/__init__.py +11 -0
  29. investing_algorithm_framework/app/reporting/tables/key_metrics_table.py +217 -0
  30. investing_algorithm_framework/app/reporting/tables/stop_loss_table.py +0 -0
  31. investing_algorithm_framework/app/reporting/tables/time_metrics_table.py +80 -0
  32. investing_algorithm_framework/app/reporting/tables/trade_metrics_table.py +147 -0
  33. investing_algorithm_framework/app/reporting/tables/trades_table.py +75 -0
  34. investing_algorithm_framework/app/reporting/tables/utils.py +29 -0
  35. investing_algorithm_framework/app/reporting/templates/report_template.html.j2 +154 -0
  36. investing_algorithm_framework/app/stateless/action_handlers/__init__.py +4 -2
  37. investing_algorithm_framework/app/stateless/action_handlers/action_handler_strategy.py +1 -1
  38. investing_algorithm_framework/app/stateless/action_handlers/check_online_handler.py +1 -1
  39. investing_algorithm_framework/app/stateless/action_handlers/run_strategy_handler.py +14 -7
  40. investing_algorithm_framework/app/strategy.py +664 -84
  41. investing_algorithm_framework/app/task.py +5 -3
  42. investing_algorithm_framework/app/web/__init__.py +2 -1
  43. investing_algorithm_framework/app/web/create_app.py +4 -2
  44. investing_algorithm_framework/cli/__init__.py +0 -0
  45. investing_algorithm_framework/cli/cli.py +226 -0
  46. investing_algorithm_framework/cli/deploy_to_aws_lambda.py +501 -0
  47. investing_algorithm_framework/cli/deploy_to_azure_function.py +718 -0
  48. investing_algorithm_framework/cli/initialize_app.py +603 -0
  49. investing_algorithm_framework/cli/templates/.gitignore.template +178 -0
  50. investing_algorithm_framework/cli/templates/app.py.template +18 -0
  51. investing_algorithm_framework/cli/templates/app_aws_lambda_function.py.template +48 -0
  52. investing_algorithm_framework/cli/templates/app_azure_function.py.template +14 -0
  53. investing_algorithm_framework/cli/templates/app_web.py.template +18 -0
  54. investing_algorithm_framework/cli/templates/aws_lambda_dockerfile.template +22 -0
  55. investing_algorithm_framework/cli/templates/aws_lambda_dockerignore.template +92 -0
  56. investing_algorithm_framework/cli/templates/aws_lambda_readme.md.template +110 -0
  57. investing_algorithm_framework/cli/templates/aws_lambda_requirements.txt.template +2 -0
  58. investing_algorithm_framework/cli/templates/azure_function_function_app.py.template +65 -0
  59. investing_algorithm_framework/cli/templates/azure_function_host.json.template +15 -0
  60. investing_algorithm_framework/cli/templates/azure_function_local.settings.json.template +8 -0
  61. investing_algorithm_framework/cli/templates/azure_function_requirements.txt.template +3 -0
  62. investing_algorithm_framework/cli/templates/data_providers.py.template +17 -0
  63. investing_algorithm_framework/cli/templates/env.example.template +2 -0
  64. investing_algorithm_framework/cli/templates/env_azure_function.example.template +4 -0
  65. investing_algorithm_framework/cli/templates/market_data_providers.py.template +9 -0
  66. investing_algorithm_framework/cli/templates/readme.md.template +135 -0
  67. investing_algorithm_framework/cli/templates/requirements.txt.template +2 -0
  68. investing_algorithm_framework/cli/templates/run_backtest.py.template +20 -0
  69. investing_algorithm_framework/cli/templates/strategy.py.template +124 -0
  70. investing_algorithm_framework/create_app.py +40 -6
  71. investing_algorithm_framework/dependency_container.py +72 -56
  72. investing_algorithm_framework/domain/__init__.py +71 -47
  73. investing_algorithm_framework/domain/backtesting/__init__.py +21 -0
  74. investing_algorithm_framework/domain/backtesting/backtest.py +503 -0
  75. investing_algorithm_framework/domain/backtesting/backtest_date_range.py +96 -0
  76. investing_algorithm_framework/domain/backtesting/backtest_evaluation_focuss.py +242 -0
  77. investing_algorithm_framework/domain/backtesting/backtest_metrics.py +459 -0
  78. investing_algorithm_framework/domain/backtesting/backtest_permutation_test.py +275 -0
  79. investing_algorithm_framework/domain/backtesting/backtest_run.py +605 -0
  80. investing_algorithm_framework/domain/backtesting/backtest_summary_metrics.py +162 -0
  81. investing_algorithm_framework/domain/backtesting/combine_backtests.py +280 -0
  82. investing_algorithm_framework/domain/config.py +59 -91
  83. investing_algorithm_framework/domain/constants.py +13 -38
  84. investing_algorithm_framework/domain/data_provider.py +334 -0
  85. investing_algorithm_framework/domain/data_structures.py +3 -2
  86. investing_algorithm_framework/domain/exceptions.py +51 -1
  87. investing_algorithm_framework/domain/models/__init__.py +17 -12
  88. investing_algorithm_framework/domain/models/data/__init__.py +7 -0
  89. investing_algorithm_framework/domain/models/data/data_source.py +214 -0
  90. investing_algorithm_framework/domain/models/data/data_type.py +46 -0
  91. investing_algorithm_framework/domain/models/event.py +35 -0
  92. investing_algorithm_framework/domain/models/market/market_credential.py +55 -1
  93. investing_algorithm_framework/domain/models/order/order.py +77 -83
  94. investing_algorithm_framework/domain/models/order/order_status.py +2 -2
  95. investing_algorithm_framework/domain/models/order/order_type.py +1 -3
  96. investing_algorithm_framework/domain/models/portfolio/portfolio.py +81 -3
  97. investing_algorithm_framework/domain/models/portfolio/portfolio_configuration.py +26 -3
  98. investing_algorithm_framework/domain/models/portfolio/portfolio_snapshot.py +108 -11
  99. investing_algorithm_framework/domain/models/position/__init__.py +2 -1
  100. investing_algorithm_framework/domain/models/position/position.py +12 -0
  101. investing_algorithm_framework/domain/models/position/position_size.py +41 -0
  102. investing_algorithm_framework/domain/models/risk_rules/__init__.py +7 -0
  103. investing_algorithm_framework/domain/models/risk_rules/stop_loss_rule.py +51 -0
  104. investing_algorithm_framework/domain/models/risk_rules/take_profit_rule.py +55 -0
  105. investing_algorithm_framework/domain/models/snapshot_interval.py +45 -0
  106. investing_algorithm_framework/domain/models/strategy_profile.py +19 -151
  107. investing_algorithm_framework/domain/models/time_frame.py +37 -0
  108. investing_algorithm_framework/domain/models/time_interval.py +33 -0
  109. investing_algorithm_framework/domain/models/time_unit.py +66 -2
  110. investing_algorithm_framework/domain/models/trade/__init__.py +8 -1
  111. investing_algorithm_framework/domain/models/trade/trade.py +295 -171
  112. investing_algorithm_framework/domain/models/trade/trade_status.py +9 -2
  113. investing_algorithm_framework/domain/models/trade/trade_stop_loss.py +332 -0
  114. investing_algorithm_framework/domain/models/trade/trade_take_profit.py +365 -0
  115. investing_algorithm_framework/domain/order_executor.py +112 -0
  116. investing_algorithm_framework/domain/portfolio_provider.py +118 -0
  117. investing_algorithm_framework/domain/services/__init__.py +2 -9
  118. investing_algorithm_framework/domain/services/portfolios/portfolio_sync_service.py +0 -6
  119. investing_algorithm_framework/domain/services/state_handler.py +38 -0
  120. investing_algorithm_framework/domain/strategy.py +1 -29
  121. investing_algorithm_framework/domain/utils/__init__.py +12 -7
  122. investing_algorithm_framework/domain/utils/custom_tqdm.py +22 -0
  123. investing_algorithm_framework/domain/utils/dates.py +57 -0
  124. investing_algorithm_framework/domain/utils/jupyter_notebook_detection.py +19 -0
  125. investing_algorithm_framework/domain/utils/polars.py +53 -0
  126. investing_algorithm_framework/domain/utils/random.py +29 -0
  127. investing_algorithm_framework/download_data.py +108 -0
  128. investing_algorithm_framework/infrastructure/__init__.py +31 -18
  129. investing_algorithm_framework/infrastructure/data_providers/__init__.py +36 -0
  130. investing_algorithm_framework/infrastructure/data_providers/ccxt.py +1143 -0
  131. investing_algorithm_framework/infrastructure/data_providers/csv.py +568 -0
  132. investing_algorithm_framework/infrastructure/data_providers/pandas.py +599 -0
  133. investing_algorithm_framework/infrastructure/database/__init__.py +6 -2
  134. investing_algorithm_framework/infrastructure/database/sql_alchemy.py +86 -12
  135. investing_algorithm_framework/infrastructure/models/__init__.py +6 -11
  136. investing_algorithm_framework/infrastructure/models/order/__init__.py +2 -1
  137. investing_algorithm_framework/infrastructure/models/order/order.py +35 -49
  138. investing_algorithm_framework/infrastructure/models/order/order_metadata.py +44 -0
  139. investing_algorithm_framework/infrastructure/models/order_trade_association.py +10 -0
  140. investing_algorithm_framework/infrastructure/models/portfolio/__init__.py +1 -1
  141. investing_algorithm_framework/infrastructure/models/portfolio/portfolio_snapshot.py +8 -0
  142. investing_algorithm_framework/infrastructure/models/portfolio/{portfolio.py → sql_portfolio.py} +17 -5
  143. investing_algorithm_framework/infrastructure/models/trades/__init__.py +9 -0
  144. investing_algorithm_framework/infrastructure/models/trades/trade.py +130 -0
  145. investing_algorithm_framework/infrastructure/models/trades/trade_stop_loss.py +59 -0
  146. investing_algorithm_framework/infrastructure/models/trades/trade_take_profit.py +55 -0
  147. investing_algorithm_framework/infrastructure/order_executors/__init__.py +21 -0
  148. investing_algorithm_framework/infrastructure/order_executors/backtest_oder_executor.py +28 -0
  149. investing_algorithm_framework/infrastructure/order_executors/ccxt_order_executor.py +200 -0
  150. investing_algorithm_framework/infrastructure/portfolio_providers/__init__.py +19 -0
  151. investing_algorithm_framework/infrastructure/portfolio_providers/ccxt_portfolio_provider.py +199 -0
  152. investing_algorithm_framework/infrastructure/repositories/__init__.py +8 -0
  153. investing_algorithm_framework/infrastructure/repositories/order_metadata_repository.py +17 -0
  154. investing_algorithm_framework/infrastructure/repositories/order_repository.py +5 -0
  155. investing_algorithm_framework/infrastructure/repositories/portfolio_repository.py +1 -1
  156. investing_algorithm_framework/infrastructure/repositories/position_repository.py +11 -0
  157. investing_algorithm_framework/infrastructure/repositories/repository.py +81 -27
  158. investing_algorithm_framework/infrastructure/repositories/trade_repository.py +71 -0
  159. investing_algorithm_framework/infrastructure/repositories/trade_stop_loss_repository.py +29 -0
  160. investing_algorithm_framework/infrastructure/repositories/trade_take_profit_repository.py +29 -0
  161. investing_algorithm_framework/infrastructure/services/__init__.py +4 -4
  162. investing_algorithm_framework/infrastructure/services/aws/__init__.py +6 -0
  163. investing_algorithm_framework/infrastructure/services/aws/state_handler.py +113 -0
  164. investing_algorithm_framework/infrastructure/services/azure/__init__.py +5 -0
  165. investing_algorithm_framework/infrastructure/services/azure/state_handler.py +158 -0
  166. investing_algorithm_framework/services/__init__.py +113 -16
  167. investing_algorithm_framework/services/backtesting/__init__.py +0 -7
  168. investing_algorithm_framework/services/backtesting/backtest_service.py +566 -359
  169. investing_algorithm_framework/services/configuration_service.py +77 -11
  170. investing_algorithm_framework/services/data_providers/__init__.py +5 -0
  171. investing_algorithm_framework/services/data_providers/data_provider_service.py +850 -0
  172. investing_algorithm_framework/services/market_credential_service.py +16 -1
  173. investing_algorithm_framework/services/metrics/__init__.py +114 -0
  174. investing_algorithm_framework/services/metrics/alpha.py +0 -0
  175. investing_algorithm_framework/services/metrics/beta.py +0 -0
  176. investing_algorithm_framework/services/metrics/cagr.py +60 -0
  177. investing_algorithm_framework/services/metrics/calmar_ratio.py +40 -0
  178. investing_algorithm_framework/services/metrics/drawdown.py +181 -0
  179. investing_algorithm_framework/services/metrics/equity_curve.py +24 -0
  180. investing_algorithm_framework/services/metrics/exposure.py +210 -0
  181. investing_algorithm_framework/services/metrics/generate.py +358 -0
  182. investing_algorithm_framework/services/metrics/mean_daily_return.py +83 -0
  183. investing_algorithm_framework/services/metrics/profit_factor.py +165 -0
  184. investing_algorithm_framework/services/metrics/recovery.py +113 -0
  185. investing_algorithm_framework/services/metrics/returns.py +452 -0
  186. investing_algorithm_framework/services/metrics/risk_free_rate.py +28 -0
  187. investing_algorithm_framework/services/metrics/sharpe_ratio.py +137 -0
  188. investing_algorithm_framework/services/metrics/sortino_ratio.py +74 -0
  189. investing_algorithm_framework/services/metrics/standard_deviation.py +157 -0
  190. investing_algorithm_framework/services/metrics/trades.py +500 -0
  191. investing_algorithm_framework/services/metrics/treynor_ratio.py +0 -0
  192. investing_algorithm_framework/services/metrics/ulcer.py +0 -0
  193. investing_algorithm_framework/services/metrics/value_at_risk.py +0 -0
  194. investing_algorithm_framework/services/metrics/volatility.py +97 -0
  195. investing_algorithm_framework/services/metrics/win_rate.py +177 -0
  196. investing_algorithm_framework/services/order_service/__init__.py +3 -1
  197. investing_algorithm_framework/services/order_service/order_backtest_service.py +76 -89
  198. investing_algorithm_framework/services/order_service/order_executor_lookup.py +110 -0
  199. investing_algorithm_framework/services/order_service/order_service.py +407 -326
  200. investing_algorithm_framework/services/portfolios/__init__.py +3 -1
  201. investing_algorithm_framework/services/portfolios/backtest_portfolio_service.py +37 -3
  202. investing_algorithm_framework/services/portfolios/portfolio_configuration_service.py +22 -8
  203. investing_algorithm_framework/services/portfolios/portfolio_provider_lookup.py +106 -0
  204. investing_algorithm_framework/services/portfolios/portfolio_service.py +96 -28
  205. investing_algorithm_framework/services/portfolios/portfolio_snapshot_service.py +97 -28
  206. investing_algorithm_framework/services/portfolios/portfolio_sync_service.py +116 -313
  207. investing_algorithm_framework/services/positions/__init__.py +7 -0
  208. investing_algorithm_framework/services/positions/position_service.py +210 -0
  209. investing_algorithm_framework/services/repository_service.py +8 -2
  210. investing_algorithm_framework/services/trade_order_evaluator/__init__.py +9 -0
  211. investing_algorithm_framework/services/trade_order_evaluator/backtest_trade_oder_evaluator.py +113 -0
  212. investing_algorithm_framework/services/trade_order_evaluator/default_trade_order_evaluator.py +51 -0
  213. investing_algorithm_framework/services/trade_order_evaluator/trade_order_evaluator.py +80 -0
  214. investing_algorithm_framework/services/trade_service/__init__.py +7 -1
  215. investing_algorithm_framework/services/trade_service/trade_service.py +1013 -315
  216. investing_algorithm_framework/services/trade_service/trade_stop_loss_service.py +39 -0
  217. investing_algorithm_framework/services/trade_service/trade_take_profit_service.py +41 -0
  218. investing_algorithm_framework-7.19.15.dist-info/METADATA +537 -0
  219. investing_algorithm_framework-7.19.15.dist-info/RECORD +263 -0
  220. investing_algorithm_framework-7.19.15.dist-info/entry_points.txt +3 -0
  221. investing_algorithm_framework/app/algorithm.py +0 -1105
  222. investing_algorithm_framework/domain/graphs.py +0 -382
  223. investing_algorithm_framework/domain/metrics/__init__.py +0 -6
  224. investing_algorithm_framework/domain/models/backtesting/__init__.py +0 -11
  225. investing_algorithm_framework/domain/models/backtesting/backtest_date_range.py +0 -43
  226. investing_algorithm_framework/domain/models/backtesting/backtest_position.py +0 -120
  227. investing_algorithm_framework/domain/models/backtesting/backtest_report.py +0 -580
  228. investing_algorithm_framework/domain/models/backtesting/backtest_reports_evaluation.py +0 -243
  229. investing_algorithm_framework/domain/models/trading_data_types.py +0 -47
  230. investing_algorithm_framework/domain/models/trading_time_frame.py +0 -223
  231. investing_algorithm_framework/domain/services/market_data_sources.py +0 -344
  232. investing_algorithm_framework/domain/services/market_service.py +0 -153
  233. investing_algorithm_framework/domain/singleton.py +0 -9
  234. investing_algorithm_framework/domain/utils/backtesting.py +0 -472
  235. investing_algorithm_framework/infrastructure/models/market_data_sources/__init__.py +0 -12
  236. investing_algorithm_framework/infrastructure/models/market_data_sources/ccxt.py +0 -559
  237. investing_algorithm_framework/infrastructure/models/market_data_sources/csv.py +0 -254
  238. investing_algorithm_framework/infrastructure/models/market_data_sources/us_treasury_yield.py +0 -47
  239. investing_algorithm_framework/infrastructure/services/market_service/__init__.py +0 -5
  240. investing_algorithm_framework/infrastructure/services/market_service/ccxt_market_service.py +0 -455
  241. investing_algorithm_framework/infrastructure/services/performance_service/__init__.py +0 -7
  242. investing_algorithm_framework/infrastructure/services/performance_service/backtest_performance_service.py +0 -2
  243. investing_algorithm_framework/infrastructure/services/performance_service/performance_service.py +0 -350
  244. investing_algorithm_framework/services/backtesting/backtest_report_writer_service.py +0 -53
  245. investing_algorithm_framework/services/backtesting/graphs.py +0 -61
  246. investing_algorithm_framework/services/market_data_source_service/__init__.py +0 -8
  247. investing_algorithm_framework/services/market_data_source_service/backtest_market_data_source_service.py +0 -150
  248. investing_algorithm_framework/services/market_data_source_service/market_data_source_service.py +0 -189
  249. investing_algorithm_framework/services/position_service.py +0 -31
  250. investing_algorithm_framework/services/strategy_orchestrator_service.py +0 -264
  251. investing_algorithm_framework-3.7.0.dist-info/METADATA +0 -339
  252. investing_algorithm_framework-3.7.0.dist-info/RECORD +0 -147
  253. /investing_algorithm_framework/{domain → services}/metrics/price_efficiency.py +0 -0
  254. /investing_algorithm_framework/services/{position_snapshot_service.py → positions/position_snapshot_service.py} +0 -0
  255. {investing_algorithm_framework-3.7.0.dist-info → investing_algorithm_framework-7.19.15.dist-info}/LICENSE +0 -0
  256. {investing_algorithm_framework-3.7.0.dist-info → investing_algorithm_framework-7.19.15.dist-info}/WHEEL +0 -0
@@ -0,0 +1,459 @@
1
+ import os
2
+ from pathlib import Path
3
+ from dataclasses import dataclass, field
4
+ from logging import getLogger
5
+ from typing import Tuple, List, Dict
6
+ from datetime import datetime, date
7
+ import json
8
+ import pandas as pd
9
+
10
+ from investing_algorithm_framework.domain.models import Trade
11
+
12
+
13
+ logger = getLogger(__name__)
14
+
15
+
16
+ @dataclass
17
+ class BacktestMetrics:
18
+ """
19
+ Represents the result of a backtest, including metrics such as
20
+ total return, annualized return, volatility, Sharpe ratio,
21
+ and maximum drawdown.
22
+
23
+ Attributes:
24
+ backtest_start_date (datetime): The start date of the backtest.
25
+ backtest_end_date (datetime): The end date of the backtest.
26
+ final_value (float): The final value of the portfolio at the end
27
+ of the backtest.
28
+ equity_curve (List[Tuple[datetime, float]]): A list of
29
+ tuples representing the equity curve, where each tuple
30
+ contains a date and the corresponding portfolio value.
31
+ total_growth (float): The growth of the portfolio over the
32
+ backtest period.
33
+ total_growth_percentage (float): The percentage growth of the portfolio
34
+ over the backtest period.
35
+ total_net_gain (float): The total return of the backtest.
36
+ total_net_gain_percentage (float): The total return percentage
37
+ total_loss (float): The total loss of the backtest.
38
+ total_loss_percentage (float): The total loss percentage
39
+ cagr (float): The compound annual growth rate of the backtest.
40
+ sharpe_ratio (float): The Sharpe ratio of the backtest, indicating
41
+ risk-adjusted return.
42
+ rolling_sharpe_ratio (List[Tuple[datetime, float]): A list of rolling
43
+ Sharpe ratios over the backtest period.
44
+ sortino_ratio (float): The Sortino ratio of the backtest, focusing
45
+ on downside risk.
46
+ calmar_ratio (float): The Calmar ratio of the backtest, comparing
47
+ CAGR to maximum drawdown.
48
+ profit_factor (float): The profit factor of the backtest, calculated
49
+ as total profit divided by total loss.
50
+ annual_volatility (float): The annualized volatility of the
51
+ portfolio returns.
52
+ monthly_returns (List[Tuple[datetime, float]]): A list of monthly
53
+ returns during the backtest.
54
+ yearly_returns (List[Tuple[datetime, float]]): A list of yearly returns
55
+ during the backtest.
56
+ drawdown_series (List[Tuple[datetime, float]]): A list of drawdown
57
+ values over the backtest period.
58
+ max_drawdown (float): The maximum drawdown observed during
59
+ the backtest.
60
+ max_drawdown_absolute (float): The maximum absolute drawdown
61
+ observed during the backtest.
62
+ max_daily_drawdown (float): The maximum daily drawdown
63
+ observed during the backtest.
64
+ max_drawdown_duration (int): The duration of the maximum
65
+ drawdown in days.
66
+ trades_per_year (float): The average number of trades
67
+ executed per year.
68
+ trade_per_day (float): The average number of trades executed per day.
69
+ exposure_ratio (float): The exposure ratio, indicating the
70
+ average exposure of the portfolio.
71
+ cumulative_exposure (float): The cumulative exposure, indicating the
72
+ total exposure of the portfolio over the backtest period.
73
+ average_trade_size (float): The average size of trades executed
74
+ during the backtest.
75
+ average_trade_loss (float): The average loss from losing trades.
76
+ average_trade_loss_percentage (float): The average loss percentage
77
+ from losing trades.
78
+ average_trade_gain (float): The average gain from winning trades.
79
+ average_trade_gain_percentage (float): The average gain percentage
80
+ from winning trades.
81
+ average_trade_return (float): The average return from all trades.
82
+ average_trade_return_percentage (float): The average return percentage
83
+ from all trades.
84
+ median_trade_return (float): The median return from all trades.
85
+ median_trade_return_percentage (float): The median return percentage
86
+ from all trades.
87
+ number_of_positive_trades (int): The total number of profitable trades
88
+ executed during the backtest.
89
+ number_of_negative_trades (int): The total number of unprofitable
90
+ trades executed during the backtest.
91
+ best_trade (float): A string representation of the best trade,
92
+ including net gain and percentage.
93
+ worst_trade (float): A string representation of the worst trade,
94
+ including net loss and percentage.
95
+ average_trade_duration (float): The average duration of
96
+ trades in hours.
97
+ number_of_trades (int): The total number of trades executed
98
+ during the backtest.
99
+ number_of_trades_closed (int): The total number of trades close
100
+ during the backtest.
101
+ number_of_trades_opened (int): The total number of trades opened
102
+ during the backtest.
103
+ number_of_trades_open_at_end (int): The number of trades
104
+ still open at the end of the backtest.
105
+ win_rate (float): The win rate of the trades, expressed
106
+ as a percentage.
107
+ current_win_rate (float): The current win rate of the trades,
108
+ including open trades.
109
+ win_loss_ratio (float): The ratio of winning trades
110
+ to losing trades.
111
+ current_win_loss_ratio (float): The current ratio of winning
112
+ trades to losing trades, including open trades.
113
+ percentage_winning_months (float): The percentage of months
114
+ with positive returns.
115
+ percentage_winning_years (float): The percentage of years with
116
+ positive returns.
117
+ percentage_positive_trades (float): The percentage of trades that
118
+ were profitable.
119
+ percentage_negative_trades (float): The percentage of trades that
120
+ were unprofitable.
121
+ average_monthly_return (float): The average monthly return
122
+ of the portfolio.
123
+ average_monthly_return_losing_months (float): The average monthly
124
+ return during losing months.
125
+ average_monthly_return_winning_months (float): The average monthly
126
+ return during winning months.
127
+ best_month (datetime): A string representation of the best month,
128
+ including return and date.
129
+ best_year (datetime): A string representation of the best year,
130
+ including return and date.
131
+ worst_month (datetime): A string representation of the worst month,
132
+ including return and date.
133
+ worst_year (datetime): A string representation of the worst year,
134
+ including return and date.
135
+ metadata (Dict[str, str]): A dictionary to store any additional
136
+ metadata related to the backtest.
137
+ """
138
+ backtest_start_date: datetime
139
+ backtest_end_date: datetime
140
+ equity_curve: List[Tuple[float, datetime]] = field(default_factory=list)
141
+ total_growth: float = 0.0
142
+ total_growth_percentage: float = 0.0
143
+ total_net_gain: float = 0.0
144
+ total_net_gain_percentage: float = 0.0
145
+ total_loss: float = 0.0
146
+ total_loss_percentage: float = 0.0
147
+ final_value: float = 0.0
148
+ cumulative_return: float = 0.0
149
+ cumulative_return_series: List[Tuple[float, datetime]] = \
150
+ field(default_factory=list)
151
+ cagr: float = 0.0
152
+ sharpe_ratio: float = 0.0
153
+ rolling_sharpe_ratio: List[Tuple[float, datetime]] = \
154
+ field(default_factory=list)
155
+ sortino_ratio: float = 0.0
156
+ calmar_ratio: float = 0.0
157
+ profit_factor: float = 0.0
158
+ gross_profit: float = None
159
+ gross_loss: float = None
160
+ annual_volatility: float = 0.0
161
+ monthly_returns: List[Tuple[float, datetime]] = field(default_factory=list)
162
+ yearly_returns: List[Tuple[float, date]] = field(default_factory=list)
163
+ drawdown_series: List[Tuple[float, datetime]] = field(default_factory=list)
164
+ max_drawdown: float = 0.0
165
+ max_drawdown_absolute: float = 0.0
166
+ max_daily_drawdown: float = 0.0
167
+ max_drawdown_duration: int = 0
168
+ trades_per_year: float = 0.0
169
+ trade_per_day: float = 0.0
170
+ exposure_ratio: float = 0.0
171
+ cumulative_exposure: float = 0.0
172
+ best_trade: Trade = None
173
+ worst_trade: Trade = None
174
+ number_of_positive_trades: int = 0
175
+ percentage_positive_trades: float = 0.0
176
+ number_of_negative_trades: int = 0
177
+ percentage_negative_trades: float = 0.0
178
+ average_trade_duration: float = 0.0
179
+ average_trade_size: float = 0.0
180
+ average_trade_loss: float = 0.0
181
+ average_trade_loss_percentage: float = 0.0
182
+ average_trade_gain: float = 0.0
183
+ average_trade_gain_percentage: float = 0.0
184
+ average_trade_return: float = 0.0
185
+ average_trade_return_percentage: float = 0.0
186
+ current_average_trade_gain: float = 0.0
187
+ current_average_trade_gain_percentage: float = 0.0
188
+ current_average_trade_return: float = 0.0
189
+ current_average_trade_return_percentage: float = 0.0
190
+ current_average_trade_duration: float = 0.0
191
+ current_average_trade_loss: float = 0.0
192
+ current_average_trade_loss_percentage: float = 0.0
193
+ median_trade_return: float = 0.0
194
+ median_trade_return_percentage: float = 0.0
195
+ number_of_trades: int = 0
196
+ number_of_trades_closed: int = 0
197
+ number_of_trades_opened: int = 0
198
+ number_of_trades_open_at_end: int = 0
199
+ win_rate: float = 0.0
200
+ current_win_rate: float = 0.0
201
+ win_loss_ratio: float = 0.0
202
+ current_win_loss_ratio: float = 0.0
203
+ percentage_winning_months: float = 0.0
204
+ percentage_winning_years: float = 0.0
205
+ average_monthly_return: float = 0.0
206
+ average_monthly_return_losing_months: float = 0.0
207
+ average_monthly_return_winning_months: float = 0.0
208
+ best_month: Tuple[float, datetime] = None
209
+ best_year: Tuple[float, date] = None
210
+ worst_month: Tuple[float, datetime] = None
211
+ worst_year: Tuple[float, date] = None
212
+ total_number_of_days: int = None
213
+ metadata: Dict[str, str] = field(default_factory=dict)
214
+
215
+ def __post_init__(self):
216
+ self.total_number_of_days = (self.backtest_end_date -
217
+ self.backtest_start_date).days
218
+
219
+ def to_dict(self) -> dict:
220
+ """
221
+ Convert the BacktestMetrics instance to a dictionary.
222
+ Ensures all datetime values are serialized to ISO format, but
223
+ leaves strings unchanged.
224
+
225
+ Returns:
226
+ dict: A dictionary representation of the BacktestMetrics instance.
227
+ """
228
+
229
+ def ensure_iso(value):
230
+ return value.isoformat() \
231
+ if hasattr(value, "isoformat") else value
232
+
233
+ return {
234
+ "backtest_start_date": ensure_iso(self.backtest_start_date),
235
+ "backtest_end_date": ensure_iso(self.backtest_end_date),
236
+ "equity_curve": [(value, ensure_iso(date))
237
+ for value, date in self.equity_curve],
238
+ "final_value": self.final_value,
239
+ "total_net_gain": self.total_net_gain,
240
+ "total_net_gain_percentage": self.total_net_gain_percentage,
241
+ "total_growth": self.total_growth,
242
+ "total_growth_percentage": self.total_growth_percentage,
243
+ "total_loss": self.total_loss,
244
+ "total_loss_percentage": self.total_loss_percentage,
245
+ "cumulative_return": self.cumulative_return,
246
+ "cumulative_return_series": [(value, ensure_iso(date))
247
+ for value, date in
248
+ self.cumulative_return_series],
249
+ "cagr": self.cagr,
250
+ "sharpe_ratio": self.sharpe_ratio,
251
+ "rolling_sharpe_ratio": [(value, ensure_iso(date))
252
+ for value, date in
253
+ self.rolling_sharpe_ratio],
254
+ "sortino_ratio": self.sortino_ratio,
255
+ "calmar_ratio": self.calmar_ratio,
256
+ "profit_factor": self.profit_factor,
257
+ "annual_volatility": self.annual_volatility,
258
+ "monthly_returns": [(value, ensure_iso(date))
259
+ for value, date in self.monthly_returns],
260
+ "yearly_returns": [(value, ensure_iso(date))
261
+ for value, date in self.yearly_returns],
262
+ "drawdown_series": [(value, ensure_iso(date))
263
+ for value, date in self.drawdown_series],
264
+ "max_drawdown": self.max_drawdown,
265
+ "max_drawdown_absolute": self.max_drawdown_absolute,
266
+ "max_daily_drawdown": self.max_daily_drawdown,
267
+ "max_drawdown_duration": self.max_drawdown_duration,
268
+ "trades_per_year": self.trades_per_year,
269
+ "trade_per_day": self.trade_per_day,
270
+ "exposure_ratio": self.exposure_ratio,
271
+ "cumulative_exposure": self.cumulative_exposure,
272
+ "average_trade_gain": self.average_trade_gain,
273
+ "average_trade_gain_percentage":
274
+ self.average_trade_gain_percentage,
275
+ "average_trade_loss": self.average_trade_loss,
276
+ "average_trade_loss_percentage":
277
+ self.average_trade_loss_percentage,
278
+ "average_trade_return": self.average_trade_return,
279
+ "average_trade_return_percentage":
280
+ self.average_trade_return_percentage,
281
+ "median_trade_return": self.median_trade_return,
282
+ "median_trade_return_percentage":
283
+ self.median_trade_return_percentage,
284
+ "number_of_positive_trades": self.number_of_positive_trades,
285
+ "percentage_positive_trades": self.percentage_positive_trades,
286
+ "number_of_negative_trades": self.number_of_negative_trades,
287
+ "percentage_negative_trades": self.percentage_negative_trades,
288
+ "best_trade": self.best_trade.to_dict()
289
+ if self.best_trade else None,
290
+ "worst_trade": self.worst_trade.to_dict()
291
+ if self.worst_trade else None,
292
+ "average_trade_duration": self.average_trade_duration,
293
+ "average_trade_size": self.average_trade_size,
294
+ "number_of_trades": self.number_of_trades,
295
+ "number_of_trades_closed": self.number_of_trades_closed,
296
+ "number_of_trades_opened": self.number_of_trades_opened,
297
+ "win_rate": self.win_rate,
298
+ "current_win_rate": self.current_win_rate,
299
+ "win_loss_ratio": self.win_loss_ratio,
300
+ "current_win_loss_ratio": self.current_win_loss_ratio,
301
+ "percentage_winning_months": self.percentage_winning_months,
302
+ "percentage_winning_years": self.percentage_winning_years,
303
+ "average_monthly_return": self.average_monthly_return,
304
+ "average_monthly_return_losing_months":
305
+ self.average_monthly_return_losing_months,
306
+ "average_monthly_return_winning_months":
307
+ self.average_monthly_return_winning_months,
308
+ "best_month": self.best_month,
309
+ "best_year": self.best_year,
310
+ "worst_month": self.worst_month,
311
+ "worst_year": self.worst_year
312
+ }
313
+
314
+ def save(self, file_path: str | Path) -> None:
315
+ """
316
+ Save the backtest metrics to a file in JSON format. The metrics will
317
+ always be saved in a file named `metrics.json`
318
+
319
+ Args:
320
+ file_path (str): The directory where the metrics
321
+ file will be saved.
322
+ """
323
+ with open(file_path, 'w') as file:
324
+ json.dump(self.to_dict(), file, indent=4, default=str)
325
+
326
+ @staticmethod
327
+ def _parse_tuple_list_datetime(
328
+ data: List[List]
329
+ ) -> List[Tuple[float, datetime]]:
330
+ """
331
+ Parse a list of [value, datetime_string]
332
+ into List[Tuple[float, datetime]]
333
+ """
334
+ return [
335
+ (float(value), datetime.fromisoformat(date_str))
336
+ for value, date_str in data
337
+ ]
338
+
339
+ @staticmethod
340
+ def _parse_tuple_list_date(data: List[List]) -> List[Tuple[float, date]]:
341
+ """
342
+ Parse a list of [value, date_string] into List[Tuple[float, date]]
343
+ """
344
+ return [
345
+ (float(value), datetime.fromisoformat(date_str).date())
346
+ for value, date_str in data
347
+ ]
348
+
349
+ @staticmethod
350
+ def _parse_tuple_datetime(data) -> Tuple[float, datetime]:
351
+ """Parse a [value, datetime_string] into Tuple[float, datetime]"""
352
+ if data is None:
353
+ return None, None
354
+
355
+ # Check if the value is NaN or None
356
+ if pd.isna(data[0]) or data[0] is None:
357
+ value = pd.NA
358
+ else:
359
+ value = float(data[0])
360
+
361
+ # Parse the datetime string
362
+ if data[1] is None or pd.isna(data[1]):
363
+ value = None
364
+ else:
365
+ # Convert the string to a datetime object
366
+ date = datetime.fromisoformat(data[1])
367
+
368
+ return (value, date)
369
+
370
+ @staticmethod
371
+ def _parse_tuple_date(data) -> Tuple[float, date]:
372
+ """Parse a [value, date_string] into Tuple[float, date]"""
373
+ if data is None:
374
+ return None, None
375
+
376
+ # Check if the value is
377
+ if pd.isna(data[0]) or data[0] is None:
378
+ value = pd.NA
379
+ else:
380
+ value = float(data[0])
381
+
382
+ # Parse the date string
383
+ if data[1] is None or pd.isna(data[1]):
384
+ date = None
385
+ else:
386
+ date = datetime.fromisoformat(data[1]).date()
387
+
388
+ return (value, date)
389
+
390
+ @staticmethod
391
+ def open(file_path: str | Path) -> 'BacktestMetrics':
392
+ """
393
+ Open a backtest metrics file from a directory and
394
+ return a BacktestMetrics instance.
395
+
396
+ Args:
397
+ file_path (str): The path to the metrics file.
398
+
399
+ Returns:
400
+ BacktestMetrics: An instance of BacktestMetrics
401
+ loaded from the file.
402
+ """
403
+ if not os.path.exists(file_path):
404
+ raise FileNotFoundError(f"Metrics file not found at {file_path}")
405
+
406
+ with open(file_path, 'r') as file:
407
+ data = json.load(file)
408
+
409
+ # Parse datetime fields
410
+ data['backtest_start_date'] = datetime.fromisoformat(
411
+ data['backtest_start_date']
412
+ )
413
+ data['backtest_end_date'] = datetime.fromisoformat(
414
+ data['backtest_end_date']
415
+ )
416
+
417
+ # Parse tuple lists with datetime
418
+ data['equity_curve'] = BacktestMetrics._parse_tuple_list_datetime(
419
+ data.get('equity_curve', [])
420
+ )
421
+ data['rolling_sharpe_ratio'] = BacktestMetrics\
422
+ ._parse_tuple_list_datetime(data.get('rolling_sharpe_ratio', []))
423
+ data['monthly_returns'] = BacktestMetrics\
424
+ ._parse_tuple_list_datetime(data.get('monthly_returns', []))
425
+ data['drawdown_series'] = BacktestMetrics\
426
+ ._parse_tuple_list_datetime(data.get('drawdown_series', []))
427
+
428
+ # Parse tuple lists with date
429
+ data['yearly_returns'] = BacktestMetrics\
430
+ ._parse_tuple_list_date(data.get('yearly_returns', []))
431
+
432
+ # Parse single tuples
433
+ data['best_month'] = BacktestMetrics\
434
+ ._parse_tuple_datetime(data.get('best_month'))
435
+ data['worst_month'] = BacktestMetrics\
436
+ ._parse_tuple_datetime(data.get('worst_month'))
437
+ data['best_year'] = BacktestMetrics\
438
+ ._parse_tuple_date(data.get('best_year'))
439
+ data['worst_year'] = BacktestMetrics\
440
+ ._parse_tuple_date(data.get('worst_year'))
441
+
442
+ # Parse Trade objects if they exist
443
+ if data.get('best_trade'):
444
+ data['best_trade'] = Trade.from_dict(data['best_trade'])
445
+ if data.get('worst_trade'):
446
+ data['worst_trade'] = Trade.from_dict(data['worst_trade'])
447
+
448
+ return BacktestMetrics(**data)
449
+
450
+ def __repr__(self):
451
+ """
452
+ Return a string representation of the Backtest instance.
453
+
454
+ Returns:
455
+ str: A string representation of the Backtest instance.
456
+ """
457
+ return json.dumps(
458
+ self.to_dict(), indent=4, sort_keys=True, default=str
459
+ )