quantnodes 3.0.1__tar.gz → 3.0.2__tar.gz

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 (441) hide show
  1. {quantnodes-3.0.1 → quantnodes-3.0.2}/PKG-INFO +1 -1
  2. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/commands/__init__.py +2 -0
  3. quantnodes-3.0.2/QuantNodes/cli/commands/mine_logics.py +196 -0
  4. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/__init__.py +9 -0
  5. quantnodes-3.0.2/QuantNodes/research/quant_alpha/factor_pool.py +459 -0
  6. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/logic_mining/__init__.py +13 -0
  7. quantnodes-3.0.2/QuantNodes/research/quant_alpha/logic_mining/batch.py +321 -0
  8. quantnodes-3.0.2/QuantNodes/research/quant_alpha/logic_mining/report.py +213 -0
  9. {quantnodes-3.0.1 → quantnodes-3.0.2}/pyproject.toml +1 -1
  10. {quantnodes-3.0.1 → quantnodes-3.0.2}/quantnodes.egg-info/PKG-INFO +1 -1
  11. {quantnodes-3.0.1 → quantnodes-3.0.2}/quantnodes.egg-info/SOURCES.txt +4 -0
  12. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/__init__.py +0 -0
  13. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/__main__.py +0 -0
  14. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/__init__.py +0 -0
  15. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/agents/__init__.py +0 -0
  16. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/agents/definition.py +0 -0
  17. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/agents/manager.py +0 -0
  18. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/__init__.py +0 -0
  19. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/executor.py +0 -0
  20. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/loader.py +0 -0
  21. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/templates/bollinger_bands.yaml +0 -0
  22. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/templates/dual_ma.yaml +0 -0
  23. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/templates/empty.yaml +0 -0
  24. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/templates/mean_reversion.yaml +0 -0
  25. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/templates/mean_reversion_zscore.yaml +0 -0
  26. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/templates/momentum.yaml +0 -0
  27. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/templates/momentum_breakout.yaml +0 -0
  28. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/templates/rsi_strategy.yaml +0 -0
  29. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/templates/volume_price.yaml +0 -0
  30. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config/types.py +0 -0
  31. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/config_mapper.py +0 -0
  32. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/core/__init__.py +0 -0
  33. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/core/dream.py +0 -0
  34. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/core/quant_dream.py +0 -0
  35. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/cron_jobs.py +0 -0
  36. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/nanobot_bridge.py +0 -0
  37. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/permission/__init__.py +0 -0
  38. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/permission/defaults.py +0 -0
  39. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/permission/evaluate.py +0 -0
  40. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/permission/models.py +0 -0
  41. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/permission/service.py +0 -0
  42. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/providers/__init__.py +0 -0
  43. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/providers/base.py +0 -0
  44. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/providers/quantnodes.py +0 -0
  45. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/providers/rate_limiter.py +0 -0
  46. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/providers/registry.py +0 -0
  47. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/__init__.py +0 -0
  48. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/base.py +0 -0
  49. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/bridge.py +0 -0
  50. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/factor/__init__.py +0 -0
  51. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/factor/correlation.py +0 -0
  52. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/factor/group_backtest.py +0 -0
  53. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/factor/ic_analysis.py +0 -0
  54. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/loader.py +0 -0
  55. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/registry.py +0 -0
  56. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/strategy/__init__.py +0 -0
  57. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/strategy/bollinger.py +0 -0
  58. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/strategy/dual_ma.py +0 -0
  59. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/strategy/momentum.py +0 -0
  60. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills/strategy/rsi_reversal.py +0 -0
  61. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills_quant/__init__.py +0 -0
  62. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills_quant/backtest-analyze/SKILL.md +0 -0
  63. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills_quant/config-driven/SKILL.md +0 -0
  64. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills_quant/factor-research/SKILL.md +0 -0
  65. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills_quant/logic-mining/SKILL.md +0 -0
  66. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills_quant/quant-dream/SKILL.md +0 -0
  67. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills_quant/risk-management/SKILL.md +0 -0
  68. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/skills_quant/strategy-design/SKILL.md +0 -0
  69. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/templates/__init__.py +0 -0
  70. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/__init__.py +0 -0
  71. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/_workspace.py +0 -0
  72. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/alpha_backtest.py +0 -0
  73. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/alpha_evaluate.py +0 -0
  74. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/backtest.py +0 -0
  75. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/base.py +0 -0
  76. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/code_search.py +0 -0
  77. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/config_backtest.py +0 -0
  78. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/context.py +0 -0
  79. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/dream_skill.py +0 -0
  80. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/echo.py +0 -0
  81. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/factor.py +0 -0
  82. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/file_ops.py +0 -0
  83. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/git_ops.py +0 -0
  84. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/operator_lookup.py +0 -0
  85. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/output_truncation.py +0 -0
  86. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/path_check.py +0 -0
  87. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/pipeline.py +0 -0
  88. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/registry.py +0 -0
  89. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/sandbox.py +0 -0
  90. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/shell_safety.py +0 -0
  91. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/strategy.py +0 -0
  92. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/task.py +0 -0
  93. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/web_fetch.py +0 -0
  94. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/web_search.py +0 -0
  95. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/tools/wiki.py +0 -0
  96. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/utils/__init__.py +0 -0
  97. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/utils/helpers.py +0 -0
  98. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/utils/prompt_templates.py +0 -0
  99. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/workflows/__init__.py +0 -0
  100. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/workflows/implementations/__init__.py +0 -0
  101. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/workflows/implementations/alpha_gpt.py +0 -0
  102. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/workflows/implementations/mcts.py +0 -0
  103. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/workflows/parsers.py +0 -0
  104. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/workflows/registry.py +0 -0
  105. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/workflows/step_agent.py +0 -0
  106. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/agent/workflows/tool.py +0 -0
  107. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ai/__init__.py +0 -0
  108. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ai/llm/__init__.py +0 -0
  109. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ai/llm/base.py +0 -0
  110. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ai/llm/decorators.py +0 -0
  111. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ai/llm/gateway.py +0 -0
  112. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ai/llm/null.py +0 -0
  113. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ai/llm/openai.py +0 -0
  114. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ai/optimizer.py +0 -0
  115. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ai/prompts/__init__.py +0 -0
  116. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ai/sandbox.py +0 -0
  117. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ai/sandbox_pandas_bridge.py +0 -0
  118. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ai/strategy_gen.py +0 -0
  119. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/backtest/__init__.py +0 -0
  120. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/backtest/backtest_node.py +0 -0
  121. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/backtest/broker_node.py +0 -0
  122. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/backtest/config_runner.py +0 -0
  123. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/backtest/config_strategy.py +0 -0
  124. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/backtest/risk_node.py +0 -0
  125. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/backtest/strategy_node.py +0 -0
  126. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cache_node/__init__.py +0 -0
  127. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cache_node/base.py +0 -0
  128. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cache_node/cache_store.py +0 -0
  129. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cache_node/metadata.py +0 -0
  130. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/__init__.py +0 -0
  131. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/_helpers.py +0 -0
  132. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/command.py +0 -0
  133. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/commands/agent.py +0 -0
  134. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/commands/alpha.py +0 -0
  135. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/commands/chat.py +0 -0
  136. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/commands/evolve.py +0 -0
  137. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/commands/factor.py +0 -0
  138. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/commands/init.py +0 -0
  139. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/commands/run.py +0 -0
  140. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/commands/serve.py +0 -0
  141. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/commands/version.py +0 -0
  142. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/cli/enhanced.py +0 -0
  143. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/conf_node/__init__.py +0 -0
  144. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/conf_node/base.py +0 -0
  145. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/conf_node/env_config.py +0 -0
  146. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/conf_node/ini_config.py +0 -0
  147. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/conf_node/json_config.py +0 -0
  148. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/conf_node/yaml_config.py +0 -0
  149. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/constants.py +0 -0
  150. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/__init__.py +0 -0
  151. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/_lookback_helpers.py +0 -0
  152. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/ast_parser.py +0 -0
  153. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/base.py +0 -0
  154. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/cache_manager.py +0 -0
  155. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/cache_utils.py +0 -0
  156. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/cond_builder.py +0 -0
  157. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/config.py +0 -0
  158. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/constants.py +0 -0
  159. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/control.py +0 -0
  160. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/data_preprocessing.py +0 -0
  161. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/data_source.py +0 -0
  162. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/events.py +0 -0
  163. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/evolution/__init__.py +0 -0
  164. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/evolution/loop.py +0 -0
  165. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/evolution/operators.py +0 -0
  166. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/evolution/settings.py +0 -0
  167. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/expression.py +0 -0
  168. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/feedback/__init__.py +0 -0
  169. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/feedback/channels.py +0 -0
  170. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/feedback/collector.py +0 -0
  171. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/feedback/dataclass.py +0 -0
  172. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/feedback/llm_judge.py +0 -0
  173. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/knowledge/__init__.py +0 -0
  174. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/knowledge/knowledge_base.py +0 -0
  175. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/knowledge/lineage_compress.py +0 -0
  176. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/knowledge/lineage_expand.py +0 -0
  177. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/knowledge/metrics/__init__.py +0 -0
  178. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/knowledge/metrics/evaluator.py +0 -0
  179. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/knowledge/metrics/metrics.py +0 -0
  180. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/knowledge/rag_prompt.py +0 -0
  181. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/knowledge/retriever.py +0 -0
  182. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/lambda_node.py +0 -0
  183. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/monitoring/__init__.py +0 -0
  184. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/monitoring/collector.py +0 -0
  185. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/monitoring/dashboard.py +0 -0
  186. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/node.py +0 -0
  187. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/pandas_utils.py +0 -0
  188. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/parallel/__init__.py +0 -0
  189. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/parallel/worker.py +0 -0
  190. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/parallel/worker_process.py +0 -0
  191. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/path_utils.py +0 -0
  192. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/pipeline.py +0 -0
  193. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/plugin.py +0 -0
  194. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/quality_gate/__init__.py +0 -0
  195. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/quality_gate/complexity.py +0 -0
  196. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/quality_gate/consistency.py +0 -0
  197. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/quality_gate/node.py +0 -0
  198. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/quality_gate/redundancy.py +0 -0
  199. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/quality_gate/settings.py +0 -0
  200. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/quality_gate/zoo.py +0 -0
  201. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/serializable.py +0 -0
  202. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/serialization.py +0 -0
  203. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/tools.py +0 -0
  204. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/trajectory/__init__.py +0 -0
  205. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/trajectory/entry.py +0 -0
  206. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/trajectory/lineage.py +0 -0
  207. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/trajectory/pool.py +0 -0
  208. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/trajectory/selector.py +0 -0
  209. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/visualization/__init__.py +0 -0
  210. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/visualization/builder.py +0 -0
  211. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/visualization/gate_breakdown.py +0 -0
  212. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/visualization/lineage_dag.py +0 -0
  213. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/visualization/metric_distribution.py +0 -0
  214. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/core/visualization/report.py +0 -0
  215. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/database_node/__init__.py +0 -0
  216. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/database_node/base.py +0 -0
  217. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/database_node/clickhouse_node.py +0 -0
  218. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/database_node/csv_node.py +0 -0
  219. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/database_node/duckdb_node.py +0 -0
  220. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/database_node/factory.py +0 -0
  221. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/database_node/mysql_node.py +0 -0
  222. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/database_node/parquet_node.py +0 -0
  223. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/database_node/sqlite_node.py +0 -0
  224. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/__init__.py +0 -0
  225. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/factor.py +0 -0
  226. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/factor_db.py +0 -0
  227. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/factor_functions/__init__.py +0 -0
  228. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/factor_functions/_helpers.py +0 -0
  229. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/factor_functions/_helpers_debug.py +0 -0
  230. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/factor_functions/composite_ops.py +0 -0
  231. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/factor_functions/math_ops.py +0 -0
  232. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/factor_functions/section_ops.py +0 -0
  233. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/factor_functions/talib_ops.py +0 -0
  234. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/factor_functions/time_ops.py +0 -0
  235. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/factor_operation.py +0 -0
  236. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/factor_table.py +0 -0
  237. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/factor_node/quant_nodes_object.py +0 -0
  238. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/mcp_server/__init__.py +0 -0
  239. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/mcp_server/__main__.py +0 -0
  240. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/mcp_server/server.py +0 -0
  241. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/methods/__init__.py +0 -0
  242. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/methods/pipeline.py +0 -0
  243. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/methods/sandbox.py +0 -0
  244. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/__init__.py +0 -0
  245. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/agent_tools/__init__.py +0 -0
  246. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/agent_tools/monitor_tool.py +0 -0
  247. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/agent_tools/schedule_tool.py +0 -0
  248. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/agent_tools/version_tool.py +0 -0
  249. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/monitor/__init__.py +0 -0
  250. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/monitor/alerter.py +0 -0
  251. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/monitor/collector.py +0 -0
  252. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/monitor/dashboard.py +0 -0
  253. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/monitor/drift.py +0 -0
  254. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/scheduler/__init__.py +0 -0
  255. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/scheduler/runner.py +0 -0
  256. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/scheduler/scheduler.py +0 -0
  257. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/storage/__init__.py +0 -0
  258. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/storage/models.py +0 -0
  259. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/storage/repository.py +0 -0
  260. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/version/__init__.py +0 -0
  261. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/version/diff.py +0 -0
  262. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/monitor/version/version_manager.py +0 -0
  263. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operator_node/__init__.py +0 -0
  264. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operator_node/base.py +0 -0
  265. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operator_node/query_node.py +0 -0
  266. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operator_node/sql_builder.py +0 -0
  267. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operator_node/sql_utils.py +0 -0
  268. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operator_node/transform.py +0 -0
  269. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/__init__.py +0 -0
  270. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/_engine.py +0 -0
  271. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/composite.py +0 -0
  272. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/composite_dag.py +0 -0
  273. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/composite_dag_ops.py +0 -0
  274. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/composite_dag_pandas_ops.py +0 -0
  275. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/custom.py +0 -0
  276. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/facade.py +0 -0
  277. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/math.py +0 -0
  278. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/proxy.py +0 -0
  279. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/registry.py +0 -0
  280. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/section.py +0 -0
  281. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/talib.py +0 -0
  282. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/templates.py +0 -0
  283. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/operators/time_series.py +0 -0
  284. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/__init__.py +0 -0
  285. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/backtest/__init__.py +0 -0
  286. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/backtest/factor_based.py +0 -0
  287. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/backtest/standard.py +0 -0
  288. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/factor/__init__.py +0 -0
  289. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/factor/correlation.py +0 -0
  290. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/factor/group_backtest.py +0 -0
  291. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/factor/ic_analysis.py +0 -0
  292. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/strategy/__init__.py +0 -0
  293. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/strategy/market_neutral.py +0 -0
  294. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/strategy/mean_reversion.py +0 -0
  295. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/strategy/momentum.py +0 -0
  296. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/strategy/pairs_trading.py +0 -0
  297. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/prompts/strategy/trend_following.py +0 -0
  298. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/README.md +0 -0
  299. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/__init__.py +0 -0
  300. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/_legacy_3c/__init__.py +0 -0
  301. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/_legacy_3c/auto_researcher.py +0 -0
  302. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/_legacy_3c/factor_evaluator.py +0 -0
  303. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/_legacy_3c/factor_miner.py +0 -0
  304. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/_legacy_3c/mcts_search.py +0 -0
  305. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/__init__.py +0 -0
  306. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/config.py +0 -0
  307. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/config_builder.py +0 -0
  308. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/e2e/data_prep.py +0 -0
  309. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/e2e/run_evolution_e2e.py +0 -0
  310. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/evolution_adapter.py +0 -0
  311. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/feedback_wrapper.py +0 -0
  312. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/ifind_db/__init__.py +0 -0
  313. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/ifind_db/fetcher.py +0 -0
  314. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/ifind_db/ifind_database.py +0 -0
  315. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/__init__.py +0 -0
  316. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/_base.py +0 -0
  317. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/adjust_date_node.py +0 -0
  318. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/configs.py +0 -0
  319. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/factor_neutralize_node.py +0 -0
  320. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/factor_preprocess_node.py +0 -0
  321. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/factor_score_node.py +0 -0
  322. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/factor_test_report_node.py +0 -0
  323. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/group_analyzer_node.py +0 -0
  324. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/ic_analyzer_node.py +0 -0
  325. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/load_data_node.py +0 -0
  326. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/long_short_node.py +0 -0
  327. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/neutralizers.py +0 -0
  328. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/preprocess_strategies.py +0 -0
  329. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/risk_correlation_node.py +0 -0
  330. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/sample_pool_filter_node.py +0 -0
  331. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/nodes/tradability_filter_node.py +0 -0
  332. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/pipeline_runner.py +0 -0
  333. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/pipeline_spec.py +0 -0
  334. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/utils/__init__.py +0 -0
  335. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/utils/constants.py +0 -0
  336. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/utils/data_loader.py +0 -0
  337. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/utils/date_utils.py +0 -0
  338. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/utils/file_loaders.py +0 -0
  339. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/utils/labels.py +0 -0
  340. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/utils/metrics_extractor.py +0 -0
  341. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/utils/performance_metrics.py +0 -0
  342. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/factor_test/utils/safe_load.py +0 -0
  343. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/CHANGELOG.md +0 -0
  344. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/README.md +0 -0
  345. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/adapters/__init__.py +0 -0
  346. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/adapters/calculator.py +0 -0
  347. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/adapters/expression.py +0 -0
  348. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/alpha101_design/__init__.py +0 -0
  349. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/alpha101_design/few_shot_examples.py +0 -0
  350. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/alpha101_design/philosophy.py +0 -0
  351. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/alpha158_design/__init__.py +0 -0
  352. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/alpha158_design/few_shot_examples.py +0 -0
  353. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/alpha158_design/philosophy.py +0 -0
  354. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/evaluation/__init__.py +0 -0
  355. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/evaluation/baselines/__init__.py +0 -0
  356. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/evaluation/baselines/g1_handcrafted.py +0 -0
  357. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/evaluation/baselines/g2_llm_only.py +0 -0
  358. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/evaluation/baselines/g3_alpha_gpt.py +0 -0
  359. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/evaluation/clickhouse_data_loader.py +0 -0
  360. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/evaluation/contracts.py +0 -0
  361. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/evaluation/evaluators/__init__.py +0 -0
  362. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/evaluation/evaluators/polars_evaluator.py +0 -0
  363. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/evaluation/mock_data_loader.py +0 -0
  364. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/evaluation/runner.py +0 -0
  365. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/llm/__init__.py +0 -0
  366. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/llm/parser.py +0 -0
  367. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/logic_driven_pipeline.py +0 -0
  368. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/logic_mining/compiler.py +0 -0
  369. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/logic_mining/generator.py +0 -0
  370. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/logic_mining/metrics.py +0 -0
  371. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/logic_mining/models.py +0 -0
  372. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/logic_mining/parser.py +0 -0
  373. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/logic_mining/pipelines.py +0 -0
  374. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/logic_mining/sources.py +0 -0
  375. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/mcts/__init__.py +0 -0
  376. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/mcts/cache.py +0 -0
  377. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/mcts/extension_ops.py +0 -0
  378. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/mcts/feedback.py +0 -0
  379. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/mcts/op_prior.py +0 -0
  380. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/mcts/search.py +0 -0
  381. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/mcts/tree.py +0 -0
  382. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/operator_vocab/__init__.py +0 -0
  383. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/operator_vocab/config.py +0 -0
  384. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/operator_vocab/metadata.py +0 -0
  385. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/operator_vocab/vocabulary.py +0 -0
  386. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/pipeline.py +0 -0
  387. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/types/__init__.py +0 -0
  388. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/types/constants.py +0 -0
  389. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/types/state.py +0 -0
  390. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/workflow/__init__.py +0 -0
  391. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/workflow/alpha_gpt.py +0 -0
  392. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/workflow/alpha_logics.py +0 -0
  393. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/quant_alpha/workflow/state.py +0 -0
  394. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/report_reproducer.py +0 -0
  395. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/research/wiki.py +0 -0
  396. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/symbolic/__init__.py +0 -0
  397. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/symbolic/compiler.py +0 -0
  398. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/symbolic/dialect.py +0 -0
  399. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/symbolic/executor.py +0 -0
  400. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/symbolic/expression.py +0 -0
  401. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/symbolic/functions.py +0 -0
  402. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/symbolic/optimizer.py +0 -0
  403. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ui_node/__init__.py +0 -0
  404. {quantnodes-3.0.1 → quantnodes-3.0.2}/QuantNodes/ui_node/base.py +0 -0
  405. {quantnodes-3.0.1 → quantnodes-3.0.2}/README.md +0 -0
  406. {quantnodes-3.0.1 → quantnodes-3.0.2}/quantnodes.egg-info/dependency_links.txt +0 -0
  407. {quantnodes-3.0.1 → quantnodes-3.0.2}/quantnodes.egg-info/entry_points.txt +0 -0
  408. {quantnodes-3.0.1 → quantnodes-3.0.2}/quantnodes.egg-info/requires.txt +0 -0
  409. {quantnodes-3.0.1 → quantnodes-3.0.2}/quantnodes.egg-info/top_level.txt +0 -0
  410. {quantnodes-3.0.1 → quantnodes-3.0.2}/setup.cfg +0 -0
  411. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_ai.py +0 -0
  412. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_backtest_node.py +0 -0
  413. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_base_node.py +0 -0
  414. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_cli_enhanced.py +0 -0
  415. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_composite_dag.py +0 -0
  416. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_composite_dag_engine.py +0 -0
  417. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_composite_dag_ops.py +0 -0
  418. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_composite_dag_pandas_engine.py +0 -0
  419. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_composite_dag_pandas_ops.py +0 -0
  420. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_config_node.py +0 -0
  421. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_control.py +0 -0
  422. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_database_node.py +0 -0
  423. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_database_node_factory.py +0 -0
  424. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_example_strategies.py +0 -0
  425. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_expression.py +0 -0
  426. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_factor_functions.py +0 -0
  427. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_full_market_performance.py +0 -0
  428. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_group_analyzer_bool.py +0 -0
  429. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_mcp_server.py +0 -0
  430. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_operator_node.py +0 -0
  431. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_operators_engine_registry_zoo.py +0 -0
  432. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_pipeline.py +0 -0
  433. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_pipeline_plugin.py +0 -0
  434. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_profile_single.py +0 -0
  435. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_sandbox_allowed_imports.py +0 -0
  436. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_sandbox_pandas_bridge.py +0 -0
  437. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_serialization.py +0 -0
  438. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_settings_page.py +0 -0
  439. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_symbolic.py +0 -0
  440. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_talib.py +0 -0
  441. {quantnodes-3.0.1 → quantnodes-3.0.2}/tests/test_ui_node.py +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: quantnodes
3
- Version: 3.0.1
3
+ Version: 3.0.2
4
4
  Summary: AI-native quantitative research framework with symbolic computation and SQL pushdown
5
5
  Author-email: sn0wfree <snowfreedom0815@gmail.com>
6
6
  License: MIT
@@ -29,6 +29,7 @@ def _register_all() -> None:
29
29
  from QuantNodes.cli.commands.agent import AgentCommand
30
30
  from QuantNodes.cli.commands.evolve import EvolveCommand
31
31
  from QuantNodes.cli.commands.alpha import AlphaMctsCommand, AlphaGptCommand, AlphaPipelineCommand
32
+ from QuantNodes.cli.commands.mine_logics import MineLogicsCommand
32
33
  from QuantNodes.cli.commands.factor import (
33
34
  FactorInfoCommand,
34
35
  FactorBestCommand,
@@ -53,6 +54,7 @@ def _register_all() -> None:
53
54
  AlphaMctsCommand(),
54
55
  AlphaGptCommand(),
55
56
  AlphaPipelineCommand(),
57
+ MineLogicsCommand(),
56
58
  FactorInfoCommand(),
57
59
  FactorBestCommand(),
58
60
  FactorVisualCommand(),
@@ -0,0 +1,196 @@
1
+ # coding=utf-8
2
+ """
3
+ CLI 命令: mine-logics — 并发批量逻辑挖掘 (v3.0.2)
4
+
5
+ Usage:
6
+ quantnodes mine-logics --max-per-lib 5 --workers 2 --live
7
+ quantnodes mine-logics --source-libs alpha101,alpha191 --max-per-lib 10
8
+ quantnodes mine-logics --wiki-path wiki_auto --output-dir data/mine_runs
9
+ """
10
+
11
+ from __future__ import annotations
12
+
13
+ import argparse
14
+ import json
15
+ import sys
16
+ import time
17
+ from datetime import datetime
18
+ from pathlib import Path
19
+ from typing import Optional
20
+
21
+ from QuantNodes.cli.command import Command
22
+ from QuantNodes.research.quant_alpha.logic_mining.batch import (
23
+ ThreadSafeMetrics,
24
+ mine_logic_library_v2,
25
+ )
26
+ from QuantNodes.research.quant_alpha.logic_mining.report import MetricsReportBuilder
27
+
28
+
29
+ class MineLogicsCommand(Command):
30
+ """quantnodes mine-logics - 并发批量逻辑挖掘
31
+
32
+ 并发挖掘 alpha101/alpha158/alpha191 的逻辑结构,
33
+ 生成 Logic pages 到 Wiki,输出 JSON + Markdown 报告。
34
+
35
+ 默认使用 NullLLMClient(离线模式);加 --live 走真实 LLM。
36
+ """
37
+ name = "mine-logics"
38
+ description = "并发批量逻辑挖掘(v3.0.2)"
39
+
40
+ def add_arguments(self, subparsers: object) -> None:
41
+ parser: argparse.ArgumentParser = subparsers.add_parser(
42
+ self.name,
43
+ help=self.description,
44
+ description=(
45
+ "基于 LogicMiningPipeline 三段式 Agent 的并发批量挖掘。\n"
46
+ "支持幂等重跑(跳过已存在 Logic pages)。\n"
47
+ "默认离线模式(NullLLMClient);加 --live 走真实 LLM。"
48
+ ),
49
+ )
50
+ parser.add_argument(
51
+ "--source-libs",
52
+ type=str,
53
+ default="alpha101,alpha158,alpha191",
54
+ help="逗号分隔的来源库列表(默认 alpha101,alpha158,alpha191)",
55
+ )
56
+ parser.add_argument(
57
+ "--max-per-lib",
58
+ type=int,
59
+ default=10,
60
+ help="每个来源库最多挖掘多少条公式(默认 10)",
61
+ )
62
+ parser.add_argument(
63
+ "--workers",
64
+ type=int,
65
+ default=4,
66
+ help="并发线程数(默认 4)",
67
+ )
68
+ parser.add_argument(
69
+ "--wiki-path",
70
+ type=str,
71
+ default="wiki_auto",
72
+ help="Wiki 根目录(默认 wiki_auto)",
73
+ )
74
+ parser.add_argument(
75
+ "--output-dir",
76
+ type=str,
77
+ default="data/mine_runs",
78
+ help="报告输出目录(默认 data/mine_runs)",
79
+ )
80
+ parser.add_argument(
81
+ "--live",
82
+ action="store_true",
83
+ help="使用真实 LLM(默认离线 NullLLMClient)",
84
+ )
85
+ parser.add_argument(
86
+ "--strict",
87
+ action="store_true",
88
+ help="启用严格模式(LLM/parse/structured 异常上抛)",
89
+ )
90
+ parser.add_argument(
91
+ "--no-skip",
92
+ action="store_true",
93
+ help="不跳过已存在的 Logic pages(默认幂等跳过)",
94
+ )
95
+ parser.add_argument(
96
+ "--quiet", "-q",
97
+ action="store_true",
98
+ help="安静模式(不打印进度)",
99
+ )
100
+
101
+ def run(self, args: argparse.Namespace) -> int:
102
+ source_libs = [s.strip() for s in args.source_libs.split(",") if s.strip()]
103
+ if not source_libs:
104
+ print("Error: --source-libs is empty", file=sys.stderr)
105
+ return 2
106
+
107
+ output_dir = Path(args.output_dir)
108
+ output_dir.mkdir(parents=True, exist_ok=True)
109
+
110
+ # 选择 LLM 客户端
111
+ llm_client = None
112
+ if args.live:
113
+ try:
114
+ from QuantNodes.ai.llm.gateway import get_llm_gateway
115
+ llm_client = get_llm_gateway()
116
+ print(f"[mine-logics] Live mode: LLM gateway initialized")
117
+ except Exception as exc:
118
+ print(f"Error: Failed to initialize LLM gateway: {exc}", file=sys.stderr)
119
+ return 2
120
+ else:
121
+ print("[mine-logics] Offline mode: using NullLLMClient")
122
+
123
+ # 构建 metrics + strict
124
+ metrics = ThreadSafeMetrics()
125
+ strict = None
126
+ if args.strict:
127
+ from QuantNodes.research.quant_alpha.logic_mining.metrics import StrictConfig
128
+ strict = StrictConfig(call=True, parse=True, structured=True)
129
+
130
+ # 进度回调
131
+ def _on_progress(done: int, total: int, fid: str) -> None:
132
+ if not args.quiet:
133
+ print(f" [{done}/{total}] {fid}", flush=True)
134
+
135
+ print(f"[mine-logics] Source libs: {source_libs}")
136
+ print(f"[mine-logics] max_per_lib={args.max_per_lib}, workers={args.workers}")
137
+ print(f"[mine-logics] wiki_path={args.wiki_path}")
138
+ print()
139
+
140
+ t0 = time.perf_counter()
141
+ try:
142
+ batch = mine_logic_library_v2(
143
+ source_libs=source_libs,
144
+ llm_client=llm_client,
145
+ max_per_lib=args.max_per_lib,
146
+ workers=args.workers,
147
+ metrics=metrics,
148
+ strict=strict,
149
+ wiki_path=args.wiki_path,
150
+ skip_existing=not args.no_skip,
151
+ on_progress=_on_progress,
152
+ )
153
+ except Exception as exc:
154
+ print(f"Fatal error: {exc}", file=sys.stderr)
155
+ return 2
156
+
157
+ # 生成报告
158
+ report = MetricsReportBuilder.from_batch(batch)
159
+ ts = datetime.utcnow().strftime("%Y%m%d_%H%M%S")
160
+ json_path = output_dir / f"metrics_{ts}.json"
161
+ md_path = output_dir / f"metrics_{ts}.md"
162
+ report.to_json(json_path)
163
+ md_text = report.to_markdown()
164
+ md_path.write_text(md_text, encoding="utf-8")
165
+
166
+ # 打印摘要
167
+ print()
168
+ print(f"[mine-logics] Done in {batch.wall_clock_s:.2f}s")
169
+ print(f" Attempted: {len(batch.attempted_ids)}")
170
+ print(f" Mined: {batch.n_mined}")
171
+ print(f" Skipped: {batch.n_skipped}")
172
+ print(f" Failed: {batch.n_failed}")
173
+ if batch.warnings:
174
+ for w in batch.warnings:
175
+ print(f" Warning: {w}")
176
+ print(f" Reports: {json_path} / {md_path}")
177
+
178
+ # Wiki 写入
179
+ if batch.pool and batch.n_mined > 0:
180
+ try:
181
+ from QuantNodes.research.wiki import WikiFactorProxy
182
+ proxy = WikiFactorProxy(wiki_path=args.wiki_path)
183
+ n_written = batch.pool.to_wiki(proxy)
184
+ print(f" Wiki: {n_written} Logic pages written to {args.wiki_path}/")
185
+ failed = batch.pool.failed_writes()
186
+ if failed:
187
+ print(f" Wiki failures: {len(failed)}")
188
+ except Exception as exc:
189
+ print(f" Wiki write failed: {exc}", file=sys.stderr)
190
+
191
+ # 退出码
192
+ if batch.n_failed > 0:
193
+ return 1 # 部分成功
194
+ if batch.n_mined == 0 and batch.n_skipped == 0:
195
+ return 2 # 空结果
196
+ return 0 # 全部成功
@@ -32,6 +32,12 @@ from QuantNodes.research.quant_alpha.operator_vocab import (
32
32
  get_vocab_metadata,
33
33
  )
34
34
 
35
+ # v3.0.2 自动化挖掘 API
36
+ from QuantNodes.research.quant_alpha.factor_pool import (
37
+ FactorEntry,
38
+ FactorPool,
39
+ )
40
+
35
41
  __all__ = [
36
42
  # 主类
37
43
  "OperatorVocab",
@@ -42,4 +48,7 @@ __all__ = [
42
48
  "list_vocab_operators",
43
49
  "get_vocab_operator",
44
50
  "get_vocab_metadata",
51
+ # v3.0.2 FactorPool
52
+ "FactorEntry",
53
+ "FactorPool",
45
54
  ]
@@ -0,0 +1,459 @@
1
+ # coding=utf-8
2
+ """
3
+ factor_pool.py - 因子池抽象 (v3.0.2)
4
+
5
+ 提供 FactorPool 类:
6
+ - 内存中的 dict[str, FactorEntry] 池
7
+ - 与 WikiFactorProxy 双向同步 (from_wiki / to_wiki)
8
+ - dedup / select / summary / save_json / load_json
9
+ - 线程安全 (add / extend / to_wiki 内部加锁)
10
+
11
+ 设计动机:
12
+ - LogicDrivenPipeline / AlphaPipeline 仅返回 final_pool: List[FactorMetrics] (临时)
13
+ - 跨多次 mine 任务 / 跨 LLM 客户端都需要持久化池
14
+ - Wiki 已经是唯一持久化层, 池作为 in-mem 镜像存在
15
+
16
+ Usage::
17
+
18
+ from QuantNodes.research.quant_alpha.factor_pool import FactorPool, FactorEntry
19
+ from QuantNodes.research.wiki import WikiFactorProxy
20
+
21
+ pool = FactorPool(wiki_path="wiki_auto")
22
+ pool.from_wiki(WikiFactorProxy(wiki_path="wiki_auto"))
23
+ pool.add(FactorEntry(formula_id="alpha101-alpha006", formula="...",
24
+ source_lib="alpha101", ir=1.0))
25
+ pool.select(top_n=5, by="ir")
26
+ n_written = pool.to_wiki(WikiFactorProxy(wiki_path="wiki_auto"))
27
+ pool.save_json(Path("data/mine_runs/pool.json"))
28
+ """
29
+
30
+ from __future__ import annotations
31
+
32
+ import json
33
+ import logging
34
+ import threading
35
+ from dataclasses import asdict, dataclass, field
36
+ from datetime import datetime
37
+ from pathlib import Path
38
+ from typing import Any, Dict, Iterable, List, Optional, Set
39
+
40
+ logger = logging.getLogger(__name__)
41
+
42
+ __all__ = ["FactorEntry", "FactorPool"]
43
+
44
+
45
+ @dataclass
46
+ class FactorEntry:
47
+ """池内单条因子记录
48
+
49
+ Attributes:
50
+ formula_id: 唯一键, 推荐 "{source_lib}-{source_id}" (例如 "alpha101-alpha006")
51
+ formula: 因子公式字符串
52
+ source_lib: 来源库 ("alpha101" / "alpha158" / "alpha191")
53
+ source_id: 在来源库中的 ID (例如 "alpha006")
54
+ ir: 信息比率 (Information Ratio)
55
+ ic_mean: IC 均值
56
+ rank_ic: Rank IC
57
+ tags: 标签列表
58
+ discovered_at: 发现时间 (ISO 8601 字符串)
59
+ wiki_path: 对应的 Wiki page path (e.g. "Logic/alpha101-alpha006.md")
60
+ structured: 序列化的 WikiLogicStructured (dict 形式, 可选)
61
+ evidence: 序列化的 LogicPerformanceEvidence (dict 形式, 可选)
62
+ """
63
+
64
+ formula_id: str
65
+ formula: str
66
+ source_lib: str = ""
67
+ source_id: str = ""
68
+ ir: float = 0.0
69
+ ic_mean: float = 0.0
70
+ rank_ic: float = 0.0
71
+ tags: List[str] = field(default_factory=list)
72
+ discovered_at: str = ""
73
+ wiki_path: Optional[str] = None
74
+ structured: Optional[Dict[str, Any]] = None
75
+ evidence: Optional[Dict[str, Any]] = None
76
+
77
+ def __post_init__(self) -> None:
78
+ if not self.discovered_at:
79
+ self.discovered_at = datetime.utcnow().isoformat(timespec="seconds")
80
+
81
+ @classmethod
82
+ def from_logic_result(
83
+ cls,
84
+ formula_id: str,
85
+ formula: str,
86
+ source_lib: str,
87
+ source_id: str = "",
88
+ ir: float = 0.0,
89
+ ic_mean: float = 0.0,
90
+ rank_ic: float = 0.0,
91
+ tags: Optional[List[str]] = None,
92
+ structured: Optional[Dict[str, Any]] = None,
93
+ evidence: Optional[Dict[str, Any]] = None,
94
+ ) -> "FactorEntry":
95
+ """便捷构造: 从 Logic Mining 结果拼 FactorEntry"""
96
+ return cls(
97
+ formula_id=formula_id,
98
+ formula=formula,
99
+ source_lib=source_lib,
100
+ source_id=source_id or formula_id,
101
+ ir=ir,
102
+ ic_mean=ic_mean,
103
+ rank_ic=rank_ic,
104
+ tags=list(tags or []),
105
+ structured=structured,
106
+ evidence=evidence,
107
+ )
108
+
109
+ def to_dict(self) -> Dict[str, Any]:
110
+ return asdict(self)
111
+
112
+ @classmethod
113
+ def from_dict(cls, data: Dict[str, Any]) -> "FactorEntry":
114
+ return cls(**{k: v for k, v in data.items() if k in cls.__dataclass_fields__})
115
+
116
+
117
+ class FactorPool:
118
+ """线程安全的 in-mem 因子池
119
+
120
+ - 内部存储 Dict[formula_id, FactorEntry]
121
+ - add / extend / dedup / select / summary 操作均线程安全
122
+ - from_wiki / to_wiki 与 WikiFactorProxy 双向同步
123
+ - save_json / load_json 支持离线持久化 (与 wiki 解耦)
124
+ """
125
+
126
+ def __init__(self, wiki_path: str = "wiki") -> None:
127
+ self.wiki_path = wiki_path
128
+ self._entries: Dict[str, FactorEntry] = {}
129
+ self._lock = threading.Lock()
130
+ self._failed_writes: List[Dict[str, str]] = []
131
+
132
+ # ------------------------------------------------------------------
133
+ # 基础 CRUD
134
+ # ------------------------------------------------------------------
135
+ def add(self, entry: FactorEntry) -> bool:
136
+ """添加一条; 已存在则覆盖; 返回 True=新增, False=覆盖"""
137
+ with self._lock:
138
+ existed = entry.formula_id in self._entries
139
+ self._entries[entry.formula_id] = entry
140
+ return not existed
141
+
142
+ def extend(self, entries: Iterable[FactorEntry]) -> int:
143
+ """批量添加; 返回新增 (不含覆盖) 数量"""
144
+ added = 0
145
+ with self._lock:
146
+ for e in entries:
147
+ if e.formula_id not in self._entries:
148
+ self._entries[e.formula_id] = e
149
+ added += 1
150
+ else:
151
+ self._entries[e.formula_id] = e
152
+ return added
153
+
154
+ def remove(self, formula_id: str) -> bool:
155
+ with self._lock:
156
+ return self._entries.pop(formula_id, None) is not None
157
+
158
+ def get(self, formula_id: str) -> Optional[FactorEntry]:
159
+ with self._lock:
160
+ return self._entries.get(formula_id)
161
+
162
+ def contains(self, formula_id: str) -> bool:
163
+ with self._lock:
164
+ return formula_id in self._entries
165
+
166
+ def clear(self) -> None:
167
+ with self._lock:
168
+ self._entries.clear()
169
+ self._failed_writes.clear()
170
+
171
+ def __len__(self) -> int:
172
+ with self._lock:
173
+ return len(self._entries)
174
+
175
+ def __iter__(self):
176
+ with self._lock:
177
+ return iter(list(self._entries.values()))
178
+
179
+ def __contains__(self, formula_id: str) -> bool:
180
+ return self.contains(formula_id)
181
+
182
+ def keys(self) -> List[str]:
183
+ with self._lock:
184
+ return list(self._entries.keys())
185
+
186
+ def values(self) -> List[FactorEntry]:
187
+ with self._lock:
188
+ return list(self._entries.values())
189
+
190
+ # ------------------------------------------------------------------
191
+ # dedup / select / summary
192
+ # ------------------------------------------------------------------
193
+ def dedup(self, by: str = "formula_id") -> int:
194
+ """去重; 返回被移除的数量
195
+
196
+ Args:
197
+ by: "formula_id" (默认, 直接按主键) /
198
+ "formula" (按公式字符串, 保留 IR 较高者) /
199
+ "source_id" (按 (source_lib, source_id) 元组)
200
+ """
201
+ if by == "formula_id":
202
+ return 0 # 主键天然唯一
203
+
204
+ with self._lock:
205
+ if by == "formula":
206
+ seen: Dict[str, FactorEntry] = {}
207
+ for fid, e in self._entries.items():
208
+ if e.formula not in seen or e.ir > seen[e.formula].ir:
209
+ seen[e.formula] = e
210
+ removed = len(self._entries) - len(seen)
211
+ self._entries = {e.formula_id: e for e in seen.values()}
212
+ return removed
213
+
214
+ if by == "source_id":
215
+ seen2: Dict[tuple, FactorEntry] = {}
216
+ for fid, e in self._entries.items():
217
+ key = (e.source_lib, e.source_id)
218
+ if key not in seen2 or e.ir > seen2[key].ir:
219
+ seen2[key] = e
220
+ removed = len(self._entries) - len(seen2)
221
+ self._entries = {e.formula_id: e for e in seen2.values()}
222
+ return removed
223
+
224
+ raise ValueError(f"Unknown dedup key: {by!r}")
225
+
226
+ def select(self, top_n: int = 10, by: str = "ir") -> List[FactorEntry]:
227
+ """按 IR (默认) / ic_mean / rank_ic 排序, 取前 top_n"""
228
+ if top_n <= 0:
229
+ return []
230
+ with self._lock:
231
+ entries = list(self._entries.values())
232
+ key_fn = {
233
+ "ir": lambda e: e.ir,
234
+ "ic_mean": lambda e: e.ic_mean,
235
+ "rank_ic": lambda e: e.rank_ic,
236
+ }.get(by, lambda e: e.ir)
237
+ entries.sort(key=key_fn, reverse=True)
238
+ return entries[:top_n]
239
+
240
+ def filter(
241
+ self,
242
+ *,
243
+ source_lib: Optional[str] = None,
244
+ min_ir: Optional[float] = None,
245
+ tags: Optional[List[str]] = None,
246
+ ) -> List[FactorEntry]:
247
+ """按条件过滤"""
248
+ with self._lock:
249
+ entries = list(self._entries.values())
250
+ out = []
251
+ for e in entries:
252
+ if source_lib and e.source_lib != source_lib:
253
+ continue
254
+ if min_ir is not None and e.ir < min_ir:
255
+ continue
256
+ if tags and not any(t in e.tags for t in tags):
257
+ continue
258
+ out.append(e)
259
+ return out
260
+
261
+ def summary(self) -> Dict[str, Any]:
262
+ """池整体统计"""
263
+ with self._lock:
264
+ entries = list(self._entries.values())
265
+ if not entries:
266
+ return {
267
+ "n_total": 0,
268
+ "by_source_lib": {},
269
+ "ir_stats": {},
270
+ "n_with_wiki": 0,
271
+ }
272
+ irs = [e.ir for e in entries]
273
+ by_lib: Dict[str, int] = {}
274
+ for e in entries:
275
+ by_lib[e.source_lib] = by_lib.get(e.source_lib, 0) + 1
276
+ irs_sorted = sorted(irs)
277
+ n = len(irs_sorted)
278
+ median = (
279
+ irs_sorted[n // 2]
280
+ if n % 2 == 1
281
+ else (irs_sorted[n // 2 - 1] + irs_sorted[n // 2]) / 2.0
282
+ )
283
+ return {
284
+ "n_total": n,
285
+ "by_source_lib": by_lib,
286
+ "ir_stats": {
287
+ "min": min(irs),
288
+ "max": max(irs),
289
+ "mean": sum(irs) / n,
290
+ "median": median,
291
+ },
292
+ "n_with_wiki": sum(1 for e in entries if e.wiki_path),
293
+ }
294
+
295
+ # ------------------------------------------------------------------
296
+ # Wiki 双向同步
297
+ # ------------------------------------------------------------------
298
+ def from_wiki(self, proxy: Any) -> int:
299
+ """从 Wiki 读取 Logic 页 → 池; 返回加载数量
300
+
301
+ Args:
302
+ proxy: WikiFactorProxy 实例
303
+ """
304
+ try:
305
+ existing_logics = proxy.list_logics(limit=10_000)
306
+ except Exception as exc:
307
+ logger.warning("from_wiki: list_logics failed: %s", exc)
308
+ return 0
309
+
310
+ loaded = 0
311
+ with self._lock:
312
+ for logic in existing_logics:
313
+ structured_dict = None
314
+ if getattr(logic, "structured", None) is not None:
315
+ try:
316
+ structured_dict = logic.structured.to_dict()
317
+ except Exception:
318
+ structured_dict = None
319
+
320
+ evidence_dict = None
321
+ if getattr(logic, "performance_evidence", None) is not None:
322
+ try:
323
+ evidence_dict = logic.performance_evidence.to_dict()
324
+ except Exception:
325
+ evidence_dict = None
326
+
327
+ src_detail = getattr(logic, "source_detail", None) or {}
328
+ entry = FactorEntry(
329
+ formula_id=logic.name,
330
+ formula=getattr(logic, "extracted_formula", "") or "",
331
+ source_lib=src_detail.get("source_lib", "wiki") if isinstance(src_detail, dict) else "wiki",
332
+ source_id=logic.name,
333
+ ir=(evidence_dict or {}).get("best_ir", 0.0),
334
+ ic_mean=(evidence_dict or {}).get("best_ic", 0.0),
335
+ rank_ic=0.0,
336
+ tags=list(src_detail.get("tags", []) if isinstance(src_detail, dict) else []),
337
+ discovered_at=getattr(logic, "created_at", "") or "",
338
+ wiki_path=getattr(logic, "wiki_page_name", None),
339
+ structured=structured_dict,
340
+ evidence=evidence_dict,
341
+ )
342
+ self._entries[entry.formula_id] = entry
343
+ loaded += 1
344
+ return loaded
345
+
346
+ def to_wiki(self, proxy: Any) -> int:
347
+ """把池内条目写入 Wiki Logic 页; 返回成功数量
348
+
349
+ 单条失败不中断, 记录到 self._failed_writes
350
+ """
351
+ from datetime import datetime as _dt
352
+
353
+ from QuantNodes.research.wiki import LogicSource, WikiLogic
354
+
355
+ with self._lock:
356
+ entries = list(self._entries.values())
357
+
358
+ written = 0
359
+ for entry in entries:
360
+ try:
361
+ logic = WikiLogic(
362
+ name=entry.formula_id,
363
+ content=(
364
+ f"# Logic: {entry.formula_id}\n\n"
365
+ f"**Source formula**: `{entry.formula}`\n\n"
366
+ f"**Source library**: `{entry.source_lib}`\n\n"
367
+ f"**IR**: {entry.ir:.4f} **IC**: {entry.ic_mean:.4f} "
368
+ f"**Rank IC**: {entry.rank_ic:.4f}\n\n"
369
+ f"**Discovered at**: {entry.discovered_at}\n\n"
370
+ f"**Tags**: {', '.join(entry.tags) or '(none)'}\n"
371
+ ),
372
+ source=LogicSource.RESEARCH_REPORT,
373
+ extracted_formula=entry.formula,
374
+ source_detail={
375
+ "source_lib": entry.source_lib,
376
+ "source_id": entry.source_id,
377
+ "tags": ",".join(entry.tags),
378
+ },
379
+ validation_status="pending",
380
+ structured=_load_structured(entry.structured),
381
+ performance_evidence=_load_evidence(entry.evidence),
382
+ refinement_round=(entry.evidence or {}).get("refinement_round", 0),
383
+ created_at=entry.discovered_at or _dt.utcnow().isoformat(timespec="seconds"),
384
+ )
385
+ page = proxy.store_logic(logic)
386
+ with self._lock:
387
+ entry.wiki_path = page
388
+ written += 1
389
+ except Exception as exc:
390
+ with self._lock:
391
+ self._failed_writes.append(
392
+ {
393
+ "formula_id": entry.formula_id,
394
+ "error": repr(exc),
395
+ }
396
+ )
397
+ logger.warning("to_wiki: failed for %s: %s", entry.formula_id, exc)
398
+ return written
399
+
400
+ def failed_writes(self) -> List[Dict[str, str]]:
401
+ """获取写入失败的条目列表 (快照)"""
402
+ with self._lock:
403
+ return list(self._failed_writes)
404
+
405
+ # ------------------------------------------------------------------
406
+ # JSON 离线持久化
407
+ # ------------------------------------------------------------------
408
+ def save_json(self, path: Path) -> None:
409
+ """序列化池到 JSON 文件"""
410
+ path = Path(path)
411
+ path.parent.mkdir(parents=True, exist_ok=True)
412
+ with self._lock:
413
+ payload = {
414
+ "version": 1,
415
+ "wiki_path": self.wiki_path,
416
+ "saved_at": datetime.utcnow().isoformat(timespec="seconds"),
417
+ "n_entries": len(self._entries),
418
+ "entries": [e.to_dict() for e in self._entries.values()],
419
+ "failed_writes": list(self._failed_writes),
420
+ }
421
+ path.write_text(json.dumps(payload, ensure_ascii=False, indent=2), encoding="utf-8")
422
+
423
+ @classmethod
424
+ def load_json(cls, path: Path) -> "FactorPool":
425
+ """从 JSON 文件恢复池"""
426
+ path = Path(path)
427
+ data = json.loads(path.read_text(encoding="utf-8"))
428
+ pool = cls(wiki_path=data.get("wiki_path", "wiki"))
429
+ for ed in data.get("entries", []):
430
+ pool.add(FactorEntry.from_dict(ed))
431
+ with pool._lock:
432
+ pool._failed_writes.extend(data.get("failed_writes", []))
433
+ return pool
434
+
435
+
436
+ def _load_structured(d: Optional[Dict[str, Any]]) -> Any:
437
+ """从 dict 还原 WikiLogicStructured 对象 (用于 wiki 写入)"""
438
+ if not d:
439
+ return None
440
+ try:
441
+ from QuantNodes.research.quant_alpha.logic_mining.models import WikiLogicStructured
442
+
443
+ return WikiLogicStructured.from_dict(d)
444
+ except Exception:
445
+ return None
446
+
447
+
448
+ def _load_evidence(d: Optional[Dict[str, Any]]) -> Any:
449
+ """从 dict 还原 LogicPerformanceEvidence 对象"""
450
+ if not d:
451
+ return None
452
+ try:
453
+ from QuantNodes.research.quant_alpha.logic_mining.models import (
454
+ LogicPerformanceEvidence,
455
+ )
456
+
457
+ return LogicPerformanceEvidence.from_dict(d)
458
+ except Exception:
459
+ return None