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,624 @@
1
+ """Tests for report_generator module."""
2
+
3
+ import json
4
+ import os
5
+ import tempfile
6
+ from datetime import datetime
7
+
8
+ import pytest
9
+
10
+ from report_generator import (
11
+ generate_json_report,
12
+ generate_markdown_report,
13
+ save_reports,
14
+ _fmt_pct,
15
+ _format_stock_list,
16
+ _origin_label,
17
+ )
18
+
19
+
20
+ # --- Fixtures ---
21
+
22
+ @pytest.fixture
23
+ def sample_themes():
24
+ """Sample theme data for testing (0-100 scale for heat/maturity)."""
25
+ return [
26
+ {
27
+ "name": "AI / Semiconductors",
28
+ "direction": "bullish",
29
+ "heat": 85.0,
30
+ "maturity": 62.0,
31
+ "stage": "growth",
32
+ "confidence": "High",
33
+ "industries": ["Semiconductors", "Software - Infrastructure",
34
+ "Information Technology Services"],
35
+ "heat_breakdown": {
36
+ "performance_momentum": 75.0,
37
+ "volume_confirmation": 80.0,
38
+ "breadth_score": 90.0,
39
+ },
40
+ "maturity_breakdown": {
41
+ "duration_score": 60.0,
42
+ "crowding_score": 55.0,
43
+ "acceleration": 70.0,
44
+ },
45
+ "representative_stocks": ["NVDA", "AVGO", "AMD", "MSFT"],
46
+ "proxy_etfs": ["SMH", "SOXX", "XLK"],
47
+ },
48
+ {
49
+ "name": "Energy Transition",
50
+ "direction": "bullish",
51
+ "heat": 63.0,
52
+ "maturity": 41.0,
53
+ "stage": "early",
54
+ "confidence": "Medium",
55
+ "industries": ["Solar", "Uranium"],
56
+ "heat_breakdown": {
57
+ "performance_momentum": 60.0,
58
+ "volume_confirmation": 55.0,
59
+ "breadth_score": 70.0,
60
+ },
61
+ "maturity_breakdown": {},
62
+ "representative_stocks": ["FSLR", "ENPH"],
63
+ "proxy_etfs": ["TAN", "URA"],
64
+ },
65
+ {
66
+ "name": "Traditional Retail",
67
+ "direction": "bearish",
68
+ "heat": 58.0,
69
+ "maturity": 75.0,
70
+ "stage": "decline",
71
+ "confidence": "Low",
72
+ "industries": ["Specialty Retail", "Department Stores"],
73
+ "heat_breakdown": {
74
+ "performance_momentum": -40.0,
75
+ "volume_confirmation": 30.0,
76
+ "breadth_score": 25.0,
77
+ },
78
+ "maturity_breakdown": {},
79
+ "representative_stocks": ["M", "KSS"],
80
+ "proxy_etfs": ["XRT"],
81
+ },
82
+ ]
83
+
84
+
85
+ @pytest.fixture
86
+ def sample_industry_rankings():
87
+ """Sample industry rankings (perf values in percent, score is momentum_score)."""
88
+ return {
89
+ "top": [
90
+ {"name": "Semiconductors", "perf_1w": 5.0, "perf_1m": 12.0,
91
+ "perf_3m": 25.0, "momentum_score": 0.89},
92
+ {"name": "Software - Infrastructure", "perf_1w": 3.0,
93
+ "perf_1m": 8.0, "perf_3m": 18.0, "momentum_score": 0.75},
94
+ ],
95
+ "bottom": [
96
+ {"name": "Department Stores", "perf_1w": -4.0, "perf_1m": -10.0,
97
+ "perf_3m": -15.0, "momentum_score": -0.65},
98
+ {"name": "Specialty Retail", "perf_1w": -3.0, "perf_1m": -7.0,
99
+ "perf_3m": -12.0, "momentum_score": -0.50},
100
+ ],
101
+ }
102
+
103
+
104
+ @pytest.fixture
105
+ def sample_sector_uptrend():
106
+ """Sample sector uptrend data."""
107
+ return {
108
+ "Technology": {
109
+ "ratio": 0.35,
110
+ "ma_10": 0.32,
111
+ "slope": 0.0025,
112
+ "trend": "up",
113
+ "latest_date": "2026-02-14",
114
+ },
115
+ "Healthcare": {
116
+ "ratio": 0.22,
117
+ "ma_10": 0.24,
118
+ "slope": -0.0015,
119
+ "trend": "down",
120
+ "latest_date": "2026-02-14",
121
+ },
122
+ }
123
+
124
+
125
+ @pytest.fixture
126
+ def sample_metadata():
127
+ """Sample metadata."""
128
+ return {
129
+ "generated_at": "2026-02-16 10:00:00",
130
+ "data_sources": {
131
+ "finviz": "ok",
132
+ "uptrend": "ok",
133
+ },
134
+ }
135
+
136
+
137
+ @pytest.fixture
138
+ def sample_json_report(sample_themes, sample_industry_rankings,
139
+ sample_sector_uptrend, sample_metadata):
140
+ """Full JSON report generated from sample data."""
141
+ return generate_json_report(
142
+ sample_themes, sample_industry_rankings,
143
+ sample_sector_uptrend, sample_metadata,
144
+ )
145
+
146
+
147
+ # --- Tests for generate_json_report ---
148
+
149
+ class TestGenerateJsonReport:
150
+
151
+ def test_report_structure(self, sample_json_report):
152
+ """JSON report has all required top-level keys."""
153
+ required_keys = [
154
+ "report_type", "generated_at", "metadata", "summary",
155
+ "themes", "industry_rankings", "sector_uptrend", "data_quality",
156
+ ]
157
+ for key in required_keys:
158
+ assert key in sample_json_report, f"Missing key: {key}"
159
+
160
+ def test_report_type(self, sample_json_report):
161
+ assert sample_json_report["report_type"] == "theme_detector"
162
+
163
+ def test_summary_counts(self, sample_json_report):
164
+ summary = sample_json_report["summary"]
165
+ assert summary["total_themes"] == 3
166
+ assert summary["bullish_count"] == 2
167
+ assert summary["bearish_count"] == 1
168
+ assert summary["top_bullish"] == "AI / Semiconductors"
169
+ assert summary["top_bearish"] == "Traditional Retail"
170
+
171
+ def test_themes_grouped(self, sample_json_report):
172
+ themes = sample_json_report["themes"]
173
+ assert "all" in themes
174
+ assert "bullish" in themes
175
+ assert "bearish" in themes
176
+ assert len(themes["all"]) == 3
177
+ assert len(themes["bullish"]) == 2
178
+ assert len(themes["bearish"]) == 1
179
+
180
+ def test_bullish_sorted_by_heat(self, sample_json_report):
181
+ bullish = sample_json_report["themes"]["bullish"]
182
+ heats = [t["heat"] for t in bullish]
183
+ assert heats == sorted(heats, reverse=True)
184
+
185
+ def test_data_quality_ok(self, sample_json_report):
186
+ dq = sample_json_report["data_quality"]
187
+ assert dq["status"] == "ok"
188
+ assert dq["flags"] == []
189
+
190
+ def test_empty_themes(self, sample_industry_rankings,
191
+ sample_sector_uptrend, sample_metadata):
192
+ """Empty themes list produces warning, not error."""
193
+ report = generate_json_report(
194
+ [], sample_industry_rankings,
195
+ sample_sector_uptrend, sample_metadata,
196
+ )
197
+ assert report["summary"]["total_themes"] == 0
198
+ assert report["summary"]["top_bullish"] is None
199
+ assert report["data_quality"]["status"] == "warning"
200
+ assert any("No themes" in f for f in report["data_quality"]["flags"])
201
+
202
+ def test_json_serializable(self, sample_json_report):
203
+ """Report can be serialized to JSON."""
204
+ serialized = json.dumps(sample_json_report, default=str)
205
+ deserialized = json.loads(serialized)
206
+ assert deserialized["report_type"] == "theme_detector"
207
+
208
+
209
+ # --- Tests for generate_markdown_report ---
210
+
211
+ class TestGenerateMarkdownReport:
212
+
213
+ def test_contains_all_sections(self, sample_json_report):
214
+ """Markdown contains all 7 required sections."""
215
+ md = generate_markdown_report(sample_json_report)
216
+ assert "## 1. Theme Dashboard" in md
217
+ assert "## 2. Bullish Themes (Top 3)" in md
218
+ assert "## 3. Bearish Themes (Top 3)" in md
219
+ assert "## 4. All Themes Summary" in md
220
+ assert "## 5. Industry Rankings" in md
221
+ assert "## 6. Sector Uptrend Ratios" in md
222
+ assert "## 7. Methodology & Data Quality" in md
223
+
224
+ def test_header_info(self, sample_json_report):
225
+ md = generate_markdown_report(sample_json_report)
226
+ assert "# Theme Detector Report" in md
227
+ assert "2026-02-16 10:00:00" in md
228
+
229
+ def test_theme_dashboard_table(self, sample_json_report):
230
+ md = generate_markdown_report(sample_json_report)
231
+ assert "AI / Semiconductors" in md
232
+ assert "BULL" in md
233
+ assert "BEAR" in md
234
+
235
+ def test_bullish_detail_stocks(self, sample_json_report):
236
+ md = generate_markdown_report(sample_json_report)
237
+ assert "NVDA" in md
238
+ assert "SMH" in md
239
+
240
+ def test_industry_rankings_present(self, sample_json_report):
241
+ md = generate_markdown_report(sample_json_report)
242
+ assert "Semiconductors" in md
243
+ assert "Department Stores" in md
244
+ assert "Top 15" in md
245
+ assert "Bottom 15" in md
246
+
247
+ def test_sector_uptrend_table(self, sample_json_report):
248
+ md = generate_markdown_report(sample_json_report)
249
+ assert "Technology" in md
250
+ assert "Healthcare" in md
251
+ assert "35.0%" in md # Technology ratio
252
+
253
+ def test_methodology_present(self, sample_json_report):
254
+ md = generate_markdown_report(sample_json_report)
255
+ assert "Methodology" in md
256
+ assert "Disclaimer" in md
257
+
258
+ def test_empty_themes_warning(self, sample_industry_rankings,
259
+ sample_sector_uptrend, sample_metadata):
260
+ """Empty themes show warning, not crash."""
261
+ report = generate_json_report(
262
+ [], sample_industry_rankings,
263
+ sample_sector_uptrend, sample_metadata,
264
+ )
265
+ md = generate_markdown_report(report)
266
+ assert "WARNING" in md or "No themes" in md
267
+
268
+ def test_no_industry_data(self, sample_themes, sample_sector_uptrend,
269
+ sample_metadata):
270
+ """Missing industry rankings handled gracefully."""
271
+ report = generate_json_report(
272
+ sample_themes, {"top": [], "bottom": []},
273
+ sample_sector_uptrend, sample_metadata,
274
+ )
275
+ md = generate_markdown_report(report)
276
+ assert "## 5. Industry Rankings" in md
277
+ assert "unavailable" in md.lower() or "Industry" in md
278
+
279
+ def test_top_n_detail_default_3(self, sample_json_report):
280
+ """Default top_n_detail=3 shows 'Top 3' in section headers."""
281
+ md = generate_markdown_report(sample_json_report)
282
+ assert "## 2. Bullish Themes (Top 3)" in md
283
+ assert "## 3. Bearish Themes (Top 3)" in md
284
+
285
+ def test_top_n_detail_custom(self, sample_json_report):
286
+ """Custom top_n_detail changes section headers and limits themes."""
287
+ md = generate_markdown_report(sample_json_report, top_n_detail=5)
288
+ assert "## 2. Bullish Themes (Top 5)" in md
289
+ assert "## 3. Bearish Themes (Top 5)" in md
290
+
291
+ def test_top_n_detail_1(self, sample_json_report):
292
+ """top_n_detail=1 only shows 1 theme per direction."""
293
+ md = generate_markdown_report(sample_json_report, top_n_detail=1)
294
+ assert "## 2. Bullish Themes (Top 1)" in md
295
+ # Only the top bullish theme detail section should appear
296
+ # AI / Semiconductors (heat 85) should appear, Energy Transition (heat 63) should not
297
+ assert "### AI / Semiconductors" in md
298
+ assert "### Energy Transition" not in md
299
+
300
+
301
+ # --- Tests for save_reports ---
302
+
303
+ class TestSaveReports:
304
+
305
+ def test_save_creates_files(self, sample_json_report):
306
+ """save_reports creates both JSON and MD files."""
307
+ md = generate_markdown_report(sample_json_report)
308
+ with tempfile.TemporaryDirectory() as tmpdir:
309
+ paths = save_reports(sample_json_report, md, tmpdir)
310
+ assert os.path.exists(paths["json"])
311
+ assert os.path.exists(paths["markdown"])
312
+
313
+ def test_filename_convention(self, sample_json_report):
314
+ """Filenames follow theme_detector_YYYY-MM-DD_HHMMSS pattern."""
315
+ md = generate_markdown_report(sample_json_report)
316
+ with tempfile.TemporaryDirectory() as tmpdir:
317
+ paths = save_reports(sample_json_report, md, tmpdir)
318
+ json_name = os.path.basename(paths["json"])
319
+ md_name = os.path.basename(paths["markdown"])
320
+ assert json_name.startswith("theme_detector_")
321
+ assert json_name.endswith(".json")
322
+ assert md_name.startswith("theme_detector_")
323
+ assert md_name.endswith(".md")
324
+
325
+ def test_json_file_valid(self, sample_json_report):
326
+ """Saved JSON file is valid JSON."""
327
+ md = generate_markdown_report(sample_json_report)
328
+ with tempfile.TemporaryDirectory() as tmpdir:
329
+ paths = save_reports(sample_json_report, md, tmpdir)
330
+ with open(paths["json"]) as f:
331
+ loaded = json.load(f)
332
+ assert loaded["report_type"] == "theme_detector"
333
+
334
+ def test_creates_output_dir(self, sample_json_report):
335
+ """save_reports creates output dir if it doesn't exist."""
336
+ md = generate_markdown_report(sample_json_report)
337
+ with tempfile.TemporaryDirectory() as tmpdir:
338
+ nested = os.path.join(tmpdir, "reports", "sub")
339
+ paths = save_reports(sample_json_report, md, nested)
340
+ assert os.path.exists(paths["json"])
341
+
342
+
343
+ # --- Tests for _fmt_pct ---
344
+
345
+ class TestFmtPct:
346
+ """_fmt_pct formats percent values directly (no *100 conversion)."""
347
+
348
+ def test_positive_value(self):
349
+ assert _fmt_pct(5.0) == "+5.0%"
350
+
351
+ def test_negative_value(self):
352
+ assert _fmt_pct(-3.2) == "-3.2%"
353
+
354
+ def test_zero(self):
355
+ assert _fmt_pct(0.0) == "+0.0%"
356
+
357
+ def test_none_returns_na(self):
358
+ assert _fmt_pct(None) == "N/A"
359
+
360
+ def test_large_value(self):
361
+ assert _fmt_pct(25.7) == "+25.7%"
362
+
363
+ def test_small_negative(self):
364
+ assert _fmt_pct(-0.5) == "-0.5%"
365
+
366
+
367
+ # --- Tests for Industry Rankings key usage ---
368
+
369
+ class TestIndustryRankingsKey:
370
+ """Verify industry rankings use momentum_score, not composite_score."""
371
+
372
+ def test_top_industries_show_momentum_score(self, sample_json_report):
373
+ md = generate_markdown_report(sample_json_report)
374
+ # momentum_score=0.89 should appear, not composite_score
375
+ assert "0.89" in md
376
+
377
+ def test_bottom_industries_show_momentum_score(self, sample_json_report):
378
+ md = generate_markdown_report(sample_json_report)
379
+ assert "-0.65" in md
380
+
381
+ def test_industry_perf_not_multiplied(self, sample_json_report):
382
+ """Perf values should appear as-is (already in percent)."""
383
+ md = generate_markdown_report(sample_json_report)
384
+ # 5.0% should appear, not 500.0%
385
+ assert "+5.0%" in md
386
+ assert "500.0%" not in md
387
+
388
+
389
+ # --- Tests for _format_stock_list ---
390
+
391
+ class TestFormatStockList:
392
+
393
+ def test_no_details_uses_tickers(self):
394
+ """Without stock_details, plain tickers are shown."""
395
+ theme = {"representative_stocks": ["NVDA", "AMD", "AVGO"]}
396
+ assert _format_stock_list(theme) == "NVDA, AMD, AVGO"
397
+
398
+ def test_empty_stocks_returns_na(self):
399
+ theme = {"representative_stocks": []}
400
+ assert _format_stock_list(theme) == "N/A"
401
+
402
+ def test_finviz_public_labels(self):
403
+ """stock_details with finviz_public source shows [Fp] labels."""
404
+ theme = {
405
+ "representative_stocks": ["NEM", "GOLD"],
406
+ "stock_details": [
407
+ {"symbol": "NEM", "source": "finviz_public"},
408
+ {"symbol": "GOLD", "source": "finviz_public"},
409
+ ],
410
+ }
411
+ result = _format_stock_list(theme)
412
+ assert "NEM[Fp]" in result
413
+ assert "GOLD[Fp]" in result
414
+
415
+ def test_finviz_elite_labels(self):
416
+ theme = {
417
+ "representative_stocks": ["NVDA"],
418
+ "stock_details": [
419
+ {"symbol": "NVDA", "source": "finviz_elite"},
420
+ ],
421
+ }
422
+ assert "NVDA[Fe]" in _format_stock_list(theme)
423
+
424
+ def test_etf_holdings_labels(self):
425
+ theme = {
426
+ "representative_stocks": ["AAPL"],
427
+ "stock_details": [
428
+ {"symbol": "AAPL", "source": "etf_holdings"},
429
+ ],
430
+ }
431
+ assert "AAPL[E]" in _format_stock_list(theme)
432
+
433
+ def test_static_labels(self):
434
+ theme = {
435
+ "representative_stocks": ["XOM"],
436
+ "stock_details": [
437
+ {"symbol": "XOM", "source": "static"},
438
+ ],
439
+ }
440
+ assert "XOM[S]" in _format_stock_list(theme)
441
+
442
+ def test_mixed_sources(self):
443
+ theme = {
444
+ "representative_stocks": ["NVDA", "AAPL", "XOM"],
445
+ "stock_details": [
446
+ {"symbol": "NVDA", "source": "finviz_public"},
447
+ {"symbol": "AAPL", "source": "etf_holdings"},
448
+ {"symbol": "XOM", "source": "static"},
449
+ ],
450
+ }
451
+ result = _format_stock_list(theme)
452
+ assert "NVDA[Fp]" in result
453
+ assert "AAPL[E]" in result
454
+ assert "XOM[S]" in result
455
+
456
+ def test_unknown_source_shows_question_mark(self):
457
+ theme = {
458
+ "representative_stocks": ["X"],
459
+ "stock_details": [
460
+ {"symbol": "X", "source": "unknown_source"},
461
+ ],
462
+ }
463
+ assert "X[?]" in _format_stock_list(theme)
464
+
465
+
466
+ class TestMarkdownWithStockDetails:
467
+
468
+ def test_markdown_shows_source_labels(self):
469
+ """Markdown report includes source labels when stock_details present."""
470
+ themes = [
471
+ {
472
+ "name": "Gold Theme",
473
+ "direction": "bullish",
474
+ "heat": 70.0,
475
+ "maturity": 40.0,
476
+ "stage": "growth",
477
+ "confidence": "Medium",
478
+ "industries": ["Gold"],
479
+ "heat_breakdown": {},
480
+ "maturity_breakdown": {},
481
+ "representative_stocks": ["NEM", "GOLD"],
482
+ "stock_details": [
483
+ {"symbol": "NEM", "source": "finviz_public"},
484
+ {"symbol": "GOLD", "source": "finviz_public"},
485
+ ],
486
+ "proxy_etfs": ["GDX"],
487
+ },
488
+ ]
489
+ metadata = {"generated_at": "2026-02-16 10:00:00", "data_sources": {}}
490
+ report = generate_json_report(themes, {"top": [], "bottom": []}, {}, metadata)
491
+ md = generate_markdown_report(report)
492
+ assert "NEM[Fp]" in md
493
+ assert "GOLD[Fp]" in md
494
+
495
+ def test_markdown_no_details_plain_tickers(self):
496
+ """Without stock_details, plain tickers are shown in markdown."""
497
+ themes = [
498
+ {
499
+ "name": "Test Theme",
500
+ "direction": "bullish",
501
+ "heat": 60.0,
502
+ "maturity": 30.0,
503
+ "stage": "early",
504
+ "confidence": "Low",
505
+ "industries": ["Tech"],
506
+ "heat_breakdown": {},
507
+ "maturity_breakdown": {},
508
+ "representative_stocks": ["AAPL", "MSFT"],
509
+ "proxy_etfs": [],
510
+ },
511
+ ]
512
+ metadata = {"generated_at": "2026-02-16 10:00:00", "data_sources": {}}
513
+ report = generate_json_report(themes, {"top": [], "bottom": []}, {}, metadata)
514
+ md = generate_markdown_report(report)
515
+ assert "AAPL, MSFT" in md
516
+ assert "[Fp]" not in md
517
+ assert "[S]" not in md
518
+
519
+
520
+ # --- Tests for theme origin display ---
521
+
522
+ class TestOriginLabel:
523
+
524
+ def test_seed_label(self):
525
+ assert _origin_label("seed") == "Seed"
526
+
527
+ def test_vertical_label(self):
528
+ assert _origin_label("vertical") == "Vertical"
529
+
530
+ def test_discovered_label(self):
531
+ assert _origin_label("discovered") == "*NEW*"
532
+
533
+ def test_none_defaults_to_seed(self):
534
+ assert _origin_label(None) == "Seed"
535
+
536
+ def test_unknown_defaults_to_seed(self):
537
+ assert _origin_label("unknown") == "Seed"
538
+
539
+
540
+ class TestOriginInDashboard:
541
+
542
+ def test_dashboard_shows_origin_column(self):
543
+ themes = [
544
+ {
545
+ "name": "Seed Theme",
546
+ "direction": "bullish",
547
+ "heat": 70.0,
548
+ "maturity": 40.0,
549
+ "stage": "growth",
550
+ "confidence": "Medium",
551
+ "industries": ["Tech"],
552
+ "heat_breakdown": {},
553
+ "maturity_breakdown": {},
554
+ "representative_stocks": ["AAPL"],
555
+ "proxy_etfs": [],
556
+ "theme_origin": "seed",
557
+ },
558
+ {
559
+ "name": "New Theme",
560
+ "direction": "bullish",
561
+ "heat": 60.0,
562
+ "maturity": 30.0,
563
+ "stage": "early",
564
+ "confidence": "Low",
565
+ "industries": ["Mining"],
566
+ "heat_breakdown": {},
567
+ "maturity_breakdown": {},
568
+ "representative_stocks": ["NEM"],
569
+ "proxy_etfs": [],
570
+ "theme_origin": "discovered",
571
+ "name_confidence": "medium",
572
+ },
573
+ ]
574
+ metadata = {"generated_at": "2026-02-16", "data_sources": {}}
575
+ report = generate_json_report(themes, {"top": [], "bottom": []}, {}, metadata)
576
+ md = generate_markdown_report(report)
577
+ assert "| Origin |" in md
578
+ assert "| Seed |" in md
579
+ assert "| *NEW* |" in md
580
+
581
+ def test_discovered_detail_shows_origin_line(self):
582
+ themes = [
583
+ {
584
+ "name": "Discovered Theme",
585
+ "direction": "bullish",
586
+ "heat": 65.0,
587
+ "maturity": 35.0,
588
+ "stage": "early",
589
+ "confidence": "Low",
590
+ "industries": ["X"],
591
+ "heat_breakdown": {},
592
+ "maturity_breakdown": {},
593
+ "representative_stocks": ["XYZ"],
594
+ "proxy_etfs": [],
595
+ "theme_origin": "discovered",
596
+ "name_confidence": "medium",
597
+ },
598
+ ]
599
+ metadata = {"generated_at": "2026-02-16", "data_sources": {}}
600
+ report = generate_json_report(themes, {"top": [], "bottom": []}, {}, metadata)
601
+ md = generate_markdown_report(report)
602
+ assert "**Origin:** Discovered (name confidence: medium)" in md
603
+
604
+ def test_seed_detail_no_origin_line(self):
605
+ themes = [
606
+ {
607
+ "name": "Seed Theme",
608
+ "direction": "bullish",
609
+ "heat": 70.0,
610
+ "maturity": 40.0,
611
+ "stage": "growth",
612
+ "confidence": "Medium",
613
+ "industries": ["Tech"],
614
+ "heat_breakdown": {},
615
+ "maturity_breakdown": {},
616
+ "representative_stocks": ["AAPL"],
617
+ "proxy_etfs": [],
618
+ "theme_origin": "seed",
619
+ },
620
+ ]
621
+ metadata = {"generated_at": "2026-02-16", "data_sources": {}}
622
+ report = generate_json_report(themes, {"top": [], "bottom": []}, {}, metadata)
623
+ md = generate_markdown_report(report)
624
+ assert "**Origin:**" not in md