quantwise 1.2.0 → 1.2.1

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 +15 -15
  362. package/package.json +4 -2
@@ -0,0 +1,673 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Representative Stock Selector - Dynamic stock selection for theme detection.
4
+
5
+ Selects representative stocks for market themes using a fallback chain:
6
+ 1. FINVIZ Elite industry screener (paid API)
7
+ 2. FINVIZ Public screener (finvizfinance, free)
8
+ 3. FMP ETF Holdings (paid API)
9
+ 4. Static stocks (config fallback)
10
+ """
11
+
12
+ import csv
13
+ import io
14
+ import logging
15
+ import sys
16
+ import time
17
+ from typing import Dict, List, Optional, Tuple
18
+
19
+ try:
20
+ import requests
21
+ except ImportError:
22
+ requests = None # type: ignore[assignment]
23
+
24
+ try:
25
+ from finvizfinance.screener.overview import Overview
26
+ except ImportError:
27
+ Overview = None # type: ignore[assignment,misc]
28
+
29
+ logger = logging.getLogger(__name__)
30
+
31
+ _MAX_CONSECUTIVE_FAILURES = 3
32
+
33
+ # ---------------------------------------------------------------------------
34
+ # FINVIZ filter mappings
35
+ # ---------------------------------------------------------------------------
36
+
37
+ # finvizfinance filter_dict option names (public screener)
38
+ CAP_FILTER_MAP = {
39
+ "micro": "+Micro (over $50mln)",
40
+ "small": "+Small (over $300mln)",
41
+ "mid": "+Mid (over $2bln)",
42
+ }
43
+
44
+ # FINVIZ Elite URL filter codes
45
+ CAP_ELITE_MAP = {
46
+ "micro": "cap_microover",
47
+ "small": "cap_smallover",
48
+ "mid": "cap_midover",
49
+ }
50
+
51
+ # Industry name -> Elite URL code (lowercase, spaces/punctuation removed)
52
+ # Derived from finvizfinance.screener.base.filter_dict["Industry"]["option"]
53
+ _INDUSTRY_CODE_CACHE: Dict[str, str] = {}
54
+
55
+
56
+ def _industry_to_code(industry: str) -> str:
57
+ """Convert FINVIZ industry name to Elite URL code."""
58
+ if industry in _INDUSTRY_CODE_CACHE:
59
+ return _INDUSTRY_CODE_CACHE[industry]
60
+ code = industry.lower()
61
+ for ch in " &-/,'()":
62
+ code = code.replace(ch, "")
63
+ _INDUSTRY_CODE_CACHE[industry] = code
64
+ return code
65
+
66
+
67
+ # ---------------------------------------------------------------------------
68
+ # Source state tracking
69
+ # ---------------------------------------------------------------------------
70
+
71
+ class _SourceState:
72
+ """Track per-source failure state for circuit breaker."""
73
+
74
+ def __init__(self):
75
+ self.consecutive_failures: int = 0
76
+ self.total_failures: int = 0
77
+ self.total_queries: int = 0
78
+ self.disabled: bool = False
79
+
80
+
81
+ # ---------------------------------------------------------------------------
82
+ # Parse helpers
83
+ # ---------------------------------------------------------------------------
84
+
85
+ def _parse_market_cap(value) -> int:
86
+ """Parse FINVIZ Market Cap string to integer.
87
+
88
+ "2.8T" -> 2_800_000_000_000
89
+ "150B" -> 150_000_000_000
90
+ "500M" -> 500_000_000
91
+ "-" -> 0
92
+ None -> 0
93
+ """
94
+ if value is None:
95
+ return 0
96
+ if isinstance(value, (int, float)):
97
+ return int(value)
98
+
99
+ s = str(value).strip()
100
+ if not s or s == "-":
101
+ return 0
102
+
103
+ multiplier = 1
104
+ if s.endswith("T"):
105
+ multiplier = 1_000_000_000_000
106
+ s = s[:-1]
107
+ elif s.endswith("B"):
108
+ multiplier = 1_000_000_000
109
+ s = s[:-1]
110
+ elif s.endswith("M"):
111
+ multiplier = 1_000_000
112
+ s = s[:-1]
113
+
114
+ try:
115
+ return int(float(s) * multiplier)
116
+ except (ValueError, TypeError):
117
+ return 0
118
+
119
+
120
+ def _parse_change(value) -> Optional[float]:
121
+ """Parse FINVIZ Change value to float percentage.
122
+
123
+ "12.50%" -> 12.50
124
+ "-3.20%" -> -3.20
125
+ 0.125 -> 12.5 (finvizfinance 0-1 range)
126
+ 12.5 -> 12.5 (abs > 1, already percent)
127
+ "-" -> None
128
+ None -> None
129
+ """
130
+ if value is None:
131
+ return None
132
+
133
+ if isinstance(value, str):
134
+ s = value.strip()
135
+ if not s or s == "-":
136
+ return None
137
+ s = s.replace("%", "")
138
+ try:
139
+ return float(s)
140
+ except (ValueError, TypeError):
141
+ return None
142
+
143
+ if isinstance(value, (int, float)):
144
+ f = float(value)
145
+ # finvizfinance returns 0.125 for 12.5%; detect by abs <= 1
146
+ if abs(f) <= 1.0:
147
+ return f * 100.0
148
+ return f
149
+
150
+ return None
151
+
152
+
153
+ def _parse_volume(value) -> Optional[int]:
154
+ """Parse FINVIZ Volume value to integer.
155
+
156
+ "1,234,567" -> 1234567
157
+ 1234567 -> 1234567
158
+ "1.2M" -> 1200000
159
+ "-" -> None
160
+ None -> None
161
+ """
162
+ if value is None:
163
+ return None
164
+
165
+ if isinstance(value, (int, float)):
166
+ return int(value)
167
+
168
+ s = str(value).strip()
169
+ if not s or s == "-":
170
+ return None
171
+
172
+ # Handle M suffix
173
+ if s.upper().endswith("M"):
174
+ try:
175
+ return int(float(s[:-1]) * 1_000_000)
176
+ except (ValueError, TypeError):
177
+ return None
178
+
179
+ # Handle comma-separated
180
+ s = s.replace(",", "")
181
+ try:
182
+ return int(float(s))
183
+ except (ValueError, TypeError):
184
+ return None
185
+
186
+
187
+ # ---------------------------------------------------------------------------
188
+ # Selector
189
+ # ---------------------------------------------------------------------------
190
+
191
+ class RepresentativeStockSelector:
192
+ """Select representative stocks for themes dynamically.
193
+
194
+ Fallback chain: FINVIZ Elite -> FINVIZ Public -> FMP ETF Holdings -> static_stocks
195
+ """
196
+
197
+ def __init__(
198
+ self,
199
+ finviz_elite_key: Optional[str] = None,
200
+ fmp_api_key: Optional[str] = None,
201
+ finviz_mode: str = "public",
202
+ rate_limit_sec: float = 1.0,
203
+ max_per_industry: int = 4,
204
+ min_cap: str = "small",
205
+ ):
206
+ self._finviz_elite_key = finviz_elite_key
207
+ self._fmp_api_key = fmp_api_key
208
+ self._finviz_mode = finviz_mode
209
+ self._rate_limit_sec = rate_limit_sec
210
+ self._max_per_industry = max_per_industry
211
+ self._min_cap = min_cap
212
+ self._industry_cache: Dict[Tuple[str, bool], List[Dict]] = {}
213
+ self._etf_cache: Dict[str, List[Dict]] = {}
214
+ self._last_request_time: float = 0.0
215
+ self._source_states: Dict[str, _SourceState] = {
216
+ "elite": _SourceState(),
217
+ "public": _SourceState(),
218
+ "fmp": _SourceState(),
219
+ }
220
+
221
+ # -- Public properties ---------------------------------------------------
222
+
223
+ @property
224
+ def query_count(self) -> int:
225
+ return sum(s.total_queries for s in self._source_states.values())
226
+
227
+ @property
228
+ def failure_count(self) -> int:
229
+ return sum(s.total_failures for s in self._source_states.values())
230
+
231
+ @property
232
+ def source_states(self) -> Dict[str, _SourceState]:
233
+ return self._source_states
234
+
235
+ @property
236
+ def _active_sources(self) -> List[str]:
237
+ """Sources actually available given current config."""
238
+ sources: List[str] = []
239
+ if self._finviz_mode == "elite" and self._finviz_elite_key:
240
+ sources.append("elite")
241
+ sources.append("public") # always available
242
+ if self._fmp_api_key:
243
+ sources.append("fmp")
244
+ return sources
245
+
246
+ @property
247
+ def status(self) -> str:
248
+ """Overall health: active / degraded / circuit_broken."""
249
+ active = self._active_sources
250
+ disabled_count = sum(
251
+ 1 for name in active if self._source_states[name].disabled
252
+ )
253
+ if disabled_count == 0:
254
+ return "active"
255
+ elif disabled_count < len(active):
256
+ return "degraded"
257
+ else:
258
+ return "circuit_broken"
259
+
260
+ # -- Main entry point ----------------------------------------------------
261
+
262
+ def select_stocks(self, theme: Dict, max_stocks: int = 10) -> List[Dict]:
263
+ """Select representative stocks for a theme.
264
+
265
+ Returns list of dicts with keys:
266
+ symbol, source, market_cap, matched_industries, reasons, composite_score
267
+ """
268
+ is_bearish = theme.get("direction") == "bearish"
269
+ candidates: List[Dict] = []
270
+ industries = [
271
+ ind.get("name", "") for ind in theme.get("matching_industries", [])
272
+ ]
273
+
274
+ # Priority 1: FINVIZ (Elite or Public) per industry
275
+ for industry in industries:
276
+ cache_key = (industry, is_bearish)
277
+ if cache_key in self._industry_cache:
278
+ candidates.extend(
279
+ self._industry_cache[cache_key][:self._max_per_industry]
280
+ )
281
+ continue
282
+
283
+ stocks: List[Dict] = []
284
+ fetch_limit = max(max_stocks, self._max_per_industry * 2)
285
+
286
+ # Elite attempt
287
+ if (
288
+ self._finviz_mode == "elite"
289
+ and self._finviz_elite_key
290
+ and not self._source_states["elite"].disabled
291
+ ):
292
+ stocks = self._fetch_finviz_elite(
293
+ industry, limit=fetch_limit, is_bearish=is_bearish
294
+ )
295
+
296
+ # Public fallback
297
+ if not stocks and not self._source_states["public"].disabled:
298
+ stocks = self._fetch_finviz_public(
299
+ industry, limit=fetch_limit, is_bearish=is_bearish
300
+ )
301
+
302
+ # Score and cache
303
+ stocks = self._compute_composite_score(stocks, is_bearish)
304
+ self._industry_cache[cache_key] = stocks
305
+ candidates.extend(stocks[:self._max_per_industry])
306
+
307
+ # Priority 1.5: 2nd pass - fill from cache for single-industry themes
308
+ unique_syms = set(c["symbol"] for c in candidates)
309
+ if len(unique_syms) < max_stocks:
310
+ for industry in industries:
311
+ cache_key = (industry, is_bearish)
312
+ cached = self._industry_cache.get(cache_key, [])
313
+ for stock in cached[self._max_per_industry:]:
314
+ if stock["symbol"] not in unique_syms:
315
+ candidates.append(stock)
316
+ unique_syms.add(stock["symbol"])
317
+ if len(unique_syms) >= max_stocks:
318
+ break
319
+ if len(unique_syms) >= max_stocks:
320
+ break
321
+
322
+ # Priority 2: FMP ETF Holdings
323
+ unique_count = len(set(c["symbol"] for c in candidates))
324
+ if (
325
+ unique_count < max_stocks
326
+ and self._fmp_api_key
327
+ and not self._source_states["fmp"].disabled
328
+ ):
329
+ for etf in theme.get("proxy_etfs", []):
330
+ if etf in self._etf_cache:
331
+ candidates.extend(self._etf_cache[etf])
332
+ continue
333
+ holdings = self._fetch_etf_holdings(etf, limit=max_stocks)
334
+ self._etf_cache[etf] = holdings
335
+ candidates.extend(holdings)
336
+
337
+ # Priority 3: static_stocks fallback
338
+ if not candidates:
339
+ for sym in theme.get("static_stocks", [])[:max_stocks]:
340
+ candidates.append({
341
+ "symbol": sym,
342
+ "source": "static",
343
+ "market_cap": 0,
344
+ "matched_industries": [],
345
+ "reasons": ["Static fallback"],
346
+ "composite_score": 0,
347
+ })
348
+
349
+ return self._merge_and_rank(candidates, max_stocks)
350
+
351
+ # -- Fetch methods -------------------------------------------------------
352
+
353
+ def _fetch_finviz_elite(
354
+ self, industry: str, limit: int, is_bearish: bool
355
+ ) -> List[Dict]:
356
+ """Fetch stocks via FINVIZ Elite CSV export."""
357
+ if requests is None:
358
+ return []
359
+
360
+ self._rate_limit()
361
+ self._source_states["elite"].total_queries += 1
362
+
363
+ ind_code = _industry_to_code(industry)
364
+ cap_code = CAP_ELITE_MAP.get(self._min_cap, "cap_smallover")
365
+ perf_code = "ta_perf2_4wdown" if is_bearish else "ta_perf2_4wup"
366
+ filter_str = f"ind_{ind_code},{cap_code},sh_avgvol_o100,sh_price_o10,{perf_code}"
367
+ url = (
368
+ f"https://elite.finviz.com/export.ashx"
369
+ f"?v=151&f={filter_str}&ft=4&auth={self._finviz_elite_key}"
370
+ )
371
+
372
+ try:
373
+ resp = requests.get(url, timeout=15)
374
+ if resp.status_code in (401, 403):
375
+ logger.warning("FINVIZ Elite auth failed (%s)", resp.status_code)
376
+ self._record_failure("elite")
377
+ return []
378
+ if resp.status_code != 200:
379
+ self._record_failure("elite")
380
+ return []
381
+
382
+ reader = csv.DictReader(io.StringIO(resp.text))
383
+ stocks: List[Dict] = []
384
+ for row in reader:
385
+ ticker = row.get("Ticker", "").strip()
386
+ if not ticker:
387
+ continue
388
+ stocks.append({
389
+ "symbol": ticker,
390
+ "source": "finviz_elite",
391
+ "market_cap": _parse_market_cap(row.get("Market Cap")),
392
+ "change": _parse_change(row.get("Change")),
393
+ "volume": _parse_volume(row.get("Volume")),
394
+ "matched_industries": [industry],
395
+ "reasons": [f"Elite screener: {industry}"],
396
+ })
397
+ if len(stocks) >= limit:
398
+ break
399
+
400
+ self._record_success("elite")
401
+ return stocks
402
+
403
+ except Exception:
404
+ logger.exception("FINVIZ Elite fetch failed for %s", industry)
405
+ self._record_failure("elite")
406
+ return []
407
+
408
+ def _fetch_finviz_public(
409
+ self, industry: str, limit: int, is_bearish: bool
410
+ ) -> List[Dict]:
411
+ """Fetch stocks via finvizfinance public screener."""
412
+ if Overview is None:
413
+ logger.warning("finvizfinance not installed")
414
+ self._record_failure("public")
415
+ return []
416
+
417
+ self._rate_limit()
418
+ self._source_states["public"].total_queries += 1
419
+
420
+ perf_filter = "Month Down" if is_bearish else "Month Up"
421
+ filters_dict = {
422
+ "Industry": industry,
423
+ "Market Cap.": CAP_FILTER_MAP.get(self._min_cap, "+Small (over $300mln)"),
424
+ "Average Volume": "Over 100K",
425
+ "Price": "Over $10",
426
+ "Performance 2": perf_filter,
427
+ }
428
+
429
+ try:
430
+ o = Overview()
431
+ o.set_filter(filters_dict=filters_dict)
432
+ df = o.screener_view(
433
+ order="Market Cap.", limit=limit, verbose=0, ascend=False
434
+ )
435
+
436
+ if df is None or df.empty:
437
+ self._record_success("public") # Empty result is not a failure
438
+ return []
439
+
440
+ stocks: List[Dict] = []
441
+ for _, row in df.iterrows():
442
+ ticker = str(row.get("Ticker", "")).strip()
443
+ if not ticker:
444
+ continue
445
+ stocks.append({
446
+ "symbol": ticker,
447
+ "source": "finviz_public",
448
+ "market_cap": _parse_market_cap(row.get("Market Cap")),
449
+ "change": _parse_change(row.get("Change")),
450
+ "volume": _parse_volume(row.get("Volume")),
451
+ "matched_industries": [industry],
452
+ "reasons": [f"Public screener: {industry}"],
453
+ })
454
+
455
+ self._record_success("public")
456
+ return stocks
457
+
458
+ except Exception:
459
+ logger.exception("FINVIZ public fetch failed for %s", industry)
460
+ self._record_failure("public")
461
+ return []
462
+
463
+ def _fetch_etf_holdings(self, etf_symbol: str, limit: int) -> List[Dict]:
464
+ """Fetch top ETF holdings via FMP API."""
465
+ if requests is None or not self._fmp_api_key:
466
+ return []
467
+
468
+ self._rate_limit()
469
+ self._source_states["fmp"].total_queries += 1
470
+
471
+ url = (
472
+ f"https://financialmodelingprep.com/api/v3/etf-holder/{etf_symbol}"
473
+ f"?apikey={self._fmp_api_key}"
474
+ )
475
+
476
+ try:
477
+ resp = requests.get(url, timeout=15)
478
+ if resp.status_code != 200:
479
+ self._record_failure("fmp")
480
+ return []
481
+
482
+ data = resp.json()
483
+ if not isinstance(data, list):
484
+ self._record_failure("fmp")
485
+ return []
486
+
487
+ stocks: List[Dict] = []
488
+ for item in data[:limit]:
489
+ ticker = item.get("asset", "").strip()
490
+ if not ticker:
491
+ continue
492
+ stocks.append({
493
+ "symbol": ticker,
494
+ "source": "etf_holdings",
495
+ "market_cap": _parse_market_cap(item.get("marketValue")),
496
+ "change": None,
497
+ "volume": None,
498
+ "matched_industries": [],
499
+ "reasons": [f"Held by {etf_symbol}"],
500
+ })
501
+
502
+ self._record_success("fmp")
503
+ return stocks
504
+
505
+ except Exception:
506
+ logger.exception("FMP ETF holdings fetch failed for %s", etf_symbol)
507
+ self._record_failure("fmp")
508
+ return []
509
+
510
+ # -- Scoring & ranking ---------------------------------------------------
511
+
512
+ def _compute_composite_score(
513
+ self, stocks: List[Dict], is_bearish: bool
514
+ ) -> List[Dict]:
515
+ """Compute composite score: 0.4*cap_rank + 0.3*change_rank + 0.3*vol_rank.
516
+
517
+ Missing fields are excluded and weights re-normalized.
518
+ """
519
+ if not stocks:
520
+ return []
521
+
522
+ n = len(stocks)
523
+ if n == 1:
524
+ stocks[0]["composite_score"] = 1.0
525
+ return stocks
526
+
527
+ # Build ranks for each metric (1 = best, n = worst)
528
+ # market_cap: larger is better
529
+ cap_vals = [(i, s.get("market_cap") or 0) for i, s in enumerate(stocks)]
530
+ cap_sorted = sorted(cap_vals, key=lambda x: x[1], reverse=True)
531
+ cap_rank = {idx: rank + 1 for rank, (idx, _) in enumerate(cap_sorted)}
532
+
533
+ # change: direction-aware
534
+ change_vals = []
535
+ has_change = False
536
+ for i, s in enumerate(stocks):
537
+ c = s.get("change")
538
+ if c is not None:
539
+ has_change = True
540
+ directional = abs(c) if is_bearish else c
541
+ change_vals.append((i, directional))
542
+ else:
543
+ change_vals.append((i, None))
544
+
545
+ change_rank: Dict[int, Optional[int]] = {}
546
+ if has_change:
547
+ valid = [(i, v) for i, v in change_vals if v is not None]
548
+ valid_sorted = sorted(valid, key=lambda x: x[1], reverse=True)
549
+ for rank, (idx, _) in enumerate(valid_sorted):
550
+ change_rank[idx] = rank + 1
551
+ for i, v in change_vals:
552
+ if v is None:
553
+ change_rank[i] = None
554
+ else:
555
+ for i in range(n):
556
+ change_rank[i] = None
557
+
558
+ # volume: larger is better
559
+ vol_vals = []
560
+ has_volume = False
561
+ for i, s in enumerate(stocks):
562
+ v = s.get("volume")
563
+ if v is not None:
564
+ has_volume = True
565
+ vol_vals.append((i, v))
566
+ else:
567
+ vol_vals.append((i, None))
568
+
569
+ vol_rank: Dict[int, Optional[int]] = {}
570
+ if has_volume:
571
+ valid = [(i, v) for i, v in vol_vals if v is not None]
572
+ valid_sorted = sorted(valid, key=lambda x: x[1], reverse=True)
573
+ for rank, (idx, _) in enumerate(valid_sorted):
574
+ vol_rank[idx] = rank + 1
575
+ for i, v in vol_vals:
576
+ if v is None:
577
+ vol_rank[i] = None
578
+ else:
579
+ for i in range(n):
580
+ vol_rank[i] = None
581
+
582
+ # Compute composite with re-normalization
583
+ for i, s in enumerate(stocks):
584
+ weights = {}
585
+ ranks = {}
586
+
587
+ # Cap is always available
588
+ weights["cap"] = 0.4
589
+ ranks["cap"] = cap_rank[i]
590
+
591
+ cr = change_rank.get(i)
592
+ if cr is not None:
593
+ weights["change"] = 0.3
594
+ ranks["change"] = cr
595
+
596
+ vr = vol_rank.get(i)
597
+ if vr is not None:
598
+ weights["vol"] = 0.3
599
+ ranks["vol"] = vr
600
+
601
+ total_weight = sum(weights.values())
602
+
603
+ # Normalized rank score: (n - rank + 1) / n => 1.0 for rank 1
604
+ # Count of valid items for each metric
605
+ valid_counts = {
606
+ "cap": n,
607
+ "change": sum(1 for _, v in change_vals if v is not None) if has_change else 0,
608
+ "vol": sum(1 for _, v in vol_vals if v is not None) if has_volume else 0,
609
+ }
610
+
611
+ score = 0.0
612
+ for key, w in weights.items():
613
+ count = valid_counts.get(key, n)
614
+ if count > 0:
615
+ normalized = (count - ranks[key] + 1) / count
616
+ score += (w / total_weight) * normalized
617
+
618
+ s["composite_score"] = round(score, 4)
619
+
620
+ # Sort by composite score descending
621
+ stocks.sort(key=lambda x: x.get("composite_score", 0), reverse=True)
622
+ return stocks
623
+
624
+ def _merge_and_rank(
625
+ self, candidates: List[Dict], max_stocks: int
626
+ ) -> List[Dict]:
627
+ """Deduplicate by symbol, merge industries/reasons, rank by score."""
628
+ merged: Dict[str, Dict] = {}
629
+ for c in candidates:
630
+ sym = c["symbol"]
631
+ if sym in merged:
632
+ for ind in c.get("matched_industries", []):
633
+ if ind not in merged[sym]["matched_industries"]:
634
+ merged[sym]["matched_industries"].append(ind)
635
+ merged[sym]["reasons"].extend(c.get("reasons", []))
636
+ merged[sym]["composite_score"] = max(
637
+ merged[sym]["composite_score"],
638
+ c.get("composite_score", 0),
639
+ )
640
+ else:
641
+ merged[sym] = dict(c)
642
+
643
+ ranked = sorted(
644
+ merged.values(),
645
+ key=lambda x: x.get("composite_score", 0),
646
+ reverse=True,
647
+ )
648
+ return ranked[:max_stocks]
649
+
650
+ # -- Rate limiting & circuit breaker -------------------------------------
651
+
652
+ def _rate_limit(self):
653
+ """Enforce minimum interval between requests."""
654
+ now = time.time()
655
+ elapsed = now - self._last_request_time
656
+ if elapsed < self._rate_limit_sec:
657
+ time.sleep(self._rate_limit_sec - elapsed)
658
+ self._last_request_time = time.time()
659
+
660
+ def _record_success(self, source: str):
661
+ """Record successful query for source."""
662
+ state = self._source_states[source]
663
+ state.consecutive_failures = 0
664
+
665
+ def _record_failure(self, source: str):
666
+ """Record failed query for source. Disable after consecutive limit."""
667
+ state = self._source_states[source]
668
+ state.consecutive_failures += 1
669
+ state.total_failures += 1
670
+ if state.consecutive_failures >= _MAX_CONSECUTIVE_FAILURES:
671
+ state.disabled = True
672
+ logger.warning("Source %s disabled after %d consecutive failures",
673
+ source, _MAX_CONSECUTIVE_FAILURES)