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,39 @@
1
+ import logging
2
+ from datetime import datetime
3
+
4
+ from investing_algorithm_framework.services.repository_service import \
5
+ RepositoryService
6
+
7
+ logger = logging.getLogger(__name__)
8
+
9
+
10
+ class TradeStopLossService(RepositoryService):
11
+
12
+ def mark_triggered(
13
+ self,
14
+ stop_loss_ids,
15
+ trigger_date: datetime
16
+ ) -> None:
17
+ """
18
+ Mark stop losses as triggered.
19
+
20
+ Args:
21
+ stop_loss_ids (List[str]): List of stop loss IDs to
22
+ mark as triggered.
23
+ trigger_date (datetime): The date when the stop loss
24
+ was triggered.
25
+
26
+ Returns:
27
+ None
28
+ """
29
+ update_data = {
30
+ "triggered": True,
31
+ "triggered_at": trigger_date,
32
+ "updated_at": trigger_date
33
+ }
34
+
35
+ for id in stop_loss_ids:
36
+ try:
37
+ self.update(id, update_data)
38
+ except Exception as e:
39
+ logger.error(f"Error marking stop loss {id} as triggered: {e}")
@@ -0,0 +1,41 @@
1
+ import logging
2
+ from datetime import datetime
3
+
4
+ from investing_algorithm_framework.services.repository_service import \
5
+ RepositoryService
6
+
7
+ logger = logging.getLogger(__name__)
8
+
9
+
10
+ class TradeTakeProfitService(RepositoryService):
11
+
12
+ def mark_triggered(
13
+ self,
14
+ take_profit_ids,
15
+ trigger_date: datetime
16
+ ) -> None:
17
+ """
18
+ Mark take profits as triggered.
19
+
20
+ Args:
21
+ take_profit_ids (List[str]): List of take profit IDs to
22
+ mark as triggered.
23
+ trigger_date (datetime): The date and time when the
24
+ take profits were triggered.
25
+
26
+ Returns:
27
+ None
28
+ """
29
+ update_data = {
30
+ "triggered": True,
31
+ "triggered_at": trigger_date,
32
+ "updated_at": trigger_date
33
+ }
34
+
35
+ for id in take_profit_ids:
36
+ try:
37
+ self.update(id, update_data)
38
+ except Exception as e:
39
+ logger.error(
40
+ f"Error marking take profit {id} as triggered: {e}"
41
+ )
@@ -0,0 +1,535 @@
1
+ Metadata-Version: 2.1
2
+ Name: investing-algorithm-framework
3
+ Version: 7.25.6
4
+ Summary: A framework for creating trading bots
5
+ Author: MDUYN
6
+ Requires-Python: >=3.10
7
+ Classifier: Programming Language :: Python :: 3
8
+ Classifier: Programming Language :: Python :: 3.10
9
+ Classifier: Programming Language :: Python :: 3.11
10
+ Classifier: Programming Language :: Python :: 3.12
11
+ Requires-Dist: Flask (>=3.1.0)
12
+ Requires-Dist: Flask-Cors (>=3.0.9,<5.0.0)
13
+ Requires-Dist: Flask-Migrate (>=2.6.0)
14
+ Requires-Dist: SQLAlchemy (>=2.0.18)
15
+ Requires-Dist: azure-identity (>=1.19.0,<2.0.0)
16
+ Requires-Dist: azure-mgmt-resource (>=23.2.0,<24.0.0)
17
+ Requires-Dist: azure-mgmt-storage (>=21.2.1,<22.0.0)
18
+ Requires-Dist: azure-mgmt-web (>=7.3.1,<8.0.0)
19
+ Requires-Dist: azure-storage-blob (>=12.24.0,<13.0.0)
20
+ Requires-Dist: boto3 (>=1.38.41,<2.0.0)
21
+ Requires-Dist: ccxt (>=4.2.48)
22
+ Requires-Dist: dependency-injector (>=4.40.0)
23
+ Requires-Dist: jupyter (>=1.0.0)
24
+ Requires-Dist: marshmallow (>=3.5.0)
25
+ Requires-Dist: plotly (>=6.1.2,<7.0.0)
26
+ Requires-Dist: polars[numpy,pandas] (>=0.20.10)
27
+ Requires-Dist: python-dateutil (>=2.8.2)
28
+ Requires-Dist: python-dotenv (>=1.0.1,<2.0.0)
29
+ Requires-Dist: schedule (>=1.1.0)
30
+ Requires-Dist: tabulate (>=0.9.0)
31
+ Requires-Dist: tqdm (>=4.66.1)
32
+ Requires-Dist: wrapt (>=1.16.0)
33
+ Requires-Dist: yfinance (>=0.2.61,<0.3.0)
34
+ Description-Content-Type: text/markdown
35
+
36
+ <div align="center">
37
+ <h1>โšก Investing Algorithm Framework</h1>
38
+
39
+ <p style="font-size: 18px; font-weight: 600; margin: 15px 0;">
40
+ ๐Ÿš€ <b>Build. Backtest. Deploy.</b> Quantitative Trading Strategies at Scale
41
+ </p>
42
+
43
+ <p style="font-size: 14px; color: #666; margin-bottom: 25px;">
44
+ The fastest way to go from trading idea to production-ready trading bot
45
+ </p>
46
+
47
+ <!-- Quick Links -->
48
+ <div style="margin: 20px 0;">
49
+ <a target="_blank" href="https://coding-kitties.github.io/investing-algorithm-framework/">
50
+ <img src="https://img.shields.io/badge/๐Ÿ“–_Documentation-blue?style=for-the-badge">
51
+ </a>
52
+ &nbsp;
53
+ <a href="https://coding-kitties.github.io/investing-algorithm-framework/Getting%20Started/installation">
54
+ <img src="https://img.shields.io/badge/๐Ÿš€_Quick_Start-green?style=for-the-badge">
55
+ </a>
56
+ </div>
57
+
58
+ <!-- Badges -->
59
+ <div style="margin-bottom: 20px;">
60
+ <a target="_blank" href="https://discord.gg/dQsRmGZP"><img src="https://img.shields.io/discord/1345358169777635410.svg?color=7289da&label=Discord&logo=discord&style=flat-square" alt="Discord"></a>
61
+ &nbsp;
62
+ <a href="https://github.com/coding-kitties/investing-algorithm-framework/actions/workflows/test.yml"><img src="https://github.com/coding-kitties/investing-algorithm-framework/actions/workflows/test.yml/badge.svg?style=flat-square" alt="Tests"></a>
63
+ &nbsp;
64
+ <a href="https://pypi.org/project/investing-algorithm-framework/"><img src="https://img.shields.io/pypi/v/investing-algorithm-framework.svg?style=flat-square" alt="PyPI"></a>
65
+ &nbsp;
66
+ <a href="https://pepy.tech/project/investing-algorithm-framework"><img src="https://pepy.tech/badge/investing-algorithm-framework/month?style=flat-square" alt="Downloads"></a>
67
+ &nbsp;
68
+ <a href="https://github.com/coding-kitties/investing-algorithm-framework/stargazers"><img src="https://img.shields.io/github/stars/coding-kitties/investing-algorithm-framework?style=flat-square" alt="Stars"></a>
69
+ </div>
70
+
71
+ <img src="static/showcase.svg" alt="Investing Algorithm Framework" style="height: 400px; max-width: 100%; margin: 30px 0;">
72
+
73
+ <hr style="margin: 30px 0; border: none; border-top: 2px solid #ddd;">
74
+
75
+ > โญ **If you like this project, please consider [starring](https://github.com/coding-kitties/investing-algorithm-framework) it!** Your support helps us build better tools for the community.
76
+
77
+ </div>
78
+
79
+ ---
80
+
81
+ ## ๐Ÿ’ก Why Investing Algorithm Framework?
82
+
83
+ Stop wasting time on boilerplate. The **Investing Algorithm Framework** handles all the heavy lifting:
84
+
85
+ โœจ **From Idea to Production** โ€” Write your strategy once, deploy everywhere
86
+ ๐Ÿ“Š **Accurate Backtesting** โ€” Event-driven and vectorized engines for realistic results
87
+ โšก **Lightning Fast** โ€” Optimized for speed and efficiency
88
+ ๐Ÿ”ง **Extensible** โ€” Connect any exchange, broker, or data source
89
+ ๐Ÿ“ˆ **Production Ready** โ€” Built for real money trading
90
+
91
+ ## Sponsors
92
+
93
+ <a href="https://www.finterion.com/" target="_blank">
94
+ <picture style="height: 30px;">
95
+ <source media="(prefers-color-scheme: dark)" srcset="static/sponsors/finterion-dark.png">
96
+ <source media="(prefers-color-scheme: light)" srcset="static/sponsors/finterion-light.png">
97
+ <img src="static/sponsors/finterion-light.png" alt="Finterion Logo" width="200px" height="50px">
98
+ </picture>
99
+ </a>
100
+
101
+
102
+ ## ๐Ÿ”Œ Plugins & Integrations
103
+
104
+ Extend your trading bot with powerful plugins:
105
+
106
+ | Plugin | Description |
107
+ |--------|-----------------------------------------------------------------------------|
108
+ | ๐ŸŽฏ **[PyIndicators](https://github.com/coding-kitties/PyIndicators)** | Technical analysis indicators for strategy development |
109
+ | ๐Ÿช **[Finterion Plugin](https://github.com/Finterion/finterion-investing-algorithm-framework-plugin)** | Monetize & share your strategies with the public on Finterion's marketplace |
110
+
111
+ ## ๐ŸŒŸ Powerful Features
112
+
113
+ | Feature | Description |
114
+ |---------|-------------|
115
+ | ๐Ÿ **Python 3.10+** | Cross-platform support for Windows, macOS, and Linux |
116
+ | โš™๏ธ **Event-Driven Backtest** | Accurate, realistic backtesting with event-driven architecture |
117
+ | โšก **Vectorized Backtest** | Lightning-fast signal research and prototyping |
118
+ | ๐Ÿ“Š **Advanced Metrics** | CAGR, Sharpe ratio, max drawdown, win rate, and 50+ more metrics |
119
+ | ๐Ÿ“ˆ **Backtest Reports** | Generate detailed, comparison-ready reports |
120
+ | ๐ŸŽฏ **Statistical Testing** | Permutation testing for strategy significance evaluation |
121
+ | ๐Ÿ’ฑ **Live Trading** | Real-time execution across multiple exchanges (via CCXT) |
122
+ | ๐Ÿ’ผ **Portfolio Management** | Full position and trade management with persistence |
123
+ | ๐Ÿ“‰ **Market Data** | OHLCV, tickers, custom data โ€” Polars & Pandas native |
124
+ | ๐Ÿ”— **Data Integrations** | PyIndicators, multiple data sources, custom providers |
125
+ | โ˜๏ธ **Cloud Deployment** | Azure Functions, AWS Lambda, and more |
126
+ | ๐ŸŒ **Web API** | REST API for bot interaction and monitoring |
127
+ | ๐Ÿงฉ **Fully Extensible** | Custom strategies, data providers, order executors |
128
+ | ๐Ÿ—๏ธ **Modular Design** | Build with reusable, composable components |
129
+
130
+
131
+
132
+ ## ๐Ÿš€ Quickstart
133
+
134
+ ### ๐Ÿ“ฆ Installation
135
+
136
+ Install the framework via [PyPI](https://pypi.org/project/investing-algorithm-framework/):
137
+
138
+ ```bash
139
+ pip install investing-algorithm-framework
140
+ ```
141
+
142
+ ### ๐ŸŽฏ Initialize Your Project
143
+
144
+ Run the following command to scaffold a new trading bot:
145
+
146
+ ```bash
147
+ investing-algorithm-framework init
148
+ ```
149
+
150
+ For an AWS Lambda-ready project:
151
+
152
+ ```bash
153
+ investing-algorithm-framework init --type aws_lambda
154
+ ```
155
+
156
+ This creates:
157
+ - **app.py** โ€” Your bot's entry point (keep as-is)
158
+ - **strategy.py** โ€” Your trading strategy (customize this!)
159
+
160
+ > ๐Ÿ’ก **Tip:** You can also create `default_web` or `azure_function` projects
161
+
162
+ ## ๐Ÿ“ˆ Example: A Simple Trading Bot
163
+ The following example trading bot implements a simple moving average strategy.
164
+ The strategy will use data from bitvavo exchange and will calculate
165
+ the 20, 50 and 100 period exponential moving averages (EMA) and the
166
+ 14 period relative strength index (RSI).
167
+
168
+ > This example uses [PyIndicators](https://github.com/coding-kitties/pyindicators) for technical analysis.
169
+ > This dependency is not part of the framework, but is used to perform technical analysis on the dataframes.
170
+ > You can install it using pip: pip install pyindicators.
171
+
172
+ ```python
173
+ from typing import Dict, Any
174
+ from datetime import datetime, timezone
175
+
176
+ import pandas as pd
177
+ from pyindicators import ema, rsi, crossover, crossunder
178
+
179
+ from investing_algorithm_framework import TradingStrategy, DataSource, \
180
+ TimeUnit, DataType, PositionSize, create_app, RESOURCE_DIRECTORY, \
181
+ BacktestDateRange, BacktestReport, TakeProfitRule, StopLossRule
182
+
183
+
184
+ class RSIEMACrossoverStrategy(TradingStrategy):
185
+ time_unit = TimeUnit.HOUR
186
+ interval = 2
187
+ symbols = ["BTC"]
188
+ position_sizes = [
189
+ PositionSize(
190
+ symbol="BTC", percentage_of_portfolio=20.0
191
+ ),
192
+ PositionSize(
193
+ symbol="ETH", percentage_of_portfolio=20.0
194
+ )
195
+ ]
196
+ take_profits = [
197
+ TakeProfitRule(
198
+ symbol="BTC",
199
+ percentage_threshold=10,
200
+ trailing=True,
201
+ sell_percentage=100
202
+ ),
203
+ TakeProfitRule(
204
+ symbol="ETH",
205
+ percentage_threshold=10,
206
+ trailing=True,
207
+ sell_percentage=100
208
+ )
209
+ ]
210
+ stop_losses = [
211
+ StopLossRule(
212
+ symbol="BTC",
213
+ percentage_threshold=5,
214
+ trailing=False,
215
+ sell_percentage=100
216
+ ),
217
+ StopLossRule(
218
+ symbol="ETH",
219
+ percentage_threshold=5,
220
+ trailing=False,
221
+ sell_percentage=100
222
+ )
223
+ ]
224
+
225
+ def __init__(
226
+ self,
227
+ time_unit: TimeUnit,
228
+ interval: int,
229
+ market: str,
230
+ rsi_time_frame: str,
231
+ rsi_period: int,
232
+ rsi_overbought_threshold,
233
+ rsi_oversold_threshold,
234
+ ema_time_frame,
235
+ ema_short_period,
236
+ ema_long_period,
237
+ ema_cross_lookback_window: int = 10
238
+ ):
239
+ self.rsi_time_frame = rsi_time_frame
240
+ self.rsi_period = rsi_period
241
+ self.rsi_result_column = f"rsi_{self.rsi_period}"
242
+ self.rsi_overbought_threshold = rsi_overbought_threshold
243
+ self.rsi_oversold_threshold = rsi_oversold_threshold
244
+ self.ema_time_frame = ema_time_frame
245
+ self.ema_short_result_column = f"ema_{ema_short_period}"
246
+ self.ema_long_result_column = f"ema_{ema_long_period}"
247
+ self.ema_crossunder_result_column = "ema_crossunder"
248
+ self.ema_crossover_result_column = "ema_crossover"
249
+ self.ema_short_period = ema_short_period
250
+ self.ema_long_period = ema_long_period
251
+ self.ema_cross_lookback_window = ema_cross_lookback_window
252
+ data_sources = []
253
+
254
+ for symbol in self.symbols:
255
+ full_symbol = f"{symbol}/EUR"
256
+ data_sources.append(
257
+ DataSource(
258
+ identifier=f"{symbol}_rsi_data",
259
+ data_type=DataType.OHLCV,
260
+ time_frame=self.rsi_time_frame,
261
+ market=market,
262
+ symbol=full_symbol,
263
+ pandas=True,
264
+ window_size=800
265
+ )
266
+ )
267
+ data_sources.append(
268
+ DataSource(
269
+ identifier=f"{symbol}_ema_data",
270
+ data_type=DataType.OHLCV,
271
+ time_frame=self.ema_time_frame,
272
+ market=market,
273
+ symbol=full_symbol,
274
+ pandas=True,
275
+ window_size=800
276
+ )
277
+ )
278
+
279
+ super().__init__(
280
+ data_sources=data_sources,
281
+ time_unit=time_unit,
282
+ interval=interval
283
+ )
284
+
285
+ def _prepare_indicators(
286
+ self,
287
+ rsi_data,
288
+ ema_data
289
+ ):
290
+ """
291
+ Helper function to prepare the indicators
292
+ for the strategy. The indicators are calculated
293
+ using the pyindicators library: https://github.com/coding-kitties/PyIndicators
294
+ """
295
+ ema_data = ema(
296
+ ema_data,
297
+ period=self.ema_short_period,
298
+ source_column="Close",
299
+ result_column=self.ema_short_result_column
300
+ )
301
+ ema_data = ema(
302
+ ema_data,
303
+ period=self.ema_long_period,
304
+ source_column="Close",
305
+ result_column=self.ema_long_result_column
306
+ )
307
+ # Detect crossover (short EMA crosses above long EMA)
308
+ ema_data = crossover(
309
+ ema_data,
310
+ first_column=self.ema_short_result_column,
311
+ second_column=self.ema_long_result_column,
312
+ result_column=self.ema_crossover_result_column
313
+ )
314
+ # Detect crossunder (short EMA crosses below long EMA)
315
+ ema_data = crossunder(
316
+ ema_data,
317
+ first_column=self.ema_short_result_column,
318
+ second_column=self.ema_long_result_column,
319
+ result_column=self.ema_crossunder_result_column
320
+ )
321
+ rsi_data = rsi(
322
+ rsi_data,
323
+ period=self.rsi_period,
324
+ source_column="Close",
325
+ result_column=self.rsi_result_column
326
+ )
327
+
328
+ return ema_data, rsi_data
329
+
330
+ def generate_buy_signals(self, data: Dict[str, Any]) -> Dict[str, pd.Series]:
331
+ """
332
+ Generate buy signals based on the moving average crossover.
333
+
334
+ data (Dict[str, Any]): Dictionary containing all the data for
335
+ the strategy data sources.
336
+
337
+ Returns:
338
+ Dict[str, pd.Series]: A dictionary where keys are symbols and values
339
+ are pandas Series indicating buy signals (True/False).
340
+ """
341
+
342
+ signals = {}
343
+
344
+ for symbol in self.symbols:
345
+ ema_data_identifier = f"{symbol}_ema_data"
346
+ rsi_data_identifier = f"{symbol}_rsi_data"
347
+ ema_data, rsi_data = self._prepare_indicators(
348
+ data[ema_data_identifier].copy(),
349
+ data[rsi_data_identifier].copy()
350
+ )
351
+
352
+ # crossover confirmed
353
+ ema_crossover_lookback = ema_data[
354
+ self.ema_crossover_result_column].rolling(
355
+ window=self.ema_cross_lookback_window
356
+ ).max().astype(bool)
357
+
358
+ # use only RSI column
359
+ rsi_oversold = rsi_data[self.rsi_result_column] \
360
+ < self.rsi_oversold_threshold
361
+
362
+ buy_signal = rsi_oversold & ema_crossover_lookback
363
+ buy_signals = buy_signal.fillna(False).astype(bool)
364
+ signals[symbol] = buy_signals
365
+ return signals
366
+
367
+ def generate_sell_signals(self, data: Dict[str, Any]) -> Dict[str, pd.Series]:
368
+ """
369
+ Generate sell signals based on the moving average crossover.
370
+
371
+ Args:
372
+ data (Dict[str, Any]): Dictionary containing all the data for
373
+ the strategy data sources.
374
+
375
+ Returns:
376
+ Dict[str, pd.Series]: A dictionary where keys are symbols and values
377
+ are pandas Series indicating sell signals (True/False).
378
+ """
379
+
380
+ signals = {}
381
+ for symbol in self.symbols:
382
+ ema_data_identifier = f"{symbol}_ema_data"
383
+ rsi_data_identifier = f"{symbol}_rsi_data"
384
+ ema_data, rsi_data = self._prepare_indicators(
385
+ data[ema_data_identifier].copy(),
386
+ data[rsi_data_identifier].copy()
387
+ )
388
+
389
+ # Confirmed by crossover between short-term EMA and long-term EMA
390
+ # within a given lookback window
391
+ ema_crossunder_lookback = ema_data[
392
+ self.ema_crossunder_result_column].rolling(
393
+ window=self.ema_cross_lookback_window
394
+ ).max().astype(bool)
395
+
396
+ # use only RSI column
397
+ rsi_overbought = rsi_data[self.rsi_result_column] \
398
+ >= self.rsi_overbought_threshold
399
+
400
+ # Combine both conditions
401
+ sell_signal = rsi_overbought & ema_crossunder_lookback
402
+ sell_signal = sell_signal.fillna(False).astype(bool)
403
+ signals[symbol] = sell_signal
404
+ return signals
405
+
406
+
407
+ if __name__ == "__main__":
408
+ app = create_app()
409
+ app.add_strategy(
410
+ RSIEMACrossoverStrategy(
411
+ time_unit=TimeUnit.HOUR,
412
+ interval=2,
413
+ market="bitvavo",
414
+ rsi_time_frame="2h",
415
+ rsi_period=14,
416
+ rsi_overbought_threshold=70,
417
+ rsi_oversold_threshold=30,
418
+ ema_time_frame="2h",
419
+ ema_short_period=12,
420
+ ema_long_period=26,
421
+ ema_cross_lookback_window=10
422
+ )
423
+ )
424
+
425
+ # Market credentials for coinbase for both the portfolio connection and data sources will
426
+ # be read from .env file, when not registering a market credential object in the app.
427
+ app.add_market(
428
+ market="bitvavo",
429
+ trading_symbol="EUR",
430
+ )
431
+ backtest_range = BacktestDateRange(
432
+ start_date=datetime(2023, 1, 1, tzinfo=timezone.utc),
433
+ end_date=datetime(2024, 6, 1, tzinfo=timezone.utc)
434
+ )
435
+ backtest = app.run_backtest(
436
+ backtest_date_range=backtest_range, initial_amount=1000
437
+ )
438
+ report = BacktestReport(backtest)
439
+ report.show(backtest_date_range=backtest_range, browser=True)
440
+ ```
441
+
442
+ > You can find more examples [here](./examples) folder.
443
+
444
+ ## ๐Ÿ“š Documentation
445
+ Comprehensive documentation is available at [github pages](https://coding-kitties.github.io/investing-algorithm-framework/).
446
+
447
+ ## ๐Ÿ› ๏ธ Development
448
+
449
+ ### Setup
450
+
451
+ Clone the repository and install dependencies using Poetry:
452
+
453
+ > Make sure you have [Poetry](https://python-poetry.org/docs/#installation) installed.
454
+
455
+ ```bash
456
+ git clone https://github.com/coding-kitties/investing-algorithm-framework.git
457
+ cd investing-algorithm-framework
458
+ poetry install
459
+ ```
460
+
461
+ ### Running Tests
462
+
463
+ ```bash
464
+ # Run all tests
465
+ python -m unittest discover -s tests
466
+
467
+ # Run specific test
468
+ python -m unittest tests.services.test_trade_service.TestTradeService
469
+ ```
470
+
471
+ ## โš ๏ธ Risk Disclaimer
472
+
473
+ ๐Ÿšจ **Use at Your Own Risk**
474
+
475
+ If you use this framework for your investments, **do not risk money which you are afraid to lose** until you have a clear understanding of how the framework works.
476
+
477
+ **BEFORE YOU START USING MONEY WITH THE FRAMEWORK:**
478
+ - โœ… Test your strategies thoroughly with backtesting
479
+ - โœ… Review the source code of any plugins you use
480
+ - โœ… Start with small amounts on paper trading first
481
+ - โœ… Understand the risks involved
482
+
483
+ **We assume no responsibility for your investment results. The authors and all affiliates disclaim any liability for losses.**
484
+
485
+ ---
486
+
487
+ ## ๐Ÿค Contributing
488
+
489
+ The investing algorithm framework is a **community-driven project**. We welcome contributions at all levels:
490
+
491
+ - ๐Ÿ› **Found a bug?** [Open an issue](https://github.com/coding-kitties/investing-algorithm-framework/issues/new)
492
+ - ๐Ÿ’ก **Have an idea?** [Share it with us](https://github.com/coding-kitties/investing-algorithm-framework/issues/new)
493
+ - ๐Ÿ”ง **Want to code?** Check the [project board](https://github.com/coding-kitties/investing-algorithm-framework/projects?query=is%3Aopen)
494
+
495
+ **Guidelines:**
496
+ - Read the [Contributing Guide](https://coding-kitties.github.io/investing-algorithm-framework/Contributing%20Guide/contributing)
497
+ - Always create PRs against the `develop` branch, not `main`
498
+ - Open an issue before starting major feature work
499
+
500
+ ---
501
+
502
+ ## ๐Ÿ“š Documentation
503
+
504
+ Comprehensive documentation is available at [GitHub Pages](https://coding-kitties.github.io/investing-algorithm-framework/)
505
+
506
+ ---
507
+
508
+ ## ๐Ÿ“ฌ Community
509
+
510
+ Join us and connect with other traders and developers:
511
+
512
+ * ๐Ÿ’ฌ [Discord Community](https://discord.gg/dQsRmGZP) โ€” Real-time chat and support
513
+ * ๐Ÿ”— [Reddit Community](https://www.reddit.com/r/InvestingBots/) โ€” Share strategies and discuss
514
+ * ๐Ÿ“– [Documentation](https://coding-kitties.github.io/investing-algorithm-framework/) โ€” Guides and API references
515
+
516
+ ---
517
+
518
+ ## ๐Ÿ† Acknowledgements
519
+
520
+ We want to thank all contributors to this project. A full list can be found in [AUTHORS.md](https://github.com/coding-kitties/investing-algorithm-framework/blob/master/AUTHORS.md)
521
+
522
+ ### Report Issues
523
+
524
+ If you discover a bug in the framework, please [search our issue tracker](https://github.com/coding-kitties/investing-algorithm-framework/issues?q=is%3Aissue) first. If it hasn't been reported, please [create a new issue](https://github.com/coding-kitties/investing-algorithm-framework/issues/new).
525
+
526
+ ---
527
+
528
+ <div align="center">
529
+ <p>
530
+ <a href="https://github.com/coding-kitties/investing-algorithm-framework/stargazers">โญ Star us on GitHub</a> ยท
531
+ <a href="https://discord.gg/dQsRmGZP">๐Ÿ’ฌ Join Discord</a> ยท
532
+ <a href="https://github.com/coding-kitties/investing-algorithm-framework/issues/new">๐Ÿ› Report Bug</a>
533
+ </p>
534
+ </div>
535
+