investing-algorithm-framework 1.5__py3-none-any.whl → 7.25.6__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.
Files changed (276) hide show
  1. investing_algorithm_framework/__init__.py +192 -16
  2. investing_algorithm_framework/analysis/__init__.py +16 -0
  3. investing_algorithm_framework/analysis/backtest_data_ranges.py +202 -0
  4. investing_algorithm_framework/analysis/data.py +170 -0
  5. investing_algorithm_framework/analysis/markdown.py +91 -0
  6. investing_algorithm_framework/analysis/ranking.py +298 -0
  7. investing_algorithm_framework/app/__init__.py +29 -4
  8. investing_algorithm_framework/app/algorithm/__init__.py +7 -0
  9. investing_algorithm_framework/app/algorithm/algorithm.py +193 -0
  10. investing_algorithm_framework/app/algorithm/algorithm_factory.py +118 -0
  11. investing_algorithm_framework/app/app.py +2220 -379
  12. investing_algorithm_framework/app/app_hook.py +28 -0
  13. investing_algorithm_framework/app/context.py +1724 -0
  14. investing_algorithm_framework/app/eventloop.py +620 -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/time_metrics_table.py +80 -0
  31. investing_algorithm_framework/app/reporting/tables/trade_metrics_table.py +147 -0
  32. investing_algorithm_framework/app/reporting/tables/trades_table.py +75 -0
  33. investing_algorithm_framework/app/reporting/tables/utils.py +29 -0
  34. investing_algorithm_framework/app/reporting/templates/report_template.html.j2 +154 -0
  35. investing_algorithm_framework/app/stateless/action_handlers/__init__.py +6 -3
  36. investing_algorithm_framework/app/stateless/action_handlers/action_handler_strategy.py +1 -1
  37. investing_algorithm_framework/app/stateless/action_handlers/check_online_handler.py +2 -1
  38. investing_algorithm_framework/app/stateless/action_handlers/run_strategy_handler.py +14 -7
  39. investing_algorithm_framework/app/strategy.py +867 -60
  40. investing_algorithm_framework/app/task.py +5 -3
  41. investing_algorithm_framework/app/web/__init__.py +2 -1
  42. investing_algorithm_framework/app/web/controllers/__init__.py +2 -2
  43. investing_algorithm_framework/app/web/controllers/orders.py +3 -2
  44. investing_algorithm_framework/app/web/controllers/positions.py +2 -2
  45. investing_algorithm_framework/app/web/create_app.py +4 -2
  46. investing_algorithm_framework/app/web/schemas/position.py +1 -0
  47. investing_algorithm_framework/cli/__init__.py +0 -0
  48. investing_algorithm_framework/cli/cli.py +231 -0
  49. investing_algorithm_framework/cli/deploy_to_aws_lambda.py +501 -0
  50. investing_algorithm_framework/cli/deploy_to_azure_function.py +718 -0
  51. investing_algorithm_framework/cli/initialize_app.py +603 -0
  52. investing_algorithm_framework/cli/templates/.gitignore.template +178 -0
  53. investing_algorithm_framework/cli/templates/app.py.template +18 -0
  54. investing_algorithm_framework/cli/templates/app_aws_lambda_function.py.template +48 -0
  55. investing_algorithm_framework/cli/templates/app_azure_function.py.template +14 -0
  56. investing_algorithm_framework/cli/templates/app_web.py.template +18 -0
  57. investing_algorithm_framework/cli/templates/aws_lambda_dockerfile.template +22 -0
  58. investing_algorithm_framework/cli/templates/aws_lambda_dockerignore.template +92 -0
  59. investing_algorithm_framework/cli/templates/aws_lambda_readme.md.template +110 -0
  60. investing_algorithm_framework/cli/templates/aws_lambda_requirements.txt.template +2 -0
  61. investing_algorithm_framework/cli/templates/azure_function_function_app.py.template +65 -0
  62. investing_algorithm_framework/cli/templates/azure_function_host.json.template +15 -0
  63. investing_algorithm_framework/cli/templates/azure_function_local.settings.json.template +8 -0
  64. investing_algorithm_framework/cli/templates/azure_function_requirements.txt.template +3 -0
  65. investing_algorithm_framework/cli/templates/data_providers.py.template +17 -0
  66. investing_algorithm_framework/cli/templates/env.example.template +2 -0
  67. investing_algorithm_framework/cli/templates/env_azure_function.example.template +4 -0
  68. investing_algorithm_framework/cli/templates/market_data_providers.py.template +9 -0
  69. investing_algorithm_framework/cli/templates/readme.md.template +135 -0
  70. investing_algorithm_framework/cli/templates/requirements.txt.template +2 -0
  71. investing_algorithm_framework/cli/templates/run_backtest.py.template +20 -0
  72. investing_algorithm_framework/cli/templates/strategy.py.template +124 -0
  73. investing_algorithm_framework/cli/validate_backtest_checkpoints.py +197 -0
  74. investing_algorithm_framework/create_app.py +40 -7
  75. investing_algorithm_framework/dependency_container.py +100 -47
  76. investing_algorithm_framework/domain/__init__.py +97 -30
  77. investing_algorithm_framework/domain/algorithm_id.py +69 -0
  78. investing_algorithm_framework/domain/backtesting/__init__.py +25 -0
  79. investing_algorithm_framework/domain/backtesting/backtest.py +548 -0
  80. investing_algorithm_framework/domain/backtesting/backtest_date_range.py +113 -0
  81. investing_algorithm_framework/domain/backtesting/backtest_evaluation_focuss.py +241 -0
  82. investing_algorithm_framework/domain/backtesting/backtest_metrics.py +470 -0
  83. investing_algorithm_framework/domain/backtesting/backtest_permutation_test.py +275 -0
  84. investing_algorithm_framework/domain/backtesting/backtest_run.py +663 -0
  85. investing_algorithm_framework/domain/backtesting/backtest_summary_metrics.py +162 -0
  86. investing_algorithm_framework/domain/backtesting/backtest_utils.py +198 -0
  87. investing_algorithm_framework/domain/backtesting/combine_backtests.py +392 -0
  88. investing_algorithm_framework/domain/config.py +59 -136
  89. investing_algorithm_framework/domain/constants.py +18 -37
  90. investing_algorithm_framework/domain/data_provider.py +334 -0
  91. investing_algorithm_framework/domain/data_structures.py +42 -0
  92. investing_algorithm_framework/domain/exceptions.py +51 -1
  93. investing_algorithm_framework/domain/models/__init__.py +26 -19
  94. investing_algorithm_framework/domain/models/app_mode.py +34 -0
  95. investing_algorithm_framework/domain/models/data/__init__.py +7 -0
  96. investing_algorithm_framework/domain/models/data/data_source.py +222 -0
  97. investing_algorithm_framework/domain/models/data/data_type.py +46 -0
  98. investing_algorithm_framework/domain/models/event.py +35 -0
  99. investing_algorithm_framework/domain/models/market/__init__.py +5 -0
  100. investing_algorithm_framework/domain/models/market/market_credential.py +88 -0
  101. investing_algorithm_framework/domain/models/order/__init__.py +3 -4
  102. investing_algorithm_framework/domain/models/order/order.py +198 -65
  103. investing_algorithm_framework/domain/models/order/order_status.py +2 -2
  104. investing_algorithm_framework/domain/models/order/order_type.py +1 -3
  105. investing_algorithm_framework/domain/models/portfolio/__init__.py +6 -2
  106. investing_algorithm_framework/domain/models/portfolio/portfolio.py +98 -3
  107. investing_algorithm_framework/domain/models/portfolio/portfolio_configuration.py +37 -43
  108. investing_algorithm_framework/domain/models/portfolio/portfolio_snapshot.py +108 -11
  109. investing_algorithm_framework/domain/models/position/__init__.py +2 -1
  110. investing_algorithm_framework/domain/models/position/position.py +20 -0
  111. investing_algorithm_framework/domain/models/position/position_size.py +41 -0
  112. investing_algorithm_framework/domain/models/position/position_snapshot.py +0 -2
  113. investing_algorithm_framework/domain/models/risk_rules/__init__.py +7 -0
  114. investing_algorithm_framework/domain/models/risk_rules/stop_loss_rule.py +51 -0
  115. investing_algorithm_framework/domain/models/risk_rules/take_profit_rule.py +55 -0
  116. investing_algorithm_framework/domain/models/snapshot_interval.py +45 -0
  117. investing_algorithm_framework/domain/models/strategy_profile.py +19 -141
  118. investing_algorithm_framework/domain/models/time_frame.py +94 -98
  119. investing_algorithm_framework/domain/models/time_interval.py +33 -0
  120. investing_algorithm_framework/domain/models/time_unit.py +66 -2
  121. investing_algorithm_framework/domain/models/tracing/__init__.py +0 -0
  122. investing_algorithm_framework/domain/models/tracing/trace.py +23 -0
  123. investing_algorithm_framework/domain/models/trade/__init__.py +11 -0
  124. investing_algorithm_framework/domain/models/trade/trade.py +389 -0
  125. investing_algorithm_framework/domain/models/trade/trade_status.py +40 -0
  126. investing_algorithm_framework/domain/models/trade/trade_stop_loss.py +332 -0
  127. investing_algorithm_framework/domain/models/trade/trade_take_profit.py +365 -0
  128. investing_algorithm_framework/domain/order_executor.py +112 -0
  129. investing_algorithm_framework/domain/portfolio_provider.py +118 -0
  130. investing_algorithm_framework/domain/services/__init__.py +11 -0
  131. investing_algorithm_framework/domain/services/market_credential_service.py +37 -0
  132. investing_algorithm_framework/domain/services/portfolios/__init__.py +5 -0
  133. investing_algorithm_framework/domain/services/portfolios/portfolio_sync_service.py +9 -0
  134. investing_algorithm_framework/domain/services/rounding_service.py +27 -0
  135. investing_algorithm_framework/domain/services/state_handler.py +38 -0
  136. investing_algorithm_framework/domain/strategy.py +1 -29
  137. investing_algorithm_framework/domain/utils/__init__.py +15 -5
  138. investing_algorithm_framework/domain/utils/csv.py +22 -0
  139. investing_algorithm_framework/domain/utils/custom_tqdm.py +22 -0
  140. investing_algorithm_framework/domain/utils/dates.py +57 -0
  141. investing_algorithm_framework/domain/utils/jupyter_notebook_detection.py +19 -0
  142. investing_algorithm_framework/domain/utils/polars.py +53 -0
  143. investing_algorithm_framework/domain/utils/random.py +29 -0
  144. investing_algorithm_framework/download_data.py +244 -0
  145. investing_algorithm_framework/infrastructure/__init__.py +37 -11
  146. investing_algorithm_framework/infrastructure/data_providers/__init__.py +36 -0
  147. investing_algorithm_framework/infrastructure/data_providers/ccxt.py +1152 -0
  148. investing_algorithm_framework/infrastructure/data_providers/csv.py +568 -0
  149. investing_algorithm_framework/infrastructure/data_providers/pandas.py +599 -0
  150. investing_algorithm_framework/infrastructure/database/__init__.py +6 -2
  151. investing_algorithm_framework/infrastructure/database/sql_alchemy.py +86 -12
  152. investing_algorithm_framework/infrastructure/models/__init__.py +7 -3
  153. investing_algorithm_framework/infrastructure/models/order/__init__.py +2 -2
  154. investing_algorithm_framework/infrastructure/models/order/order.py +53 -53
  155. investing_algorithm_framework/infrastructure/models/order/order_metadata.py +44 -0
  156. investing_algorithm_framework/infrastructure/models/order_trade_association.py +10 -0
  157. investing_algorithm_framework/infrastructure/models/portfolio/__init__.py +1 -1
  158. investing_algorithm_framework/infrastructure/models/portfolio/portfolio_snapshot.py +8 -2
  159. investing_algorithm_framework/infrastructure/models/portfolio/{portfolio.py → sql_portfolio.py} +17 -6
  160. investing_algorithm_framework/infrastructure/models/position/position_snapshot.py +3 -1
  161. investing_algorithm_framework/infrastructure/models/trades/__init__.py +9 -0
  162. investing_algorithm_framework/infrastructure/models/trades/trade.py +130 -0
  163. investing_algorithm_framework/infrastructure/models/trades/trade_stop_loss.py +59 -0
  164. investing_algorithm_framework/infrastructure/models/trades/trade_take_profit.py +55 -0
  165. investing_algorithm_framework/infrastructure/order_executors/__init__.py +21 -0
  166. investing_algorithm_framework/infrastructure/order_executors/backtest_oder_executor.py +28 -0
  167. investing_algorithm_framework/infrastructure/order_executors/ccxt_order_executor.py +200 -0
  168. investing_algorithm_framework/infrastructure/portfolio_providers/__init__.py +19 -0
  169. investing_algorithm_framework/infrastructure/portfolio_providers/ccxt_portfolio_provider.py +199 -0
  170. investing_algorithm_framework/infrastructure/repositories/__init__.py +10 -4
  171. investing_algorithm_framework/infrastructure/repositories/order_metadata_repository.py +17 -0
  172. investing_algorithm_framework/infrastructure/repositories/order_repository.py +16 -5
  173. investing_algorithm_framework/infrastructure/repositories/portfolio_repository.py +2 -2
  174. investing_algorithm_framework/infrastructure/repositories/position_repository.py +11 -0
  175. investing_algorithm_framework/infrastructure/repositories/repository.py +84 -30
  176. investing_algorithm_framework/infrastructure/repositories/trade_repository.py +71 -0
  177. investing_algorithm_framework/infrastructure/repositories/trade_stop_loss_repository.py +29 -0
  178. investing_algorithm_framework/infrastructure/repositories/trade_take_profit_repository.py +29 -0
  179. investing_algorithm_framework/infrastructure/services/__init__.py +9 -4
  180. investing_algorithm_framework/infrastructure/services/aws/__init__.py +6 -0
  181. investing_algorithm_framework/infrastructure/services/aws/state_handler.py +193 -0
  182. investing_algorithm_framework/infrastructure/services/azure/__init__.py +5 -0
  183. investing_algorithm_framework/infrastructure/services/azure/state_handler.py +158 -0
  184. investing_algorithm_framework/infrastructure/services/backtesting/__init__.py +9 -0
  185. investing_algorithm_framework/infrastructure/services/backtesting/backtest_service.py +2596 -0
  186. investing_algorithm_framework/infrastructure/services/backtesting/event_backtest_service.py +285 -0
  187. investing_algorithm_framework/infrastructure/services/backtesting/vector_backtest_service.py +468 -0
  188. investing_algorithm_framework/services/__init__.py +123 -15
  189. investing_algorithm_framework/services/configuration_service.py +77 -11
  190. investing_algorithm_framework/services/data_providers/__init__.py +5 -0
  191. investing_algorithm_framework/services/data_providers/data_provider_service.py +1058 -0
  192. investing_algorithm_framework/services/market_credential_service.py +40 -0
  193. investing_algorithm_framework/services/metrics/__init__.py +119 -0
  194. investing_algorithm_framework/services/metrics/alpha.py +0 -0
  195. investing_algorithm_framework/services/metrics/beta.py +0 -0
  196. investing_algorithm_framework/services/metrics/cagr.py +60 -0
  197. investing_algorithm_framework/services/metrics/calmar_ratio.py +40 -0
  198. investing_algorithm_framework/services/metrics/drawdown.py +218 -0
  199. investing_algorithm_framework/services/metrics/equity_curve.py +24 -0
  200. investing_algorithm_framework/services/metrics/exposure.py +210 -0
  201. investing_algorithm_framework/services/metrics/generate.py +358 -0
  202. investing_algorithm_framework/services/metrics/mean_daily_return.py +84 -0
  203. investing_algorithm_framework/services/metrics/price_efficiency.py +57 -0
  204. investing_algorithm_framework/services/metrics/profit_factor.py +165 -0
  205. investing_algorithm_framework/services/metrics/recovery.py +113 -0
  206. investing_algorithm_framework/services/metrics/returns.py +452 -0
  207. investing_algorithm_framework/services/metrics/risk_free_rate.py +28 -0
  208. investing_algorithm_framework/services/metrics/sharpe_ratio.py +137 -0
  209. investing_algorithm_framework/services/metrics/sortino_ratio.py +74 -0
  210. investing_algorithm_framework/services/metrics/standard_deviation.py +156 -0
  211. investing_algorithm_framework/services/metrics/trades.py +473 -0
  212. investing_algorithm_framework/services/metrics/treynor_ratio.py +0 -0
  213. investing_algorithm_framework/services/metrics/ulcer.py +0 -0
  214. investing_algorithm_framework/services/metrics/value_at_risk.py +0 -0
  215. investing_algorithm_framework/services/metrics/volatility.py +118 -0
  216. investing_algorithm_framework/services/metrics/win_rate.py +177 -0
  217. investing_algorithm_framework/services/order_service/__init__.py +9 -0
  218. investing_algorithm_framework/services/order_service/order_backtest_service.py +178 -0
  219. investing_algorithm_framework/services/order_service/order_executor_lookup.py +110 -0
  220. investing_algorithm_framework/services/order_service/order_service.py +826 -0
  221. investing_algorithm_framework/services/portfolios/__init__.py +16 -0
  222. investing_algorithm_framework/services/portfolios/backtest_portfolio_service.py +54 -0
  223. investing_algorithm_framework/services/{portfolio_configuration_service.py → portfolios/portfolio_configuration_service.py} +27 -12
  224. investing_algorithm_framework/services/portfolios/portfolio_provider_lookup.py +106 -0
  225. investing_algorithm_framework/services/portfolios/portfolio_service.py +188 -0
  226. investing_algorithm_framework/services/portfolios/portfolio_snapshot_service.py +136 -0
  227. investing_algorithm_framework/services/portfolios/portfolio_sync_service.py +182 -0
  228. investing_algorithm_framework/services/positions/__init__.py +7 -0
  229. investing_algorithm_framework/services/positions/position_service.py +210 -0
  230. investing_algorithm_framework/services/repository_service.py +8 -2
  231. investing_algorithm_framework/services/trade_order_evaluator/__init__.py +9 -0
  232. investing_algorithm_framework/services/trade_order_evaluator/backtest_trade_oder_evaluator.py +117 -0
  233. investing_algorithm_framework/services/trade_order_evaluator/default_trade_order_evaluator.py +51 -0
  234. investing_algorithm_framework/services/trade_order_evaluator/trade_order_evaluator.py +80 -0
  235. investing_algorithm_framework/services/trade_service/__init__.py +9 -0
  236. investing_algorithm_framework/services/trade_service/trade_service.py +1099 -0
  237. investing_algorithm_framework/services/trade_service/trade_stop_loss_service.py +39 -0
  238. investing_algorithm_framework/services/trade_service/trade_take_profit_service.py +41 -0
  239. investing_algorithm_framework-7.25.6.dist-info/METADATA +535 -0
  240. investing_algorithm_framework-7.25.6.dist-info/RECORD +268 -0
  241. {investing_algorithm_framework-1.5.dist-info → investing_algorithm_framework-7.25.6.dist-info}/WHEEL +1 -2
  242. investing_algorithm_framework-7.25.6.dist-info/entry_points.txt +3 -0
  243. investing_algorithm_framework/app/algorithm.py +0 -630
  244. investing_algorithm_framework/domain/models/backtest_profile.py +0 -414
  245. investing_algorithm_framework/domain/models/market_data/__init__.py +0 -11
  246. investing_algorithm_framework/domain/models/market_data/asset_price.py +0 -50
  247. investing_algorithm_framework/domain/models/market_data/ohlcv.py +0 -105
  248. investing_algorithm_framework/domain/models/market_data/order_book.py +0 -63
  249. investing_algorithm_framework/domain/models/market_data/ticker.py +0 -92
  250. investing_algorithm_framework/domain/models/order/order_fee.py +0 -45
  251. investing_algorithm_framework/domain/models/trade.py +0 -78
  252. investing_algorithm_framework/domain/models/trading_data_types.py +0 -47
  253. investing_algorithm_framework/domain/models/trading_time_frame.py +0 -223
  254. investing_algorithm_framework/domain/singleton.py +0 -9
  255. investing_algorithm_framework/domain/utils/backtesting.py +0 -82
  256. investing_algorithm_framework/infrastructure/models/order/order_fee.py +0 -21
  257. investing_algorithm_framework/infrastructure/repositories/order_fee_repository.py +0 -15
  258. investing_algorithm_framework/infrastructure/services/market_backtest_service.py +0 -360
  259. investing_algorithm_framework/infrastructure/services/market_service.py +0 -410
  260. investing_algorithm_framework/infrastructure/services/performance_service.py +0 -192
  261. investing_algorithm_framework/services/backtest_service.py +0 -268
  262. investing_algorithm_framework/services/market_data_service.py +0 -77
  263. investing_algorithm_framework/services/order_backtest_service.py +0 -122
  264. investing_algorithm_framework/services/order_service.py +0 -752
  265. investing_algorithm_framework/services/portfolio_service.py +0 -164
  266. investing_algorithm_framework/services/portfolio_snapshot_service.py +0 -68
  267. investing_algorithm_framework/services/position_cost_service.py +0 -5
  268. investing_algorithm_framework/services/position_service.py +0 -63
  269. investing_algorithm_framework/services/strategy_orchestrator_service.py +0 -225
  270. investing_algorithm_framework-1.5.dist-info/AUTHORS.md +0 -8
  271. investing_algorithm_framework-1.5.dist-info/METADATA +0 -230
  272. investing_algorithm_framework-1.5.dist-info/RECORD +0 -119
  273. investing_algorithm_framework-1.5.dist-info/top_level.txt +0 -1
  274. /investing_algorithm_framework/{infrastructure/services/performance_backtest_service.py → app/reporting/tables/stop_loss_table.py} +0 -0
  275. /investing_algorithm_framework/services/{position_snapshot_service.py → positions/position_snapshot_service.py} +0 -0
  276. {investing_algorithm_framework-1.5.dist-info → investing_algorithm_framework-7.25.6.dist-info}/LICENSE +0 -0
@@ -0,0 +1,603 @@
1
+ import os
2
+ from enum import Enum
3
+
4
+
5
+ class AppType(Enum):
6
+ DEFAULT = "DEFAULT"
7
+ DEFAULT_WEB = "DEFAULT_WEB"
8
+ AZURE_FUNCTION = "AZURE_FUNCTION"
9
+ AWS_LAMBDA = "AWS_LAMBDA"
10
+
11
+ @staticmethod
12
+ def from_string(value: str):
13
+
14
+ if isinstance(value, str):
15
+ for app_type in AppType:
16
+
17
+ if value.upper() == app_type.value:
18
+ return app_type
19
+
20
+ raise ValueError("Could not convert value to AppType")
21
+
22
+ @staticmethod
23
+ def from_value(value):
24
+
25
+ if isinstance(value, AppType):
26
+ for app_type in AppType:
27
+
28
+ if value == app_type:
29
+ return app_type
30
+
31
+ elif isinstance(value, str):
32
+ return AppType.from_string(value)
33
+
34
+ raise ValueError(f"Could not convert value {value} to AppType")
35
+
36
+ def equals(self, other):
37
+ return AppType.from_value(other) == self
38
+
39
+
40
+ def create_directory(directory_path):
41
+ """
42
+ Creates a new directory.
43
+
44
+ Args:
45
+ directory_path (str): The path to the directory to create.
46
+
47
+ Returns:
48
+ None
49
+ """
50
+
51
+ if not os.path.exists(directory_path):
52
+ os.makedirs(directory_path)
53
+
54
+
55
+ def create_file(file_path, replace=False):
56
+ """
57
+ Creates a new file.
58
+
59
+ Args:
60
+ file_path (str): The path to the file to create.
61
+
62
+ Returns:
63
+ None
64
+ """
65
+
66
+ # Check if file already exists
67
+ if os.path.exists(file_path):
68
+ if replace:
69
+ os.remove(file_path)
70
+
71
+ if not os.path.exists(file_path):
72
+ with open(file_path, "w") as file:
73
+ file.write("")
74
+
75
+
76
+ def create_file_from_template(template_path, output_path, replace=False):
77
+ """
78
+ Creates a new file by replacing placeholders in a template file.
79
+
80
+ Args:
81
+ template_path (str): The path to the template file.
82
+ output_path (str): The path to the output file.
83
+ replacements (dict): A dictionary of placeholder keys and
84
+ their replacements.
85
+ replace (bool): If True, the template file will be replaced
86
+
87
+ Returns:
88
+ None
89
+ """
90
+ if replace:
91
+ if os.path.exists(output_path):
92
+ os.remove(output_path)
93
+
94
+ # Check if output path already exists
95
+ if not os.path.exists(output_path):
96
+ with open(template_path, "r") as file:
97
+ template = file.read()
98
+
99
+ with open(output_path, "w") as file:
100
+ file.write(template)
101
+
102
+
103
+ def command(path=None, app_type="default", replace=False):
104
+ """
105
+ Function to create an azure function app skeleton.
106
+
107
+ Args:
108
+ path (str): Path to directory to initialize the app in.
109
+ app_type (str): Type of app to create. Options are: 'default',
110
+ 'default-web', 'azure-function', 'aws-lambda'.
111
+ replace (bool): If True, existing files will be replaced.
112
+ If False, existing files will not be replaced.
113
+ Default is False.
114
+
115
+ Returns:
116
+ None
117
+ """
118
+
119
+ if path is None:
120
+ path = os.getcwd()
121
+ else:
122
+ # check if directory exists
123
+ if not os.path.exists(path) or not os.path.isdir(path):
124
+ return
125
+
126
+ if AppType.DEFAULT.equals(app_type):
127
+ create_default_app(path=path, replace=replace)
128
+ elif AppType.DEFAULT_WEB.equals(app_type):
129
+ create_default_web_app(path=path, replace=replace)
130
+ elif AppType.AZURE_FUNCTION.equals(app_type):
131
+ create_azure_function_app(path=path, replace=replace)
132
+ elif AppType.AWS_LAMBDA.equals(app_type):
133
+ create_aws_lambda_app(path=path, replace=replace)
134
+ else:
135
+ raise ValueError(
136
+ f"Unknown app type: {app_type}. "
137
+ "Supported types are: 'default', 'default-web', "
138
+ "'azure-function', 'aws-lambda'."
139
+ )
140
+
141
+
142
+ def create_default_app(path=None, replace=False):
143
+ """
144
+ Function to create a default app skeleton.
145
+
146
+ Args:
147
+ path (str): Path to directory to initialize the app in.
148
+ replace (bool): If True, existing files will be replaced.
149
+ If False, existing files will not be replaced.
150
+ Default is False.
151
+
152
+ Returns:
153
+ None
154
+ """
155
+ # Get the path of this script (command.py)
156
+ current_script_path = os.path.abspath(__file__)
157
+
158
+ # Construct the path to the template file
159
+ template_app_file_path = os.path.join(
160
+ os.path.dirname(current_script_path),
161
+ "templates",
162
+ "app.py.template"
163
+ )
164
+ requirements_path = os.path.join(
165
+ os.path.dirname(current_script_path),
166
+ "templates",
167
+ "requirements.txt.template"
168
+ )
169
+ run_backtest_template_path = os.path.join(
170
+ os.path.dirname(current_script_path),
171
+ "templates",
172
+ "run_backtest.py.template"
173
+ )
174
+ env_template_path = os.path.join(
175
+ os.path.dirname(current_script_path),
176
+ "templates",
177
+ "env.example.template"
178
+ )
179
+ create_file_from_template(
180
+ env_template_path,
181
+ os.path.join(path, ".env.example"),
182
+ replace=replace
183
+ )
184
+ strategy_template_path = os.path.join(
185
+ os.path.dirname(current_script_path),
186
+ "templates",
187
+ "strategy.py.template"
188
+ )
189
+ data_providers_template_path = os.path.join(
190
+ os.path.dirname(current_script_path),
191
+ "templates",
192
+ "data_providers.py.template"
193
+ )
194
+ create_file(os.path.join(path, "__init__.py"))
195
+ create_file_from_template(
196
+ template_app_file_path,
197
+ os.path.join(path, "app.py"),
198
+ replace=replace
199
+ )
200
+ create_file_from_template(
201
+ requirements_path,
202
+ os.path.join(path, "requirements.txt"),
203
+ replace=replace
204
+ )
205
+ create_file_from_template(
206
+ run_backtest_template_path,
207
+ os.path.join(path, "run_backtest.py"),
208
+ replace=replace
209
+ )
210
+ # Create the main directory
211
+ create_directory(os.path.join(path, "strategies"))
212
+ strategies_path = os.path.join(path, "strategies")
213
+ create_file(os.path.join(strategies_path, "__init__.py"))
214
+ create_file_from_template(
215
+ strategy_template_path,
216
+ os.path.join(strategies_path, "strategy.py")
217
+ )
218
+ create_file_from_template(
219
+ data_providers_template_path,
220
+ os.path.join(strategies_path, "data_providers.py"),
221
+ replace=replace
222
+ )
223
+ gitignore_template_path = os.path.join(
224
+ os.path.dirname(current_script_path),
225
+ "templates",
226
+ ".gitignore.template"
227
+ )
228
+ create_file_from_template(
229
+ gitignore_template_path,
230
+ os.path.join(path, ".gitignore"),
231
+ replace=replace
232
+ )
233
+ readme_template_path = os.path.join(
234
+ os.path.dirname(current_script_path),
235
+ "templates",
236
+ "readme.md.template"
237
+ )
238
+ create_file_from_template(
239
+ readme_template_path,
240
+ os.path.join(path, "README.md"),
241
+ replace=replace
242
+ )
243
+
244
+
245
+ def create_default_web_app(path=None, replace=False):
246
+ """
247
+ Function to create a default web app skeleton.
248
+
249
+ Args:
250
+ path (str): Path to directory to initialize the app in.
251
+ replace (bool): If True, existing files will be replaced.
252
+ If False, existing files will not be replaced.
253
+ Default is False.
254
+
255
+ Returns:
256
+ None
257
+ """
258
+ # Get the path of this script (command.py)
259
+ current_script_path = os.path.abspath(__file__)
260
+
261
+ # Construct the path to the template file
262
+ template_app_file_path = os.path.join(
263
+ os.path.dirname(current_script_path),
264
+ "templates",
265
+ "app_web.py.template"
266
+ )
267
+ requirements_path = os.path.join(
268
+ os.path.dirname(current_script_path),
269
+ "templates",
270
+ "requirements.txt.template"
271
+ )
272
+ run_backtest_template_path = os.path.join(
273
+ os.path.dirname(current_script_path),
274
+ "templates",
275
+ "run_backtest.py.template"
276
+ )
277
+ env_template_path = os.path.join(
278
+ os.path.dirname(current_script_path),
279
+ "templates",
280
+ "env.example.template"
281
+ )
282
+ create_file_from_template(
283
+ env_template_path,
284
+ os.path.join(path, ".env.example"),
285
+ replace=replace
286
+ )
287
+ strategy_template_path = os.path.join(
288
+ os.path.dirname(current_script_path),
289
+ "templates",
290
+ "strategy.py.template"
291
+ )
292
+
293
+ data_providers_template_path = os.path.join(
294
+ os.path.dirname(current_script_path),
295
+ "templates",
296
+ "data_providers.py.template"
297
+ )
298
+ create_file(os.path.join(path, "__init__.py"))
299
+ create_file_from_template(
300
+ template_app_file_path,
301
+ os.path.join(path, "app.py"),
302
+ replace=replace
303
+ )
304
+ create_file_from_template(
305
+ requirements_path,
306
+ os.path.join(path, "requirements.txt"),
307
+ replace=replace
308
+ )
309
+ create_file_from_template(
310
+ run_backtest_template_path,
311
+ os.path.join(path, "run_backtest.py"),
312
+ replace=replace
313
+ )
314
+ # Create the main directory
315
+ create_directory(os.path.join(path, "strategies"))
316
+ strategies_path = os.path.join(path, "strategies")
317
+ create_file(os.path.join(strategies_path, "__init__.py"))
318
+ create_file_from_template(
319
+ strategy_template_path,
320
+ os.path.join(strategies_path, "strategy.py")
321
+ )
322
+ create_file_from_template(
323
+ data_providers_template_path,
324
+ os.path.join(strategies_path, "data_providers.py"),
325
+ replace=replace
326
+ )
327
+ gitignore_template_path = os.path.join(
328
+ os.path.dirname(current_script_path),
329
+ "templates",
330
+ ".gitignore.template"
331
+ )
332
+ create_file_from_template(
333
+ gitignore_template_path,
334
+ os.path.join(path, ".gitignore"),
335
+ replace=replace
336
+ )
337
+ readme_template_path = os.path.join(
338
+ os.path.dirname(current_script_path),
339
+ "templates",
340
+ "readme.md.template"
341
+ )
342
+ create_file_from_template(
343
+ readme_template_path,
344
+ os.path.join(path, "README.md"),
345
+ replace=replace
346
+ )
347
+
348
+
349
+ def create_aws_lambda_app(path=None, replace=False):
350
+ """
351
+ Function to create an AWS Lambda app skeleton.
352
+
353
+ Args:
354
+ path (str): Path to directory to initialize the app in.
355
+ replace (bool): If True, existing files will be replaced.
356
+ If False, existing files will not be replaced.
357
+ Default is False.
358
+
359
+ Returns:
360
+ None
361
+ """
362
+ # Get the path of this script (command.py)
363
+ current_script_path = os.path.abspath(__file__)
364
+ requirements_path = os.path.join(
365
+ os.path.dirname(current_script_path),
366
+ "templates",
367
+ "requirements.txt.template"
368
+ )
369
+ aws_lambda_handler_template_path = os.path.join(
370
+ os.path.dirname(current_script_path),
371
+ "templates",
372
+ "app_aws_lambda_function.py.template"
373
+ )
374
+ aws_dockerfile_template_path = os.path.join(
375
+ os.path.dirname(current_script_path),
376
+ "templates",
377
+ "aws_lambda_dockerfile.template"
378
+ )
379
+ aws_dockerignore_template_path = os.path.join(
380
+ os.path.dirname(current_script_path),
381
+ "templates",
382
+ "aws_lambda_dockerignore.template"
383
+ )
384
+ run_backtest_template_path = os.path.join(
385
+ os.path.dirname(current_script_path),
386
+ "templates",
387
+ "run_backtest.py.template"
388
+ )
389
+ env_template_path = os.path.join(
390
+ os.path.dirname(current_script_path),
391
+ "templates",
392
+ "env.example.template"
393
+ )
394
+ create_file_from_template(
395
+ env_template_path,
396
+ os.path.join(path, ".env"),
397
+ replace=replace
398
+ )
399
+ strategy_template_path = os.path.join(
400
+ os.path.dirname(current_script_path),
401
+ "templates",
402
+ "strategy.py.template"
403
+ )
404
+ data_providers_template_path = os.path.join(
405
+ os.path.dirname(current_script_path),
406
+ "templates",
407
+ "data_providers.py.template"
408
+ )
409
+ create_file(os.path.join(path, "__init__.py"))
410
+ create_file_from_template(
411
+ requirements_path,
412
+ os.path.join(path, "requirements.txt"),
413
+ replace=replace
414
+ )
415
+ create_file_from_template(
416
+ run_backtest_template_path,
417
+ os.path.join(path, "run_backtest.py"),
418
+ replace=replace
419
+ )
420
+ # Create the main directory
421
+ create_directory(os.path.join(path, "strategies"))
422
+ strategies_path = os.path.join(path, "strategies")
423
+ create_file(os.path.join(strategies_path, "__init__.py"))
424
+ create_file_from_template(
425
+ strategy_template_path,
426
+ os.path.join(strategies_path, "strategy.py")
427
+ )
428
+ create_file_from_template(
429
+ data_providers_template_path,
430
+ os.path.join(strategies_path, "data_providers.py"),
431
+ replace=replace
432
+ )
433
+ gitignore_template_path = os.path.join(
434
+ os.path.dirname(current_script_path),
435
+ "templates",
436
+ ".gitignore.template"
437
+ )
438
+ create_file_from_template(
439
+ gitignore_template_path,
440
+ os.path.join(path, ".gitignore"),
441
+ replace=replace
442
+ )
443
+ readme_template_path = os.path.join(
444
+ os.path.dirname(current_script_path),
445
+ "templates",
446
+ "readme.md.template"
447
+ )
448
+ create_file_from_template(
449
+ readme_template_path,
450
+ os.path.join(path, "README.md"),
451
+ replace=replace
452
+ )
453
+ create_file_from_template(
454
+ aws_lambda_handler_template_path,
455
+ os.path.join(path, "aws_function.py"),
456
+ replace=replace
457
+ )
458
+ create_file_from_template(
459
+ aws_dockerfile_template_path,
460
+ os.path.join(path, "Dockerfile"),
461
+ replace=replace
462
+ )
463
+ create_file_from_template(
464
+ aws_dockerignore_template_path,
465
+ os.path.join(path, ".dockerignore"),
466
+ replace=replace
467
+ )
468
+
469
+
470
+ def create_azure_function_app(path=None, replace=False):
471
+ """
472
+ Function to create an Azure Function app skeleton.
473
+
474
+ Args:
475
+ path (str): Path to directory to initialize the app in.
476
+ replace (bool): If True, existing files will be replaced.
477
+ If False, existing files will not be replaced.
478
+ Default is False.
479
+
480
+ Returns:
481
+ None
482
+ """
483
+ # Get the path of this script (command.py)
484
+ current_script_path = os.path.abspath(__file__)
485
+
486
+ # Construct the path to the template file
487
+ template_app_file_path = os.path.join(
488
+ os.path.dirname(current_script_path),
489
+ "templates",
490
+ "app_azure_function.py.template"
491
+ )
492
+ requirements_path = os.path.join(
493
+ os.path.dirname(current_script_path),
494
+ "templates",
495
+ "azure_function_requirements.txt.template"
496
+ )
497
+ azure_function_template_path = os.path.join(
498
+ os.path.dirname(current_script_path),
499
+ "templates",
500
+ "azure_function_function_app.py.template"
501
+ )
502
+ run_backtest_template_path = os.path.join(
503
+ os.path.dirname(current_script_path),
504
+ "templates",
505
+ "run_backtest.py.template"
506
+ )
507
+ # Create the framework app file as app_entry.py
508
+ create_file_from_template(
509
+ azure_function_template_path,
510
+ os.path.join(path, "app_web.py"),
511
+ replace=replace
512
+ )
513
+ # Create the host.json file
514
+ host_json_template_path = os.path.join(
515
+ os.path.dirname(current_script_path),
516
+ "templates",
517
+ "azure_function_host.json.template"
518
+ )
519
+ create_file_from_template(
520
+ host_json_template_path,
521
+ os.path.join(path, "host.json"),
522
+ replace=replace
523
+ )
524
+ # Create the local.settings.json file
525
+ local_settings_json_template_path = os.path.join(
526
+ os.path.dirname(current_script_path),
527
+ "templates",
528
+ "azure_function_local.settings.json.template"
529
+ )
530
+ create_file_from_template(
531
+ local_settings_json_template_path,
532
+ os.path.join(path, "local.settings.json"),
533
+ replace=replace
534
+ )
535
+ env_template_path = os.path.join(
536
+ os.path.dirname(current_script_path),
537
+ "templates",
538
+ "env_azure_function.example.template"
539
+ )
540
+ create_file_from_template(
541
+ env_template_path,
542
+ os.path.join(path, ".env.example"),
543
+ replace=replace
544
+ )
545
+ strategy_template_path = os.path.join(
546
+ os.path.dirname(current_script_path),
547
+ "templates",
548
+ "strategy.py.template"
549
+ )
550
+ data_providers_template_path = os.path.join(
551
+ os.path.dirname(current_script_path),
552
+ "templates",
553
+ "data_providers.py.template"
554
+ )
555
+ create_file(os.path.join(path, "__init__.py"))
556
+ create_file_from_template(
557
+ template_app_file_path,
558
+ os.path.join(path, "app.py"),
559
+ replace=replace
560
+ )
561
+ create_file_from_template(
562
+ requirements_path,
563
+ os.path.join(path, "requirements.txt"),
564
+ replace=replace
565
+ )
566
+ create_file_from_template(
567
+ run_backtest_template_path,
568
+ os.path.join(path, "run_backtest.py"),
569
+ replace=replace
570
+ )
571
+ # Create the main directory
572
+ create_directory(os.path.join(path, "strategies"))
573
+ strategies_path = os.path.join(path, "strategies")
574
+ create_file(os.path.join(strategies_path, "__init__.py"))
575
+ create_file_from_template(
576
+ strategy_template_path,
577
+ os.path.join(strategies_path, "strategy.py")
578
+ )
579
+ create_file_from_template(
580
+ data_providers_template_path,
581
+ os.path.join(strategies_path, "data_providers.py"),
582
+ replace=replace
583
+ )
584
+ gitignore_template_path = os.path.join(
585
+ os.path.dirname(current_script_path),
586
+ "templates",
587
+ ".gitignore.template"
588
+ )
589
+ create_file_from_template(
590
+ gitignore_template_path,
591
+ os.path.join(path, ".gitignore"),
592
+ replace=replace
593
+ )
594
+ readme_template_path = os.path.join(
595
+ os.path.dirname(current_script_path),
596
+ "templates",
597
+ "readme.md.template"
598
+ )
599
+ create_file_from_template(
600
+ readme_template_path,
601
+ os.path.join(path, "README.md"),
602
+ replace=replace
603
+ )