quantwise 1.2.0 → 1.2.2

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 (362) hide show
  1. package/.claude/skills/README.md +80 -0
  2. package/.claude/skills/backtest-expert/SKILL.md +206 -0
  3. package/.claude/skills/backtest-expert/references/failed_tests.md +236 -0
  4. package/.claude/skills/backtest-expert/references/methodology.md +227 -0
  5. package/.claude/skills/breadth-chart-analyst/SKILL.md +583 -0
  6. package/.claude/skills/breadth-chart-analyst/assets/SP500_Breadth_Index_200MA_8MA.jpeg +0 -0
  7. package/.claude/skills/breadth-chart-analyst/assets/US_Stock_Market_Uptrend_Ratio.jpeg +0 -0
  8. package/.claude/skills/breadth-chart-analyst/assets/breadth_analysis_template.md +558 -0
  9. package/.claude/skills/breadth-chart-analyst/references/breadth_chart_methodology.md +590 -0
  10. package/.claude/skills/canslim-screener/SKILL.md +599 -0
  11. package/.claude/skills/canslim-screener/references/canslim_methodology.md +606 -0
  12. package/.claude/skills/canslim-screener/references/fmp_api_endpoints.md +707 -0
  13. package/.claude/skills/canslim-screener/references/interpretation_guide.md +516 -0
  14. package/.claude/skills/canslim-screener/references/scoring_system.md +597 -0
  15. package/.claude/skills/canslim-screener/scripts/calculators/earnings_calculator.py +343 -0
  16. package/.claude/skills/canslim-screener/scripts/calculators/growth_calculator.py +334 -0
  17. package/.claude/skills/canslim-screener/scripts/calculators/institutional_calculator.py +347 -0
  18. package/.claude/skills/canslim-screener/scripts/calculators/leadership_calculator.py +380 -0
  19. package/.claude/skills/canslim-screener/scripts/calculators/market_calculator.py +244 -0
  20. package/.claude/skills/canslim-screener/scripts/calculators/new_highs_calculator.py +194 -0
  21. package/.claude/skills/canslim-screener/scripts/calculators/supply_demand_calculator.py +221 -0
  22. package/.claude/skills/canslim-screener/scripts/finviz_stock_client.py +227 -0
  23. package/.claude/skills/canslim-screener/scripts/fmp_client.py +393 -0
  24. package/.claude/skills/canslim-screener/scripts/report_generator.py +405 -0
  25. package/.claude/skills/canslim-screener/scripts/scorer.py +625 -0
  26. package/.claude/skills/canslim-screener/scripts/screen_canslim.py +361 -0
  27. package/.claude/skills/canslim-screener/scripts/test_institutional_endpoint.py +109 -0
  28. package/.claude/skills/chart/SKILL.md +20 -0
  29. package/.claude/skills/dividend-growth-pullback-screener/SKILL.md +322 -0
  30. package/.claude/skills/dividend-growth-pullback-screener/references/dividend_growth_compounding.md +400 -0
  31. package/.claude/skills/dividend-growth-pullback-screener/references/fmp_api_guide.md +642 -0
  32. package/.claude/skills/dividend-growth-pullback-screener/references/rsi_oversold_strategy.md +333 -0
  33. package/.claude/skills/dividend-growth-pullback-screener/scripts/screen_dividend_growth_rsi.py +1155 -0
  34. package/.claude/skills/earnings-calendar/SKILL.md +721 -0
  35. package/.claude/skills/earnings-calendar/assets/earnings_report_template.md +102 -0
  36. package/.claude/skills/earnings-calendar/references/fmp_api_guide.md +590 -0
  37. package/.claude/skills/earnings-calendar/scripts/fetch_earnings_fmp.py +443 -0
  38. package/.claude/skills/earnings-calendar/scripts/generate_report.py +366 -0
  39. package/.claude/skills/economic-calendar-fetcher/SKILL.md +365 -0
  40. package/.claude/skills/economic-calendar-fetcher/references/fmp_api_documentation.md +345 -0
  41. package/.claude/skills/economic-calendar-fetcher/scripts/get_economic_calendar.py +267 -0
  42. package/.claude/skills/ftd-detector/SKILL.md +147 -0
  43. package/.claude/skills/ftd-detector/references/ftd_methodology.md +188 -0
  44. package/.claude/skills/ftd-detector/references/post_ftd_guide.md +185 -0
  45. package/.claude/skills/ftd-detector/scripts/fmp_client.py +158 -0
  46. package/.claude/skills/ftd-detector/scripts/ftd_detector.py +280 -0
  47. package/.claude/skills/ftd-detector/scripts/post_ftd_monitor.py +404 -0
  48. package/.claude/skills/ftd-detector/scripts/rally_tracker.py +508 -0
  49. package/.claude/skills/ftd-detector/scripts/report_generator.py +341 -0
  50. package/.claude/skills/ftd-detector/scripts/tests/conftest.py +9 -0
  51. package/.claude/skills/ftd-detector/scripts/tests/helpers.py +107 -0
  52. package/.claude/skills/ftd-detector/scripts/tests/test_post_ftd_monitor.py +311 -0
  53. package/.claude/skills/ftd-detector/scripts/tests/test_rally_tracker.py +302 -0
  54. package/.claude/skills/institutional-flow-tracker/README.md +362 -0
  55. package/.claude/skills/institutional-flow-tracker/SKILL.md +357 -0
  56. package/.claude/skills/institutional-flow-tracker/references/13f_filings_guide.md +383 -0
  57. package/.claude/skills/institutional-flow-tracker/references/institutional_investor_types.md +580 -0
  58. package/.claude/skills/institutional-flow-tracker/references/interpretation_framework.md +573 -0
  59. package/.claude/skills/institutional-flow-tracker/scripts/analyze_single_stock.py +457 -0
  60. package/.claude/skills/institutional-flow-tracker/scripts/track_institution_portfolio.py +108 -0
  61. package/.claude/skills/institutional-flow-tracker/scripts/track_institutional_flow.py +450 -0
  62. package/.claude/skills/macro-regime-detector/SKILL.md +86 -0
  63. package/.claude/skills/macro-regime-detector/references/historical_regimes.md +124 -0
  64. package/.claude/skills/macro-regime-detector/references/indicator_interpretation_guide.md +144 -0
  65. package/.claude/skills/macro-regime-detector/references/regime_detection_methodology.md +138 -0
  66. package/.claude/skills/macro-regime-detector/scripts/calculators/__init__.py +1 -0
  67. package/.claude/skills/macro-regime-detector/scripts/calculators/concentration_calculator.py +165 -0
  68. package/.claude/skills/macro-regime-detector/scripts/calculators/credit_conditions_calculator.py +124 -0
  69. package/.claude/skills/macro-regime-detector/scripts/calculators/equity_bond_calculator.py +198 -0
  70. package/.claude/skills/macro-regime-detector/scripts/calculators/sector_rotation_calculator.py +123 -0
  71. package/.claude/skills/macro-regime-detector/scripts/calculators/size_factor_calculator.py +131 -0
  72. package/.claude/skills/macro-regime-detector/scripts/calculators/utils.py +347 -0
  73. package/.claude/skills/macro-regime-detector/scripts/calculators/yield_curve_calculator.py +279 -0
  74. package/.claude/skills/macro-regime-detector/scripts/fmp_client.py +134 -0
  75. package/.claude/skills/macro-regime-detector/scripts/macro_regime_detector.py +278 -0
  76. package/.claude/skills/macro-regime-detector/scripts/report_generator.py +327 -0
  77. package/.claude/skills/macro-regime-detector/scripts/scorer.py +574 -0
  78. package/.claude/skills/macro-regime-detector/scripts/tests/conftest.py +9 -0
  79. package/.claude/skills/macro-regime-detector/scripts/tests/test_concentration.py +78 -0
  80. package/.claude/skills/macro-regime-detector/scripts/tests/test_credit_conditions.py +59 -0
  81. package/.claude/skills/macro-regime-detector/scripts/tests/test_equity_bond.py +74 -0
  82. package/.claude/skills/macro-regime-detector/scripts/tests/test_helpers.py +90 -0
  83. package/.claude/skills/macro-regime-detector/scripts/tests/test_scorer.py +439 -0
  84. package/.claude/skills/macro-regime-detector/scripts/tests/test_sector_rotation.py +78 -0
  85. package/.claude/skills/macro-regime-detector/scripts/tests/test_size_factor.py +59 -0
  86. package/.claude/skills/macro-regime-detector/scripts/tests/test_utils.py +126 -0
  87. package/.claude/skills/macro-regime-detector/scripts/tests/test_yield_curve.py +64 -0
  88. package/.claude/skills/market-breadth-analyzer/SKILL.md +121 -0
  89. package/.claude/skills/market-breadth-analyzer/references/breadth_analysis_methodology.md +168 -0
  90. package/.claude/skills/market-breadth-analyzer/scripts/calculators/__init__.py +1 -0
  91. package/.claude/skills/market-breadth-analyzer/scripts/calculators/bearish_signal_calculator.py +150 -0
  92. package/.claude/skills/market-breadth-analyzer/scripts/calculators/cycle_calculator.py +168 -0
  93. package/.claude/skills/market-breadth-analyzer/scripts/calculators/divergence_calculator.py +119 -0
  94. package/.claude/skills/market-breadth-analyzer/scripts/calculators/historical_context_calculator.py +120 -0
  95. package/.claude/skills/market-breadth-analyzer/scripts/calculators/ma_crossover_calculator.py +115 -0
  96. package/.claude/skills/market-breadth-analyzer/scripts/calculators/trend_level_calculator.py +103 -0
  97. package/.claude/skills/market-breadth-analyzer/scripts/csv_client.py +225 -0
  98. package/.claude/skills/market-breadth-analyzer/scripts/market_breadth_analyzer.py +307 -0
  99. package/.claude/skills/market-breadth-analyzer/scripts/report_generator.py +330 -0
  100. package/.claude/skills/market-breadth-analyzer/scripts/scorer.py +271 -0
  101. package/.claude/skills/market-environment-analysis/SKILL.md +139 -0
  102. package/.claude/skills/market-environment-analysis/references/analysis_patterns.md +124 -0
  103. package/.claude/skills/market-environment-analysis/references/indicators.md +99 -0
  104. package/.claude/skills/market-environment-analysis/scripts/market_utils.py +127 -0
  105. package/.claude/skills/market-news-analyst/SKILL.md +714 -0
  106. package/.claude/skills/market-news-analyst/references/corporate_news_impact.md +446 -0
  107. package/.claude/skills/market-news-analyst/references/geopolitical_commodity_correlations.md +499 -0
  108. package/.claude/skills/market-news-analyst/references/market_event_patterns.md +393 -0
  109. package/.claude/skills/market-news-analyst/references/trusted_news_sources.md +510 -0
  110. package/.claude/skills/market-top-detector/SKILL.md +159 -0
  111. package/.claude/skills/market-top-detector/references/distribution_day_guide.md +100 -0
  112. package/.claude/skills/market-top-detector/references/historical_tops.md +142 -0
  113. package/.claude/skills/market-top-detector/references/market_top_methodology.md +167 -0
  114. package/.claude/skills/market-top-detector/scripts/calculators/__init__.py +17 -0
  115. package/.claude/skills/market-top-detector/scripts/calculators/breadth_calculator.py +116 -0
  116. package/.claude/skills/market-top-detector/scripts/calculators/defensive_rotation_calculator.py +127 -0
  117. package/.claude/skills/market-top-detector/scripts/calculators/distribution_day_calculator.py +161 -0
  118. package/.claude/skills/market-top-detector/scripts/calculators/index_technical_calculator.py +254 -0
  119. package/.claude/skills/market-top-detector/scripts/calculators/leading_stock_calculator.py +198 -0
  120. package/.claude/skills/market-top-detector/scripts/calculators/sentiment_calculator.py +213 -0
  121. package/.claude/skills/market-top-detector/scripts/fmp_client.py +158 -0
  122. package/.claude/skills/market-top-detector/scripts/market_top_detector.py +349 -0
  123. package/.claude/skills/market-top-detector/scripts/report_generator.py +314 -0
  124. package/.claude/skills/market-top-detector/scripts/scorer.py +473 -0
  125. package/.claude/skills/market-top-detector/scripts/tests/conftest.py +9 -0
  126. package/.claude/skills/market-top-detector/scripts/tests/helpers.py +49 -0
  127. package/.claude/skills/market-top-detector/scripts/tests/test_breadth.py +62 -0
  128. package/.claude/skills/market-top-detector/scripts/tests/test_defensive_rotation.py +56 -0
  129. package/.claude/skills/market-top-detector/scripts/tests/test_distribution_day.py +92 -0
  130. package/.claude/skills/market-top-detector/scripts/tests/test_index_technical.py +73 -0
  131. package/.claude/skills/market-top-detector/scripts/tests/test_leading_stock.py +57 -0
  132. package/.claude/skills/market-top-detector/scripts/tests/test_scorer.py +180 -0
  133. package/.claude/skills/market-top-detector/scripts/tests/test_sentiment.py +64 -0
  134. package/.claude/skills/options-strategy-advisor/README.md +469 -0
  135. package/.claude/skills/options-strategy-advisor/SKILL.md +959 -0
  136. package/.claude/skills/options-strategy-advisor/scripts/black_scholes.py +495 -0
  137. package/.claude/skills/pair-trade-screener/README.md +389 -0
  138. package/.claude/skills/pair-trade-screener/SKILL.md +622 -0
  139. package/.claude/skills/pair-trade-screener/references/cointegration_guide.md +745 -0
  140. package/.claude/skills/pair-trade-screener/references/methodology.md +853 -0
  141. package/.claude/skills/pair-trade-screener/scripts/analyze_spread.py +394 -0
  142. package/.claude/skills/pair-trade-screener/scripts/find_pairs.py +535 -0
  143. package/.claude/skills/portfolio-manager/README.md +394 -0
  144. package/.claude/skills/portfolio-manager/SKILL.md +750 -0
  145. package/.claude/skills/portfolio-manager/references/alpaca-mcp-setup.md +367 -0
  146. package/.claude/skills/portfolio-manager/references/asset-allocation.md +502 -0
  147. package/.claude/skills/portfolio-manager/references/diversification-principles.md +553 -0
  148. package/.claude/skills/portfolio-manager/references/portfolio-risk-metrics.md +603 -0
  149. package/.claude/skills/portfolio-manager/references/position-evaluation.md +477 -0
  150. package/.claude/skills/portfolio-manager/references/rebalancing-strategies.md +715 -0
  151. package/.claude/skills/portfolio-manager/references/risk-profile-questionnaire.md +608 -0
  152. package/.claude/skills/portfolio-manager/references/target-allocations.md +558 -0
  153. package/.claude/skills/portfolio-manager/scripts/test_alpaca_connection.py +286 -0
  154. package/.claude/skills/scenario-analyzer/SKILL.md +317 -0
  155. package/.claude/skills/scenario-analyzer/references/headline_event_patterns.md +264 -0
  156. package/.claude/skills/scenario-analyzer/references/scenario_playbooks.md +320 -0
  157. package/.claude/skills/scenario-analyzer/references/sector_sensitivity_matrix.md +217 -0
  158. package/.claude/skills/sector-analyst/SKILL.md +206 -0
  159. package/.claude/skills/sector-analyst/assets/industory_performance_1.jpeg +0 -0
  160. package/.claude/skills/sector-analyst/assets/industory_performance_2.jpeg +0 -0
  161. package/.claude/skills/sector-analyst/assets/sector_performance.jpeg +0 -0
  162. package/.claude/skills/sector-analyst/references/sector_rotation.md +170 -0
  163. package/.claude/skills/stanley-druckenmiller-investment/SKILL.md +84 -0
  164. package/.claude/skills/stanley-druckenmiller-investment/references/case-studies.md +148 -0
  165. package/.claude/skills/stanley-druckenmiller-investment/references/investment-philosophy.md +80 -0
  166. package/.claude/skills/stanley-druckenmiller-investment/references/market-analysis-guide.md +146 -0
  167. package/.claude/skills/stock/NOTION_SETUP.md +33 -0
  168. package/.claude/skills/stock/SKILL.md +38 -0
  169. package/.claude/skills/technical-analyst/SKILL.md +238 -0
  170. package/.claude/skills/technical-analyst/assets/analysis_template.md +183 -0
  171. package/.claude/skills/technical-analyst/references/technical_analysis_framework.md +282 -0
  172. package/.claude/skills/theme-detector/SKILL.md +320 -0
  173. package/.claude/skills/theme-detector/assets/report_template.md +155 -0
  174. package/.claude/skills/theme-detector/references/cross_sector_themes.md +252 -0
  175. package/.claude/skills/theme-detector/references/finviz_industry_codes.md +403 -0
  176. package/.claude/skills/theme-detector/references/thematic_etf_catalog.md +333 -0
  177. package/.claude/skills/theme-detector/references/theme_detection_methodology.md +430 -0
  178. package/.claude/skills/theme-detector/scripts/calculators/__init__.py +1 -0
  179. package/.claude/skills/theme-detector/scripts/calculators/heat_calculator.py +123 -0
  180. package/.claude/skills/theme-detector/scripts/calculators/industry_ranker.py +98 -0
  181. package/.claude/skills/theme-detector/scripts/calculators/lifecycle_calculator.py +172 -0
  182. package/.claude/skills/theme-detector/scripts/calculators/theme_classifier.py +195 -0
  183. package/.claude/skills/theme-detector/scripts/calculators/theme_discoverer.py +280 -0
  184. package/.claude/skills/theme-detector/scripts/config_loader.py +142 -0
  185. package/.claude/skills/theme-detector/scripts/default_theme_config.py +254 -0
  186. package/.claude/skills/theme-detector/scripts/etf_scanner.py +609 -0
  187. package/.claude/skills/theme-detector/scripts/finviz_performance_client.py +131 -0
  188. package/.claude/skills/theme-detector/scripts/report_generator.py +490 -0
  189. package/.claude/skills/theme-detector/scripts/representative_stock_selector.py +673 -0
  190. package/.claude/skills/theme-detector/scripts/scorer.py +87 -0
  191. package/.claude/skills/theme-detector/scripts/tests/README.md +21 -0
  192. package/.claude/skills/theme-detector/scripts/tests/conftest.py +9 -0
  193. package/.claude/skills/theme-detector/scripts/tests/test_config_loader.py +239 -0
  194. package/.claude/skills/theme-detector/scripts/tests/test_etf_scanner.py +810 -0
  195. package/.claude/skills/theme-detector/scripts/tests/test_heat_calculator.py +245 -0
  196. package/.claude/skills/theme-detector/scripts/tests/test_industry_ranker.py +256 -0
  197. package/.claude/skills/theme-detector/scripts/tests/test_lifecycle_calculator.py +301 -0
  198. package/.claude/skills/theme-detector/scripts/tests/test_report_generator.py +624 -0
  199. package/.claude/skills/theme-detector/scripts/tests/test_representative_stock_selector.py +898 -0
  200. package/.claude/skills/theme-detector/scripts/tests/test_scorer.py +185 -0
  201. package/.claude/skills/theme-detector/scripts/tests/test_theme_classifier.py +534 -0
  202. package/.claude/skills/theme-detector/scripts/tests/test_theme_detector_e2e.py +467 -0
  203. package/.claude/skills/theme-detector/scripts/tests/test_theme_discoverer.py +458 -0
  204. package/.claude/skills/theme-detector/scripts/tests/test_uptrend_client.py +76 -0
  205. package/.claude/skills/theme-detector/scripts/theme_detector.py +815 -0
  206. package/.claude/skills/theme-detector/scripts/themes.yaml +168 -0
  207. package/.claude/skills/theme-detector/scripts/uptrend_client.py +241 -0
  208. package/.claude/skills/uptrend-analyzer/SKILL.md +108 -0
  209. package/.claude/skills/uptrend-analyzer/references/uptrend_methodology.md +215 -0
  210. package/.claude/skills/uptrend-analyzer/scripts/calculators/__init__.py +1 -0
  211. package/.claude/skills/uptrend-analyzer/scripts/calculators/historical_context_calculator.py +122 -0
  212. package/.claude/skills/uptrend-analyzer/scripts/calculators/market_breadth_calculator.py +145 -0
  213. package/.claude/skills/uptrend-analyzer/scripts/calculators/momentum_calculator.py +183 -0
  214. package/.claude/skills/uptrend-analyzer/scripts/calculators/sector_participation_calculator.py +204 -0
  215. package/.claude/skills/uptrend-analyzer/scripts/calculators/sector_rotation_calculator.py +218 -0
  216. package/.claude/skills/uptrend-analyzer/scripts/data_fetcher.py +236 -0
  217. package/.claude/skills/uptrend-analyzer/scripts/report_generator.py +329 -0
  218. package/.claude/skills/uptrend-analyzer/scripts/scorer.py +276 -0
  219. package/.claude/skills/uptrend-analyzer/scripts/uptrend_analyzer.py +219 -0
  220. package/.claude/skills/us-market-bubble-detector/CHANGELOG.md +118 -0
  221. package/.claude/skills/us-market-bubble-detector/SKILL.md +545 -0
  222. package/.claude/skills/us-market-bubble-detector/references/bubble_framework.md +335 -0
  223. package/.claude/skills/us-market-bubble-detector/references/historical_cases.md +327 -0
  224. package/.claude/skills/us-market-bubble-detector/references/implementation_guide.md +473 -0
  225. package/.claude/skills/us-market-bubble-detector/references/quick_reference.md +354 -0
  226. package/.claude/skills/us-market-bubble-detector/references/quick_reference_en.md +342 -0
  227. package/.claude/skills/us-market-bubble-detector/scripts/bubble_scorer.py +309 -0
  228. package/.claude/skills/us-stock-analysis/SKILL.md +294 -0
  229. package/.claude/skills/us-stock-analysis/references/financial-metrics.md +172 -0
  230. package/.claude/skills/us-stock-analysis/references/fundamental-analysis.md +129 -0
  231. package/.claude/skills/us-stock-analysis/references/report-template.md +207 -0
  232. package/.claude/skills/us-stock-analysis/references/technical-analysis.md +93 -0
  233. package/.claude/skills/value-dividend-screener/SKILL.md +562 -0
  234. package/.claude/skills/value-dividend-screener/references/fmp_api_guide.md +348 -0
  235. package/.claude/skills/value-dividend-screener/references/screening_methodology.md +315 -0
  236. package/.claude/skills/value-dividend-screener/scripts/screen_dividend_stocks.py +1138 -0
  237. package/.claude/skills/vcp-screener/SKILL.md +79 -0
  238. package/.claude/skills/vcp-screener/references/fmp_api_endpoints.md +45 -0
  239. package/.claude/skills/vcp-screener/references/scoring_system.md +154 -0
  240. package/.claude/skills/vcp-screener/references/vcp_methodology.md +124 -0
  241. package/.claude/skills/vcp-screener/scripts/calculators/__init__.py +1 -0
  242. package/.claude/skills/vcp-screener/scripts/calculators/pivot_proximity_calculator.py +139 -0
  243. package/.claude/skills/vcp-screener/scripts/calculators/relative_strength_calculator.py +161 -0
  244. package/.claude/skills/vcp-screener/scripts/calculators/trend_template_calculator.py +228 -0
  245. package/.claude/skills/vcp-screener/scripts/calculators/vcp_pattern_calculator.py +322 -0
  246. package/.claude/skills/vcp-screener/scripts/calculators/volume_pattern_calculator.py +121 -0
  247. package/.claude/skills/vcp-screener/scripts/fmp_client.py +162 -0
  248. package/.claude/skills/vcp-screener/scripts/report_generator.py +317 -0
  249. package/.claude/skills/vcp-screener/scripts/scorer.py +155 -0
  250. package/.claude/skills/vcp-screener/scripts/screen_vcp.py +536 -0
  251. package/.claude/skills/vcp-screener/scripts/tests/__init__.py +0 -0
  252. package/.claude/skills/vcp-screener/scripts/tests/conftest.py +9 -0
  253. package/.claude/skills/vcp-screener/scripts/tests/test_vcp_screener.py +834 -0
  254. package/.claude/skills/weekly-trade-strategy/.claude/agents/druckenmiller-strategy-planner.md +300 -0
  255. package/.claude/skills/weekly-trade-strategy/.claude/agents/market-news-analyzer.md +239 -0
  256. package/.claude/skills/weekly-trade-strategy/.claude/agents/technical-market-analyst.md +187 -0
  257. package/.claude/skills/weekly-trade-strategy/.claude/agents/us-market-analyst.md +218 -0
  258. package/.claude/skills/weekly-trade-strategy/.claude/agents/weekly-trade-blog-writer.md +318 -0
  259. package/.claude/skills/weekly-trade-strategy/.claude/skills/breadth-chart-analyst/SKILL.md +662 -0
  260. package/.claude/skills/weekly-trade-strategy/.claude/skills/breadth-chart-analyst/assets/SP500_Breadth_Index_200MA_8MA.jpeg +0 -0
  261. package/.claude/skills/weekly-trade-strategy/.claude/skills/breadth-chart-analyst/assets/US_Stock_Market_Uptrend_Ratio.jpeg +0 -0
  262. package/.claude/skills/weekly-trade-strategy/.claude/skills/breadth-chart-analyst/assets/breadth_analysis_template.md +558 -0
  263. package/.claude/skills/weekly-trade-strategy/.claude/skills/breadth-chart-analyst/references/breadth_chart_methodology.md +590 -0
  264. package/.claude/skills/weekly-trade-strategy/.claude/skills/earnings-calendar/SKILL.md +721 -0
  265. package/.claude/skills/weekly-trade-strategy/.claude/skills/earnings-calendar/assets/earnings_report_template.md +102 -0
  266. package/.claude/skills/weekly-trade-strategy/.claude/skills/earnings-calendar/earnings_calendar_2025-11-02.md +447 -0
  267. package/.claude/skills/weekly-trade-strategy/.claude/skills/earnings-calendar/references/fmp_api_guide.md +590 -0
  268. package/.claude/skills/weekly-trade-strategy/.claude/skills/earnings-calendar/scripts/fetch_earnings_fmp.py +443 -0
  269. package/.claude/skills/weekly-trade-strategy/.claude/skills/earnings-calendar/scripts/generate_report.py +366 -0
  270. package/.claude/skills/weekly-trade-strategy/.claude/skills/economic-calendar-fetcher/SKILL.md +365 -0
  271. package/.claude/skills/weekly-trade-strategy/.claude/skills/economic-calendar-fetcher/references/fmp_api_documentation.md +345 -0
  272. package/.claude/skills/weekly-trade-strategy/.claude/skills/economic-calendar-fetcher/scripts/get_economic_calendar.py +267 -0
  273. package/.claude/skills/weekly-trade-strategy/.claude/skills/market-environment-analysis/SKILL.md +139 -0
  274. package/.claude/skills/weekly-trade-strategy/.claude/skills/market-environment-analysis/references/analysis_patterns.md +124 -0
  275. package/.claude/skills/weekly-trade-strategy/.claude/skills/market-environment-analysis/references/indicators.md +99 -0
  276. package/.claude/skills/weekly-trade-strategy/.claude/skills/market-environment-analysis/scripts/market_utils.py +127 -0
  277. package/.claude/skills/weekly-trade-strategy/.claude/skills/market-news-analyst/SKILL.md +714 -0
  278. package/.claude/skills/weekly-trade-strategy/.claude/skills/market-news-analyst/references/corporate_news_impact.md +446 -0
  279. package/.claude/skills/weekly-trade-strategy/.claude/skills/market-news-analyst/references/geopolitical_commodity_correlations.md +499 -0
  280. package/.claude/skills/weekly-trade-strategy/.claude/skills/market-news-analyst/references/market_event_patterns.md +393 -0
  281. package/.claude/skills/weekly-trade-strategy/.claude/skills/market-news-analyst/references/trusted_news_sources.md +510 -0
  282. package/.claude/skills/weekly-trade-strategy/.claude/skills/sector-analyst/SKILL.md +206 -0
  283. package/.claude/skills/weekly-trade-strategy/.claude/skills/sector-analyst/assets/industory_performance_1.jpeg +0 -0
  284. package/.claude/skills/weekly-trade-strategy/.claude/skills/sector-analyst/assets/industory_performance_2.jpeg +0 -0
  285. package/.claude/skills/weekly-trade-strategy/.claude/skills/sector-analyst/assets/sector_performance.jpeg +0 -0
  286. package/.claude/skills/weekly-trade-strategy/.claude/skills/sector-analyst/references/sector_rotation.md +170 -0
  287. package/.claude/skills/weekly-trade-strategy/.claude/skills/stanley-druckenmiller-investment/SKILL.md +84 -0
  288. package/.claude/skills/weekly-trade-strategy/.claude/skills/stanley-druckenmiller-investment/references/case-studies.md +148 -0
  289. package/.claude/skills/weekly-trade-strategy/.claude/skills/stanley-druckenmiller-investment/references/investment-philosophy.md +80 -0
  290. package/.claude/skills/weekly-trade-strategy/.claude/skills/stanley-druckenmiller-investment/references/market-analysis-guide.md +146 -0
  291. package/.claude/skills/weekly-trade-strategy/.claude/skills/technical-analyst/SKILL.md +238 -0
  292. package/.claude/skills/weekly-trade-strategy/.claude/skills/technical-analyst/assets/analysis_template.md +183 -0
  293. package/.claude/skills/weekly-trade-strategy/.claude/skills/technical-analyst/references/technical_analysis_framework.md +282 -0
  294. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-market-bubble-detector/CHANGELOG.md +118 -0
  295. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-market-bubble-detector/SKILL.md +545 -0
  296. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-market-bubble-detector/references/bubble_framework.md +335 -0
  297. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-market-bubble-detector/references/historical_cases.md +327 -0
  298. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-market-bubble-detector/references/implementation_guide.md +473 -0
  299. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-market-bubble-detector/references/quick_reference.md +354 -0
  300. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-market-bubble-detector/references/quick_reference_en.md +342 -0
  301. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-market-bubble-detector/scripts/bubble_scorer.py +309 -0
  302. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-stock-analysis/SKILL.md +294 -0
  303. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-stock-analysis/references/financial-metrics.md +172 -0
  304. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-stock-analysis/references/fundamental-analysis.md +129 -0
  305. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-stock-analysis/references/report-template.md +207 -0
  306. package/.claude/skills/weekly-trade-strategy/.claude/skills/us-stock-analysis/references/technical-analysis.md +93 -0
  307. package/.claude/skills/weekly-trade-strategy/CLAUDE.md +454 -0
  308. package/.claude/skills/weekly-trade-strategy/README.md +287 -0
  309. package/.claude/skills/weekly-trade-strategy/blogs/.gitkeep +0 -0
  310. package/.claude/skills/weekly-trade-strategy/charts/.gitkeep +0 -0
  311. package/.claude/skills/weekly-trade-strategy/earnings_data.json +10054 -0
  312. package/.claude/skills/weekly-trade-strategy/skills/breadth-chart-analyst/SKILL.md +662 -0
  313. package/.claude/skills/weekly-trade-strategy/skills/breadth-chart-analyst/assets/SP500_Breadth_Index_200MA_8MA.jpeg +0 -0
  314. package/.claude/skills/weekly-trade-strategy/skills/breadth-chart-analyst/assets/US_Stock_Market_Uptrend_Ratio.jpeg +0 -0
  315. package/.claude/skills/weekly-trade-strategy/skills/breadth-chart-analyst/assets/breadth_analysis_template.md +558 -0
  316. package/.claude/skills/weekly-trade-strategy/skills/breadth-chart-analyst/references/breadth_chart_methodology.md +590 -0
  317. package/.claude/skills/weekly-trade-strategy/skills/earnings-calendar/SKILL.md +721 -0
  318. package/.claude/skills/weekly-trade-strategy/skills/earnings-calendar/assets/earnings_report_template.md +102 -0
  319. package/.claude/skills/weekly-trade-strategy/skills/earnings-calendar/earnings_calendar_2025-11-02.md +447 -0
  320. package/.claude/skills/weekly-trade-strategy/skills/earnings-calendar/references/fmp_api_guide.md +590 -0
  321. package/.claude/skills/weekly-trade-strategy/skills/earnings-calendar/scripts/fetch_earnings_fmp.py +443 -0
  322. package/.claude/skills/weekly-trade-strategy/skills/earnings-calendar/scripts/generate_report.py +366 -0
  323. package/.claude/skills/weekly-trade-strategy/skills/economic-calendar-fetcher/SKILL.md +365 -0
  324. package/.claude/skills/weekly-trade-strategy/skills/economic-calendar-fetcher/references/fmp_api_documentation.md +345 -0
  325. package/.claude/skills/weekly-trade-strategy/skills/economic-calendar-fetcher/scripts/get_economic_calendar.py +267 -0
  326. package/.claude/skills/weekly-trade-strategy/skills/market-environment-analysis/SKILL.md +139 -0
  327. package/.claude/skills/weekly-trade-strategy/skills/market-environment-analysis/references/analysis_patterns.md +124 -0
  328. package/.claude/skills/weekly-trade-strategy/skills/market-environment-analysis/references/indicators.md +99 -0
  329. package/.claude/skills/weekly-trade-strategy/skills/market-environment-analysis/scripts/market_utils.py +127 -0
  330. package/.claude/skills/weekly-trade-strategy/skills/market-news-analyst/SKILL.md +714 -0
  331. package/.claude/skills/weekly-trade-strategy/skills/market-news-analyst/references/corporate_news_impact.md +446 -0
  332. package/.claude/skills/weekly-trade-strategy/skills/market-news-analyst/references/geopolitical_commodity_correlations.md +499 -0
  333. package/.claude/skills/weekly-trade-strategy/skills/market-news-analyst/references/market_event_patterns.md +393 -0
  334. package/.claude/skills/weekly-trade-strategy/skills/market-news-analyst/references/trusted_news_sources.md +510 -0
  335. package/.claude/skills/weekly-trade-strategy/skills/sector-analyst/SKILL.md +206 -0
  336. package/.claude/skills/weekly-trade-strategy/skills/sector-analyst/assets/industory_performance_1.jpeg +0 -0
  337. package/.claude/skills/weekly-trade-strategy/skills/sector-analyst/assets/industory_performance_2.jpeg +0 -0
  338. package/.claude/skills/weekly-trade-strategy/skills/sector-analyst/assets/sector_performance.jpeg +0 -0
  339. package/.claude/skills/weekly-trade-strategy/skills/sector-analyst/references/sector_rotation.md +170 -0
  340. package/.claude/skills/weekly-trade-strategy/skills/stanley-druckenmiller-investment/SKILL.md +84 -0
  341. package/.claude/skills/weekly-trade-strategy/skills/stanley-druckenmiller-investment/references/case-studies.md +148 -0
  342. package/.claude/skills/weekly-trade-strategy/skills/stanley-druckenmiller-investment/references/investment-philosophy.md +80 -0
  343. package/.claude/skills/weekly-trade-strategy/skills/stanley-druckenmiller-investment/references/market-analysis-guide.md +146 -0
  344. package/.claude/skills/weekly-trade-strategy/skills/technical-analyst/SKILL.md +238 -0
  345. package/.claude/skills/weekly-trade-strategy/skills/technical-analyst/assets/analysis_template.md +183 -0
  346. package/.claude/skills/weekly-trade-strategy/skills/technical-analyst/references/technical_analysis_framework.md +282 -0
  347. package/.claude/skills/weekly-trade-strategy/skills/us-market-bubble-detector/CHANGELOG.md +118 -0
  348. package/.claude/skills/weekly-trade-strategy/skills/us-market-bubble-detector/SKILL.md +545 -0
  349. package/.claude/skills/weekly-trade-strategy/skills/us-market-bubble-detector/references/bubble_framework.md +335 -0
  350. package/.claude/skills/weekly-trade-strategy/skills/us-market-bubble-detector/references/historical_cases.md +327 -0
  351. package/.claude/skills/weekly-trade-strategy/skills/us-market-bubble-detector/references/implementation_guide.md +473 -0
  352. package/.claude/skills/weekly-trade-strategy/skills/us-market-bubble-detector/references/quick_reference.md +354 -0
  353. package/.claude/skills/weekly-trade-strategy/skills/us-market-bubble-detector/references/quick_reference_en.md +342 -0
  354. package/.claude/skills/weekly-trade-strategy/skills/us-market-bubble-detector/scripts/bubble_scorer.py +309 -0
  355. package/.claude/skills/weekly-trade-strategy/skills/us-stock-analysis/SKILL.md +294 -0
  356. package/.claude/skills/weekly-trade-strategy/skills/us-stock-analysis/references/financial-metrics.md +172 -0
  357. package/.claude/skills/weekly-trade-strategy/skills/us-stock-analysis/references/fundamental-analysis.md +129 -0
  358. package/.claude/skills/weekly-trade-strategy/skills/us-stock-analysis/references/report-template.md +207 -0
  359. package/.claude/skills/weekly-trade-strategy/skills/us-stock-analysis/references/technical-analysis.md +93 -0
  360. package/.mcp.json +3 -0
  361. package/cli.mjs +16 -16
  362. package/package.json +4 -2
@@ -0,0 +1,405 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ CANSLIM Report Generator - Phase 3 (Full CANSLIM)
4
+
5
+ Generates JSON and Markdown reports for CANSLIM screening results.
6
+
7
+ Outputs:
8
+ - JSON: Structured data for programmatic use
9
+ - Markdown: Human-readable ranked list with component breakdowns
10
+ """
11
+
12
+ import json
13
+ from datetime import datetime
14
+ from typing import Dict, List
15
+
16
+
17
+ def generate_json_report(results: List[Dict], metadata: Dict, output_file: str):
18
+ """
19
+ Generate JSON report with screening results
20
+
21
+ Args:
22
+ results: List of analyzed stocks with scores
23
+ metadata: Screening metadata (date, parameters, etc.)
24
+ output_file: Output file path
25
+ """
26
+ report = {
27
+ "metadata": metadata,
28
+ "results": results,
29
+ "summary": generate_summary_stats(results)
30
+ }
31
+
32
+ with open(output_file, 'w') as f:
33
+ json.dump(report, f, indent=2)
34
+
35
+ print(f"✓ JSON report saved to: {output_file}")
36
+
37
+
38
+ def generate_markdown_report(results: List[Dict], metadata: Dict, output_file: str):
39
+ """
40
+ Generate Markdown report with screening results
41
+
42
+ Args:
43
+ results: List of analyzed stocks with scores
44
+ metadata: Screening metadata
45
+ output_file: Output file path
46
+ """
47
+ lines = []
48
+
49
+ # Header
50
+ lines.append("# CANSLIM Stock Screener Report - Phase 3 (Full CANSLIM)")
51
+ lines.append(f"**Generated:** {metadata['generated_at']}")
52
+
53
+ # Extract components from metadata
54
+ components = metadata.get('components_included', ['C', 'A', 'N', 'S', 'I', 'M'])
55
+ components_str = ', '.join(components)
56
+ lines.append(f"**Phase:** {metadata['phase']} (Components: {components_str})")
57
+ lines.append(f"**Stocks Analyzed:** {metadata['candidates_analyzed']}")
58
+ lines.append("")
59
+ lines.append("---")
60
+ lines.append("")
61
+
62
+ # Market condition summary
63
+ if 'market_condition' in metadata:
64
+ market = metadata['market_condition']
65
+ lines.append("## Market Condition Summary")
66
+ lines.append(f"- **Trend:** {market.get('trend', 'unknown')}")
67
+ lines.append(f"- **M Score:** {market.get('M_score', 'N/A')}/100")
68
+
69
+ if market.get('warning'):
70
+ lines.append(f"- **⚠️ Warning:** {market['warning']}")
71
+
72
+ lines.append("")
73
+ lines.append("---")
74
+ lines.append("")
75
+
76
+ # Top candidates
77
+ lines.append(f"## Top {min(len(results), 20)} CANSLIM Candidates")
78
+ lines.append("")
79
+
80
+ for i, stock in enumerate(results[:20], 1):
81
+ lines.extend(format_stock_entry(i, stock))
82
+
83
+ # Summary statistics
84
+ lines.append("---")
85
+ lines.append("")
86
+ lines.append("## Summary Statistics")
87
+ summary = generate_summary_stats(results)
88
+ lines.append(f"- **Total Stocks Screened:** {summary['total_stocks']}")
89
+ lines.append(f"- **Exceptional (90+):** {summary['exceptional']} stocks")
90
+ lines.append(f"- **Strong (80-89):** {summary['strong']} stocks")
91
+ lines.append(f"- **Above Average (70-79):** {summary['above_average']} stocks")
92
+ lines.append(f"- **Average (60-69):** {summary['average']} stocks")
93
+ lines.append(f"- **Below Average (<60):** {summary['below_average']} stocks")
94
+ lines.append("")
95
+
96
+ # Methodology note
97
+ lines.append("---")
98
+ lines.append("")
99
+ lines.append("## Methodology")
100
+ lines.append("")
101
+ lines.append("This Phase 3 implementation includes all 7 CANSLIM components (100% coverage):")
102
+ lines.append("")
103
+ lines.append("- **C** (Current Earnings) - 15% weight: Quarterly EPS growth YoY")
104
+ lines.append("- **A** (Annual Growth) - 20% weight: 3-year EPS CAGR")
105
+ lines.append("- **N** (Newness) - 15% weight: Price position vs 52-week high")
106
+ lines.append("- **S** (Supply/Demand) - 15% weight: Volume accumulation/distribution")
107
+ lines.append("- **L** (Leadership) - 20% weight: Relative Strength vs S&P 500 (52-week)")
108
+ lines.append("- **I** (Institutional) - 10% weight: Institutional holder analysis")
109
+ lines.append("- **M** (Market Direction) - 5% weight: S&P 500 trend")
110
+ lines.append("")
111
+ lines.append("Component weights follow William O'Neil's original CANSLIM methodology,")
112
+ lines.append("with L (Leadership/RS Rank) as the most weighted component alongside A (Annual Growth).")
113
+ lines.append("")
114
+ lines.append("For detailed methodology, see `references/canslim_methodology.md`.")
115
+ lines.append("")
116
+
117
+ # Disclaimer
118
+ lines.append("---")
119
+ lines.append("")
120
+ lines.append("**Disclaimer:** This screener is for educational and informational purposes only. "
121
+ "Not investment advice. Conduct your own research and consult a financial advisor "
122
+ "before making investment decisions.")
123
+ lines.append("")
124
+
125
+ # Write to file
126
+ with open(output_file, 'w') as f:
127
+ f.write('\n'.join(lines))
128
+
129
+ print(f"✓ Markdown report saved to: {output_file}")
130
+
131
+
132
+ def format_stock_entry(rank: int, stock: Dict) -> List[str]:
133
+ """
134
+ Format a single stock entry for Markdown report
135
+
136
+ Args:
137
+ rank: Stock rank (1-20)
138
+ stock: Stock data dict
139
+
140
+ Returns:
141
+ List of formatted lines
142
+ """
143
+ lines = []
144
+
145
+ # Header with rank and rating emoji
146
+ rating_emoji = get_rating_emoji(stock['composite_score'])
147
+ lines.append(f"### {rank}. {stock['symbol']} - {stock['company_name']} {rating_emoji}")
148
+
149
+ # Basic info
150
+ lines.append(f"**Price:** ${stock.get('price', 'N/A'):.2f} | "
151
+ f"**Market Cap:** ${stock.get('market_cap', 0)/1e9:.1f}B | "
152
+ f"**Sector:** {stock.get('sector', 'N/A')}")
153
+
154
+ # Composite score
155
+ lines.append(f"**Composite Score:** {stock['composite_score']:.1f}/100 ({stock['rating']})")
156
+ lines.append("")
157
+
158
+ # Component breakdown table
159
+ lines.append("#### Component Breakdown")
160
+ lines.append("")
161
+ lines.append("| Component | Score | Details |")
162
+ lines.append("|-----------|-------|---------|")
163
+
164
+ # C component
165
+ c_details = stock.get('c_component', {})
166
+ c_score = c_details.get('score', 0)
167
+ c_eps = c_details.get('latest_qtr_eps_growth', 'N/A')
168
+ c_rev = c_details.get('latest_qtr_revenue_growth', 'N/A')
169
+ c_eps_str = f"{c_eps:+.1f}%" if isinstance(c_eps, (int, float)) else str(c_eps)
170
+ c_rev_str = f"{c_rev:+.1f}%" if isinstance(c_rev, (int, float)) else str(c_rev)
171
+ lines.append(f"| 🅲 Current Earnings | {c_score}/100 | "
172
+ f"EPS: {c_eps_str}, Revenue: {c_rev_str} |")
173
+
174
+ # A component
175
+ a_details = stock.get('a_component', {})
176
+ a_score = a_details.get('score', 0)
177
+ a_cagr = a_details.get('eps_cagr_3yr', 'N/A')
178
+ a_stability = a_details.get('stability', 'unknown')
179
+ a_cagr_str = f"{a_cagr:.1f}%" if isinstance(a_cagr, (int, float)) else str(a_cagr)
180
+ lines.append(f"| 🅰 Annual Growth | {a_score}/100 | "
181
+ f"3yr CAGR: {a_cagr_str}, {a_stability} |")
182
+
183
+ # N component
184
+ n_details = stock.get('n_component', {})
185
+ n_score = n_details.get('score', 0)
186
+ n_distance = n_details.get('distance_from_high_pct', 'N/A')
187
+ n_breakout = "✓ Breakout" if n_details.get('breakout_detected') else ""
188
+ n_distance_str = f"{n_distance:+.1f}%" if isinstance(n_distance, (int, float)) else str(n_distance)
189
+ lines.append(f"| 🅽 Newness | {n_score}/100 | "
190
+ f"{n_distance_str} from 52wk high {n_breakout} |")
191
+
192
+ # S component
193
+ s_details = stock.get('s_component', {})
194
+ s_score = s_details.get('score', 0)
195
+ s_ratio = s_details.get('up_down_ratio', 'N/A')
196
+ s_accumulation = "✓ Accumulation" if s_details.get('accumulation_detected') else ""
197
+ s_ratio_str = f"{s_ratio:.2f}" if isinstance(s_ratio, (int, float)) else str(s_ratio)
198
+ lines.append(f"| 🅂 Supply/Demand | {s_score}/100 | "
199
+ f"Up/Down Volume Ratio: {s_ratio_str} {s_accumulation} |")
200
+
201
+ # L component (Phase 3 - Leadership / Relative Strength)
202
+ l_details = stock.get('l_component', {})
203
+ l_score = l_details.get('score', 0)
204
+ l_stock_perf = l_details.get('stock_52w_performance', 'N/A')
205
+ l_relative = l_details.get('relative_performance', 'N/A')
206
+ l_rs_rank = l_details.get('rs_rank_estimate', 'N/A')
207
+ l_stock_perf_str = f"{l_stock_perf:+.1f}%" if isinstance(l_stock_perf, (int, float)) else str(l_stock_perf)
208
+ l_relative_str = f"{l_relative:+.1f}% vs S&P" if isinstance(l_relative, (int, float)) else str(l_relative)
209
+ l_rs_str = f"RS: {l_rs_rank}" if isinstance(l_rs_rank, (int, float)) else ""
210
+ lines.append(f"| 🅻 Leadership | {l_score}/100 | "
211
+ f"52wk: {l_stock_perf_str} ({l_relative_str}) {l_rs_str} |")
212
+
213
+ # I component
214
+ i_details = stock.get('i_component', {})
215
+ i_score = i_details.get('score', 0)
216
+ i_holders = i_details.get('num_holders', 'N/A')
217
+ i_ownership = i_details.get('ownership_pct', 'N/A')
218
+ i_ownership_str = f"{i_ownership:.1f}%" if isinstance(i_ownership, (int, float)) else str(i_ownership)
219
+ i_superinvestor = "⭐ Superinvestor" if i_details.get('superinvestor_present') else ""
220
+ lines.append(f"| 🅸 Institutional | {i_score}/100 | "
221
+ f"{i_holders} holders, {i_ownership_str} ownership {i_superinvestor} |")
222
+
223
+ # M component
224
+ m_details = stock.get('m_component', {})
225
+ m_score = m_details.get('score', 0)
226
+ m_trend = m_details.get('trend', 'unknown')
227
+ lines.append(f"| 🅼 Market Direction | {m_score}/100 | {m_trend.replace('_', ' ').title()} |")
228
+
229
+ lines.append("")
230
+
231
+ # Interpretation
232
+ lines.append("#### Interpretation")
233
+ lines.append(f"**Rating:** {stock['rating']} - {stock['rating_description']}")
234
+ lines.append("")
235
+ lines.append(f"**Guidance:** {stock['guidance']}")
236
+ lines.append("")
237
+
238
+ # Weakest component
239
+ lines.append(f"**Weakest Component:** {stock['weakest_component']} "
240
+ f"({stock['weakest_score']}/100)")
241
+
242
+ # Warnings
243
+ warnings = []
244
+ if c_details.get('quality_warning'):
245
+ warnings.append(f"⚠️ {c_details['quality_warning']}")
246
+ if a_details.get('quality_warning'):
247
+ warnings.append(f"⚠️ {a_details['quality_warning']}")
248
+ if s_details.get('quality_warning'):
249
+ warnings.append(f"⚠️ {s_details['quality_warning']}")
250
+ if l_details.get('quality_warning'):
251
+ warnings.append(f"⚠️ {l_details['quality_warning']}")
252
+ if i_details.get('quality_warning'):
253
+ warnings.append(f"⚠️ {i_details['quality_warning']}")
254
+ if m_details.get('warning'):
255
+ warnings.append(f"⚠️ {m_details['warning']}")
256
+
257
+ if warnings:
258
+ lines.append("")
259
+ lines.append("**Warnings:**")
260
+ for warning in warnings:
261
+ lines.append(f"- {warning}")
262
+
263
+ lines.append("")
264
+ lines.append("---")
265
+ lines.append("")
266
+
267
+ return lines
268
+
269
+
270
+ def get_rating_emoji(score: float) -> str:
271
+ """Get emoji for rating"""
272
+ if score >= 90:
273
+ return "⭐⭐⭐" # Exceptional+
274
+ elif score >= 80:
275
+ return "⭐⭐" # Exceptional
276
+ elif score >= 70:
277
+ return "⭐" # Strong
278
+ elif score >= 60:
279
+ return "✓" # Above Average
280
+ else:
281
+ return ""
282
+
283
+
284
+ def generate_summary_stats(results: List[Dict]) -> Dict:
285
+ """
286
+ Generate summary statistics for results
287
+
288
+ Args:
289
+ results: List of analyzed stocks
290
+
291
+ Returns:
292
+ Dict with summary statistics
293
+ """
294
+ total = len(results)
295
+
296
+ exceptional = sum(1 for s in results if s['composite_score'] >= 90)
297
+ strong = sum(1 for s in results if 80 <= s['composite_score'] < 90)
298
+ above_avg = sum(1 for s in results if 70 <= s['composite_score'] < 80)
299
+ average = sum(1 for s in results if 60 <= s['composite_score'] < 70)
300
+ below_avg = sum(1 for s in results if s['composite_score'] < 60)
301
+
302
+ return {
303
+ "total_stocks": total,
304
+ "exceptional": exceptional,
305
+ "strong": strong,
306
+ "above_average": above_avg,
307
+ "average": average,
308
+ "below_average": below_avg
309
+ }
310
+
311
+
312
+ # Example usage
313
+ if __name__ == "__main__":
314
+ print("Testing Report Generator...\n")
315
+
316
+ # Sample data
317
+ sample_results = [
318
+ {
319
+ "symbol": "NVDA",
320
+ "company_name": "NVIDIA Corporation",
321
+ "price": 495.50,
322
+ "market_cap": 1220000000000,
323
+ "sector": "Technology",
324
+ "composite_score": 97.2,
325
+ "rating": "Exceptional+",
326
+ "rating_description": "Rare multi-bagger setup",
327
+ "guidance": "Immediate buy, aggressive sizing",
328
+ "weakest_component": "A",
329
+ "weakest_score": 95,
330
+ "c_component": {
331
+ "score": 100,
332
+ "latest_qtr_eps_growth": 429,
333
+ "latest_qtr_revenue_growth": 101
334
+ },
335
+ "a_component": {
336
+ "score": 95,
337
+ "eps_cagr_3yr": 76,
338
+ "stability": "stable"
339
+ },
340
+ "n_component": {
341
+ "score": 98,
342
+ "distance_from_high_pct": -0.5,
343
+ "breakout_detected": True
344
+ },
345
+ "m_component": {
346
+ "score": 100,
347
+ "trend": "strong_uptrend"
348
+ }
349
+ },
350
+ {
351
+ "symbol": "META",
352
+ "company_name": "Meta Platforms Inc",
353
+ "price": 389.50,
354
+ "market_cap": 1000000000000,
355
+ "sector": "Technology",
356
+ "composite_score": 82.8,
357
+ "rating": "Exceptional",
358
+ "rating_description": "Outstanding fundamentals",
359
+ "guidance": "Strong buy, standard sizing",
360
+ "weakest_component": "A",
361
+ "weakest_score": 78,
362
+ "c_component": {
363
+ "score": 85,
364
+ "latest_qtr_eps_growth": 164,
365
+ "latest_qtr_revenue_growth": 23
366
+ },
367
+ "a_component": {
368
+ "score": 78,
369
+ "eps_cagr_3yr": 28,
370
+ "stability": "stable"
371
+ },
372
+ "n_component": {
373
+ "score": 88,
374
+ "distance_from_high_pct": -5.0,
375
+ "breakout_detected": False
376
+ },
377
+ "m_component": {
378
+ "score": 80,
379
+ "trend": "uptrend"
380
+ }
381
+ }
382
+ ]
383
+
384
+ sample_metadata = {
385
+ "generated_at": datetime.now().strftime("%Y-%m-%d %H:%M:%S UTC"),
386
+ "phase": "1 (MVP)",
387
+ "components_included": ["C", "A", "N", "M"],
388
+ "candidates_analyzed": 40,
389
+ "market_condition": {
390
+ "trend": "uptrend",
391
+ "M_score": 80,
392
+ "warning": None
393
+ }
394
+ }
395
+
396
+ # Generate reports
397
+ json_file = "test_canslim_report.json"
398
+ md_file = "test_canslim_report.md"
399
+
400
+ generate_json_report(sample_results, sample_metadata, json_file)
401
+ generate_markdown_report(sample_results, sample_metadata, md_file)
402
+
403
+ print("\n✓ Test reports generated successfully")
404
+ print(f" JSON: {json_file}")
405
+ print(f" Markdown: {md_file}")