cnhkmcp 2.1.2__py3-none-any.whl → 2.1.4__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (101) hide show
  1. cnhkmcp/__init__.py +126 -125
  2. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/config.json +1 -1
  3. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/get_knowledgeBase_tool/ace_lib.py +4 -0
  4. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/001_10_Steps_to_Start_on_BRAIN_documentation.json +14 -0
  5. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/001_Intermediate_Pack_-_Improve_your_Alpha_2_2_documentation.json +174 -0
  6. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/001_Intermediate_Pack_-_Understand_Results_1_2_documentation.json +167 -0
  7. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/001_Introduction_to_Alphas_documentation.json +145 -0
  8. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/001_Introduction_to_BRAIN_Expression_Language_documentation.json +107 -0
  9. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/001_WorldQuant_Challenge_documentation.json +56 -0
  10. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/001__Read_this_First_-_Starter_Pack_documentation.json +404 -0
  11. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/002_How_to_choose_the_Simulation_Settings_documentation.json +268 -0
  12. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/002_Simulate_your_first_Alpha_documentation.json +88 -0
  13. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/002__Alpha_Examples_for_Beginners_documentation.json +254 -0
  14. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/002__Alpha_Examples_for_Bronze_Users_documentation.json +114 -0
  15. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/002__Alpha_Examples_for_Silver_Users_documentation.json +79 -0
  16. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/002__How_BRAIN_works_documentation.json +184 -0
  17. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/003_Clear_these_tests_before_submitting_an_Alpha_documentation.json +388 -0
  18. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/003_Parameters_in_the_Simulation_results_documentation.json +243 -0
  19. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/004_Group_Data_Fields_documentation.json +69 -0
  20. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/004_How_to_use_the_Data_Explorer_documentation.json +142 -0
  21. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/004_Model77_dataset_documentation.json +14 -0
  22. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/004_Sentiment1_dataset_documentation.json +14 -0
  23. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/004_Understanding_Data_in_BRAIN_Key_Concepts_and_Tips_documentation.json +182 -0
  24. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/004_Vector_Data_Fields_documentation.json +30 -0
  25. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/005_Crowding_Risk-Neutralized_Alphas_documentation.json +64 -0
  26. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/005_D0_documentation.json +66 -0
  27. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/005_Double_Neutralization_documentation.json +53 -0
  28. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/005_Fast_D1_Documentation_documentation.json +304 -0
  29. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/005_Investability_Constrained_Metrics_documentation.json +129 -0
  30. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/005_Must-read_posts_How_to_improve_your_Alphas_documentation.json +14 -0
  31. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/005_Neutralization_documentation.json +29 -0
  32. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/005_RAM_Risk-Neutralized_Alphas_documentation.json +64 -0
  33. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/005_Risk_Neutralization_Default_setting_documentation.json +75 -0
  34. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/005_Risk_Neutralized_Alphas_documentation.json +171 -0
  35. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/005_Statistical_Risk-Neutralized_Alphas_documentation.json +51 -0
  36. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/006_EUR_TOP2500_Universe_documentation.json +35 -0
  37. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/006_GLB_TOPDIV3000_Universe_documentation.json +48 -0
  38. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/006_Getting_Started_China_Research_for_Consultants_Gold_documentation.json +142 -0
  39. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/006_Getting_started_on_Illiquid_Universes_Gold_documentation.json +46 -0
  40. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/006_Getting_started_with_USA_TOPSP500_universe_Gold_documentation.json +62 -0
  41. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/006_Global_Alphas_Gold_documentation.json +66 -0
  42. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/006_India_Alphas_documentation.json +35 -0
  43. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/007_Consultant_Dos_and_Don_ts_documentation.json +35 -0
  44. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/007_Consultant_Features_documentation.json +239 -0
  45. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/007_Consultant_Simulation_Features_documentation.json +149 -0
  46. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/007_Consultant_Submission_Tests_documentation.json +363 -0
  47. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/007_Finding_Consultant_Alphas_documentation.json +333 -0
  48. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/007_Power_Pool_Alphas_documentation.json +14 -0
  49. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/007_Research_Advisory_Program_documentation.json +35 -0
  50. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/007_Starting_Guide_for_Research_Consultants_documentation.json +14 -0
  51. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/007_Visualization_Tool_documentation.json +99 -0
  52. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/007_Your_Advisor_-_Kunqi_Jiang_documentation.json +53 -0
  53. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/007__Brain_Genius_documentation.json +288 -0
  54. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/007__Single_Dataset_Alphas_documentation.json +41 -0
  55. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/008_Advisory_Theme_Calendar_documentation.json +14 -0
  56. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/008_Multiplier_Rules_documentation.json +14 -0
  57. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/008_Overview_of_Themes_documentation.json +14 -0
  58. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/008_Theme_Calendar_documentation.json +14 -0
  59. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/009_Combo_Expression_documentation.json +272 -0
  60. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/009_Global_SuperAlphas_documentation.json +14 -0
  61. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/009_Helpful_Tips_documentation.json +58 -0
  62. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/009_Selection_Expression_documentation.json +1546 -0
  63. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/009_SuperAlpha_Operators_documentation.json +890 -0
  64. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/009_SuperAlpha_Results_documentation.json +83 -0
  65. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/009_What_is_a_SuperAlpha_documentation.json +261 -0
  66. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/010_BRAIN_API_documentation.json +515 -0
  67. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/010_Documentation_for_ACE_API_Library_Gold_documentation.json +27 -0
  68. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/010__Understanding_simulation_limits_documentation.json +210 -0
  69. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/arithmetic_operators.json +209 -0
  70. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/cross_sectional_operators.json +98 -0
  71. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/group_operators.json +121 -0
  72. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/logical_operators.json +145 -0
  73. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/reduce_operators.json +156 -0
  74. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/special_operators.json +35 -0
  75. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/time_series_operators.json +386 -0
  76. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/transformational_operators.json +61 -0
  77. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/knowledge/vector_operators.json +38 -0
  78. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/vector_db/_manifest.json +302 -0
  79. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/vector_db/_meta.json +1 -0
  80. cnhkmcp/untracked/AI/321/206/320/261/320/234/321/211/320/255/320/262/321/206/320/237/320/242/321/204/342/225/227/342/225/242/vector_db/chroma.sqlite3 +0 -0
  81. cnhkmcp/untracked/APP/Tranformer/Transformer.py +5 -1
  82. cnhkmcp/untracked/APP/Tranformer/ace_lib.py +4 -0
  83. cnhkmcp/untracked/APP/Tranformer/output/Alpha_candidates.json +6008 -1242
  84. cnhkmcp/untracked/APP/Tranformer/output/Alpha_generated_expressions_error.json +1 -1034
  85. cnhkmcp/untracked/APP/Tranformer/output/Alpha_generated_expressions_success.json +47310 -442
  86. cnhkmcp/untracked/APP/ace_lib.py +4 -0
  87. cnhkmcp/untracked/APP/give_me_idea/ace_lib.py +4 -0
  88. cnhkmcp/untracked/APP/hkSimulator/ace_lib.py +4 -0
  89. cnhkmcp/untracked/APP/simulator/wqb20260107015647.log +57 -0
  90. cnhkmcp/untracked/APP//321/207/342/225/235/320/250/321/205/320/230/320/226/321/204/342/225/225/320/220/321/211/320/221/320/243/321/206/320/261/320/265/ace_lib.py +4 -0
  91. cnhkmcp/untracked/APP//321/207/342/225/235/320/250/321/205/320/230/320/226/321/204/342/225/225/320/220/321/211/320/221/320/243/321/206/320/261/320/265/brain_alpha_inspector.py +2 -2
  92. cnhkmcp/untracked/APP//321/210/342/224/220/320/240/321/210/320/261/320/234/321/206/320/231/320/243/321/205/342/225/235/320/220/321/206/320/230/320/241.py +4 -0
  93. cnhkmcp/untracked/__init__.py +0 -0
  94. cnhkmcp/untracked/mcp/321/206/320/246/320/227/321/204/342/225/227/342/225/242/321/210/320/276/342/225/221/321/205/320/255/320/253/321/207/320/231/320/2302_/321/205/320/266/320/222/321/206/320/256/320/254/321/205/320/236/320/257/321/207/320/231/320/230/321/205/320/240/320/277/321/205/320/232/320/270/321/204/342/225/225/320/235/321/204/342/225/221/320/226/321/206/342/225/241/320/237/321/210/320/267/320/230/321/205/320/251/320/270/321/205/342/226/221/342/226/222/321/210/320/277/320/245/321/210/342/224/220/320/251/321/204/342/225/225/320/272/platform_functions.py +352 -166
  95. {cnhkmcp-2.1.2.dist-info → cnhkmcp-2.1.4.dist-info}/METADATA +1 -1
  96. cnhkmcp-2.1.4.dist-info/RECORD +190 -0
  97. cnhkmcp-2.1.2.dist-info/RECORD +0 -111
  98. {cnhkmcp-2.1.2.dist-info → cnhkmcp-2.1.4.dist-info}/WHEEL +0 -0
  99. {cnhkmcp-2.1.2.dist-info → cnhkmcp-2.1.4.dist-info}/entry_points.txt +0 -0
  100. {cnhkmcp-2.1.2.dist-info → cnhkmcp-2.1.4.dist-info}/licenses/LICENSE +0 -0
  101. {cnhkmcp-2.1.2.dist-info → cnhkmcp-2.1.4.dist-info}/top_level.txt +0 -0
@@ -16,6 +16,7 @@ from dataclasses import dataclass, asdict
16
16
  from datetime import datetime, timedelta
17
17
  import os
18
18
  import sys
19
+ import math
19
20
  from time import sleep
20
21
 
21
22
  import requests
@@ -80,7 +81,20 @@ class BrainApiClient:
80
81
 
81
82
  def log(self, message: str, level: str = "INFO"):
82
83
  """Log messages to stderr to avoid MCP protocol interference."""
83
- print(f"[{level}] {message}", file=sys.stderr)
84
+ try:
85
+ # Try to print with original message first
86
+ print(f"[{level}] {message}", file=sys.stderr)
87
+ except UnicodeEncodeError:
88
+ # Fallback: remove problematic characters and try again
89
+ try:
90
+ safe_message = message.encode('ascii', 'ignore').decode('ascii')
91
+ print(f"[{level}] {safe_message}", file=sys.stderr)
92
+ except Exception:
93
+ # Final fallback: just print the level and a safe message
94
+ print(f"[{level}] Log message", file=sys.stderr)
95
+ except Exception:
96
+ # Final fallback: just print the level and a safe message
97
+ print(f"[{level}] Log message", file=sys.stderr)
84
98
 
85
99
  async def authenticate(self, email: str, password: str) -> Dict[str, Any]:
86
100
  """Authenticate with WorldQuant BRAIN platform with biometric support."""
@@ -322,7 +336,9 @@ class BrainApiClient:
322
336
  print("Alpha done simulating, getting alpha details")
323
337
  alpha_id = simulation_progress.json()["alpha"]
324
338
  alpha = self.session.get("https://api.worldquantbrain.com/alphas/" + alpha_id)
325
- return alpha.json()
339
+ result = alpha.json()
340
+ result['note'] = "if you got a negative alpha sharpe, you can just add a minus sign in front of the last line of the Alpha to flip then think the next step."
341
+ return result
326
342
 
327
343
  except Exception as e:
328
344
  self.log(f"❌ Failed to create simulation: {str(e)}", "ERROR")
@@ -414,12 +430,13 @@ class BrainApiClient:
414
430
  response = self.session.get(f"{self.base_url}/alphas/{alpha_id}/recordsets/pnl")
415
431
  response.raise_for_status()
416
432
 
433
+ # Some alphas may return 204 No Content or an empty body
417
434
  text = (response.text or "").strip()
418
435
  if not text:
419
436
  if attempt < max_retries - 1:
420
437
  self.log(f"Empty PnL response for {alpha_id}, retrying in {retry_delay} seconds...", "WARNING")
421
438
  await asyncio.sleep(retry_delay)
422
- retry_delay *= 1.5
439
+ retry_delay *= 1.5 # Exponential backoff
423
440
  continue
424
441
  else:
425
442
  self.log(f"Empty PnL response after {max_retries} attempts for {alpha_id}", "WARNING")
@@ -440,7 +457,7 @@ class BrainApiClient:
440
457
  self.log(f"Empty PnL JSON after {max_retries} attempts for {alpha_id}", "WARNING")
441
458
  return {}
442
459
 
443
- except json.JSONDecodeError as parse_err:
460
+ except Exception as parse_err:
444
461
  if attempt < max_retries - 1:
445
462
  self.log(f"PnL JSON parse failed for {alpha_id} (attempt {attempt + 1}), retrying in {retry_delay} seconds...", "WARNING")
446
463
  await asyncio.sleep(retry_delay)
@@ -450,7 +467,7 @@ class BrainApiClient:
450
467
  self.log(f"PnL JSON parse failed for {alpha_id} after {max_retries} attempts: {parse_err}", "WARNING")
451
468
  return {}
452
469
 
453
- except requests.RequestException as e:
470
+ except Exception as e:
454
471
  if attempt < max_retries - 1:
455
472
  self.log(f"Failed to get alpha PnL for {alpha_id} (attempt {attempt + 1}), retrying in {retry_delay} seconds: {str(e)}", "WARNING")
456
473
  await asyncio.sleep(retry_delay)
@@ -460,6 +477,7 @@ class BrainApiClient:
460
477
  self.log(f"Failed to get alpha PnL for {alpha_id} after {max_retries} attempts: {str(e)}", "ERROR")
461
478
  raise
462
479
 
480
+ # This should never be reached, but just in case
463
481
  return {}
464
482
 
465
483
  async def get_user_alphas(
@@ -508,12 +526,17 @@ class BrainApiClient:
508
526
  await self.ensure_authenticated()
509
527
 
510
528
  try:
529
+ self.log(f"📤 Submitting alpha {alpha_id} for production...", "INFO")
530
+
511
531
  response = self.session.post(f"{self.base_url}/alphas/{alpha_id}/submit")
512
532
  response.raise_for_status()
513
- return True
533
+
534
+ self.log(f"Alpha {alpha_id} submitted successfully", "SUCCESS")
535
+ return response.__dict__
536
+
514
537
  except Exception as e:
515
- self.log(f"Failed to submit alpha: {str(e)}", "ERROR")
516
- raise
538
+ self.log(f"Failed to submit alpha: {str(e)}", "ERROR")
539
+ return False
517
540
 
518
541
  async def get_events(self) -> Dict[str, Any]:
519
542
  """Get available events and competitions."""
@@ -701,7 +724,13 @@ class BrainApiClient:
701
724
  try:
702
725
  response = self.session.get(f"{self.base_url}/operators")
703
726
  response.raise_for_status()
704
- return response.json()
727
+ operators_data = response.json()
728
+
729
+ # Ensure we return a dictionary format even if API returns a list
730
+ if isinstance(operators_data, list):
731
+ return {"operators": operators_data, "count": len(operators_data)}
732
+ else:
733
+ return operators_data
705
734
  except Exception as e:
706
735
  self.log(f"Failed to get operators: {str(e)}", "ERROR")
707
736
  raise
@@ -761,85 +790,135 @@ class BrainApiClient:
761
790
 
762
791
  async def get_messages(self, limit: Optional[int] = None, offset: int = 0) -> Dict[str, Any]:
763
792
  """Get messages for the current user with optional pagination.
764
-
765
- This function retrieves messages, processes their descriptions to extract
766
- and format embedded JSON, and handles file attachments by saving them locally.
793
+
794
+ Image / large binary payload mitigation:
795
+ Some messages embed base64 encoded images (e.g. <img src="data:image/png;base64,..."/>).
796
+ Returning full base64 can explode token usage for an LLM client. We post-process each
797
+ message description and (by default) extract embedded base64 images to disk and replace
798
+ them with lightweight placeholders while preserving context.
799
+
800
+ Strategies (environment driven in future – currently parameterless public API):
801
+ - placeholder (default): save images to message_images/ and replace with marker text.
802
+ - ignore: strip image tags entirely, leaving a note.
803
+ - keep: leave description unchanged (unsafe for LLM token limits).
804
+
805
+ A message dict gains an 'extracted_images' list when images are processed.
767
806
  """
807
+ await self.ensure_authenticated()
808
+
809
+ import re, base64, pathlib
810
+
811
+ image_handling = os.environ.get("BRAIN_MESSAGE_IMAGE_MODE", "placeholder").lower()
812
+ save_dir = pathlib.Path("message_images")
813
+
768
814
  from typing import Tuple
769
-
770
815
  def process_description(desc: str, message_id: str) -> Tuple[str, List[str]]:
771
- """
772
- Processes message description to handle HTML, embedded images, and JSON.
773
- """
774
- attachments = []
775
-
776
- # Handle embedded images
777
- soup = BeautifulSoup(desc, 'html.parser')
778
- for idx, img_tag in enumerate(soup.find_all('img')):
779
- src = img_tag.get('src', '')
780
- if src.startswith('data:image'):
816
+ try:
817
+ if not desc or image_handling == "keep":
818
+ return desc, []
819
+ attachments: List[str] = []
820
+ # Regex to capture full <img ...> tag with data URI
821
+ img_tag_pattern = re.compile(r"<img[^>]+src=\"(data:image/[^\"]+)\"[^>]*>", re.IGNORECASE)
822
+ # Iterate over unique matches to avoid double work
823
+ matches = list(img_tag_pattern.finditer(desc))
824
+ if not matches:
825
+ # Additional heuristic: very long base64-looking token inside quotes followed by </img>
826
+ # (legacy format noted by user sample). Replace with placeholder.
827
+ heuristic_pattern = re.compile(r"([A-Za-z0-9+/]{500,}={0,2})\"\s*</img>")
828
+ if image_handling != "keep" and heuristic_pattern.search(desc):
829
+ placeholder = "[Embedded image removed - large base64 sequence truncated]"
830
+ return heuristic_pattern.sub(placeholder + "</img>", desc), []
831
+ return desc, []
832
+
833
+ # Ensure save directory exists only if we will store something
834
+ if image_handling == "placeholder" and not save_dir.exists():
781
835
  try:
782
- # Extract image data
783
- header, encoded = src.split(',', 1)
784
- ext = header.split(';')[0].split('/')[1]
785
- safe_ext = re.sub(r'[^a-zA-Z0-9]', '', ext)
786
-
787
- # Decode and save image
788
- content = base64.b64decode(encoded)
789
- file_name = f"{message_id}_img_{idx}.{safe_ext}"
790
- with open(file_name, "wb") as f:
791
- f.write(content)
792
-
793
- # Update HTML and add attachment info
794
- img_tag['src'] = file_name
795
- attachments.append(f"Saved embedded image to ./{file_name}")
796
-
836
+ save_dir.mkdir(parents=True, exist_ok=True)
797
837
  except Exception as e:
798
- attachments.append(f"Could not process embedded image: {e}")
799
-
800
- desc = str(soup)
838
+ self.log(f"Could not create image save directory: {e}", "WARNING")
801
839
 
802
- # Handle JSON content
803
- try:
804
- json_part_match = re.search(r'```json\n({.*?})\n```', desc, re.DOTALL)
805
- if json_part_match:
806
- json_str = json_part_match.group(1)
807
- desc = desc.replace(json_part_match.group(0), "").strip()
808
-
809
- try:
810
- data = json.loads(json_str)
811
- formatted_json = json.dumps(data, indent=2)
812
- desc += f"\n\n---\n**Details**\n```json\n{formatted_json}\n```"
813
- except json.JSONDecodeError:
814
- desc += f"\n\n---\n**Details (raw)**\n{json_str}"
815
- except Exception:
816
- pass
817
-
818
- return desc, attachments
840
+ new_desc = desc
841
+ for idx, match in enumerate(matches, start=1):
842
+ data_uri = match.group(1) # 
843
+ if not data_uri.lower().startswith("data:image"):
844
+ continue
845
+ # Split header and base64 payload
846
+ if "," not in data_uri:
847
+ continue
848
+ header, b64_data = data_uri.split(",", 1)
849
+ mime_part = header.split(";")[0] # data:image/png
850
+ ext = "png"
851
+ if "/" in mime_part:
852
+ ext = mime_part.split("/")[1]
853
+ safe_ext = (ext or "img").split("?")[0]
854
+ placeholder_text = "[Embedded image]"
855
+ if image_handling == "ignore":
856
+ replacement = f"[Image removed: {safe_ext}]"
857
+ elif image_handling == "placeholder":
858
+ # Try decode & save
859
+ file_name = f"{message_id}_{idx}.{safe_ext}"
860
+ file_path = save_dir / file_name
861
+ try:
862
+ # Guard extremely large strings (>5MB ~ 6.7M base64 chars) to avoid memory blow
863
+ if len(b64_data) > 7_000_000:
864
+ raise ValueError("Image too large to decode safely")
865
+ with open(file_path, "wb") as f:
866
+ f.write(base64.b64decode(b64_data))
867
+ attachments.append(str(file_path))
868
+ replacement = f"[Image extracted -> {file_path}]"
869
+ except Exception as e:
870
+ self.log(f"Failed to decode embedded image in message {message_id}: {e}", "WARNING")
871
+ replacement = "[Image extraction failed - content omitted]"
872
+ else: # keep
873
+ replacement = placeholder_text # shouldn't be used since early return, but safe
874
+ # Replace only the matched tag (not global) – use re.sub with count=1 on substring slice
875
+ # Safer to operate on new_desc using the exact matched string
876
+ original_tag = match.group(0)
877
+ new_desc = new_desc.replace(original_tag, replacement, 1)
878
+ return new_desc, attachments
879
+ except UnicodeEncodeError as ue:
880
+ self.log(f"Unicode encoding error in process_description: {ue}", "WARNING")
881
+ return desc, []
882
+ except Exception as e:
883
+ self.log(f"Error in process_description: {e}", "WARNING")
884
+ return desc, []
819
885
 
820
- await self.ensure_authenticated()
821
-
822
886
  try:
823
- params = {"limit": limit, "offset": offset}
824
- params = {k: v for k, v in params.items() if v is not None}
825
-
887
+ params = {}
888
+ if limit is not None:
889
+ params['limit'] = limit
890
+ if offset > 0:
891
+ params['offset'] = offset
892
+
826
893
  response = self.session.get(f"{self.base_url}/users/self/messages", params=params)
827
894
  response.raise_for_status()
828
- messages_data = response.json()
829
-
830
- # Process descriptions and attachments
831
- for msg in messages_data.get("results", []):
832
- try:
833
- msg_id = msg.get("id", "unknown_id")
834
- new_desc, attachments = process_description(msg.get("description", ""), msg_id)
835
- msg["description"] = new_desc
836
- if attachments:
837
- msg["attachments_info"] = attachments
838
- except Exception as e:
839
- self.log(f"Error processing message {msg.get('id')}: {e}", "ERROR")
895
+ data = response.json()
840
896
 
841
- return messages_data
842
-
897
+ # Post-process results for image handling
898
+ results = data.get('results', [])
899
+ for msg in results:
900
+ try:
901
+ desc = msg.get('description')
902
+ processed_desc, attachments = process_description(desc, msg.get('id', 'msg'))
903
+ if attachments or desc != processed_desc:
904
+ msg['description'] = processed_desc
905
+ if attachments:
906
+ msg['extracted_images'] = attachments
907
+ else:
908
+ # If changed but no attachments (ignore mode) mark sanitized
909
+ msg['sanitized'] = True
910
+ except UnicodeEncodeError as ue:
911
+ self.log(f"Unicode encoding error sanitizing message {msg.get('id')}: {ue}", "WARNING")
912
+ # Keep original description if encoding fails
913
+ continue
914
+ except Exception as inner_e:
915
+ self.log(f"Failed to sanitize message {msg.get('id')}: {inner_e}", "WARNING")
916
+ data['results'] = results
917
+ data['image_handling'] = image_handling
918
+ return data
919
+ except UnicodeEncodeError as ue:
920
+ self.log(f"Failed to get messages due to encoding error: {str(ue)}", "ERROR")
921
+ raise
843
922
  except Exception as e:
844
923
  self.log(f"Failed to get messages: {str(e)}", "ERROR")
845
924
  raise
@@ -871,11 +950,11 @@ class BrainApiClient:
871
950
  raise
872
951
 
873
952
  async def get_alpha_yearly_stats(self, alpha_id: str) -> Dict[str, Any]:
874
- """Get yearly statistics for an alpha."""
953
+ """Get yearly statistics for an alpha with retry logic."""
875
954
  await self.ensure_authenticated()
876
955
 
877
956
  max_retries = 5
878
- retry_delay = 2
957
+ retry_delay = 2 # seconds
879
958
 
880
959
  for attempt in range(max_retries):
881
960
  try:
@@ -884,55 +963,62 @@ class BrainApiClient:
884
963
  response = self.session.get(f"{self.base_url}/alphas/{alpha_id}/recordsets/yearly-stats")
885
964
  response.raise_for_status()
886
965
 
966
+ # Check if response has content
887
967
  text = (response.text or "").strip()
888
968
  if not text:
889
969
  if attempt < max_retries - 1:
890
- self.log(f"Empty yearly stats response for {alpha_id}, retrying...", "WARNING")
970
+ self.log(f"Empty yearly stats response for {alpha_id}, retrying in {retry_delay} seconds...", "WARNING")
891
971
  await asyncio.sleep(retry_delay)
892
- retry_delay *= 1.5
972
+ retry_delay *= 1.5 # Exponential backoff
893
973
  continue
894
974
  else:
975
+ self.log(f"Empty yearly stats response after {max_retries} attempts for {alpha_id}", "WARNING")
895
976
  return {}
896
977
 
897
978
  try:
898
- stats_data = response.json()
899
- if stats_data:
900
- return stats_data
979
+ yearly_stats = response.json()
980
+ if yearly_stats:
981
+ self.log(f"Successfully retrieved yearly stats for alpha {alpha_id}", "SUCCESS")
982
+ return yearly_stats
901
983
  else:
902
984
  if attempt < max_retries - 1:
903
- self.log(f"Empty yearly stats JSON for {alpha_id}, retrying...", "WARNING")
985
+ self.log(f"Empty yearly stats JSON for {alpha_id}, retrying in {retry_delay} seconds...", "WARNING")
904
986
  await asyncio.sleep(retry_delay)
905
987
  retry_delay *= 1.5
906
988
  continue
907
989
  else:
990
+ self.log(f"Empty yearly stats JSON after {max_retries} attempts for {alpha_id}", "WARNING")
908
991
  return {}
909
992
 
910
- except json.JSONDecodeError as parse_err:
993
+ except Exception as parse_err:
911
994
  if attempt < max_retries - 1:
912
- self.log(f"Yearly stats JSON parse failed for {alpha_id}, retrying...", "WARNING")
995
+ self.log(f"Yearly stats JSON parse failed for {alpha_id} (attempt {attempt + 1}), retrying in {retry_delay} seconds...", "WARNING")
913
996
  await asyncio.sleep(retry_delay)
914
997
  retry_delay *= 1.5
915
998
  continue
916
999
  else:
917
- raise
1000
+ self.log(f"Yearly stats JSON parse failed for {alpha_id} after {max_retries} attempts: {parse_err}", "WARNING")
1001
+ return {}
918
1002
 
919
- except requests.RequestException as e:
1003
+ except Exception as e:
920
1004
  if attempt < max_retries - 1:
921
- self.log(f"Failed to get yearly stats for {alpha_id}, retrying: {e}", "WARNING")
1005
+ self.log(f"Failed to get alpha yearly stats for {alpha_id} (attempt {attempt + 1}), retrying in {retry_delay} seconds: {str(e)}", "WARNING")
922
1006
  await asyncio.sleep(retry_delay)
923
1007
  retry_delay *= 1.5
924
1008
  continue
925
1009
  else:
1010
+ self.log(f"Failed to get alpha yearly stats for {alpha_id} after {max_retries} attempts: {str(e)}", "ERROR")
926
1011
  raise
927
1012
 
1013
+ # This should never be reached, but just in case
928
1014
  return {}
929
1015
 
930
1016
  async def get_production_correlation(self, alpha_id: str) -> Dict[str, Any]:
931
- """Get production correlation data for an alpha."""
1017
+ """Get production correlation data for an alpha with retry logic."""
932
1018
  await self.ensure_authenticated()
933
1019
 
934
1020
  max_retries = 5
935
- retry_delay = 2
1021
+ retry_delay = 20 # seconds
936
1022
 
937
1023
  for attempt in range(max_retries):
938
1024
  try:
@@ -953,44 +1039,46 @@ class BrainApiClient:
953
1039
  return {}
954
1040
 
955
1041
  try:
956
- corr_data = response.json()
957
- if corr_data:
958
- return corr_data
1042
+ correlation_data = response.json()
1043
+ if correlation_data:
1044
+ self.log(f"Successfully retrieved production correlation for alpha {alpha_id}", "SUCCESS")
1045
+ return correlation_data
959
1046
  else:
960
1047
  if attempt < max_retries - 1:
961
- self.log(f"Empty production correlation JSON for {alpha_id}, retrying...", "WARNING")
1048
+ self.log(f"Empty production correlation JSON for {alpha_id}, retrying in {retry_delay} seconds...", "WARNING")
962
1049
  await asyncio.sleep(retry_delay)
963
- retry_delay *= 1.5
964
1050
  continue
965
1051
  else:
1052
+ self.log(f"Empty production correlation JSON after {max_retries} attempts for {alpha_id}", "WARNING")
966
1053
  return {}
967
1054
 
968
- except json.JSONDecodeError:
1055
+ except Exception as parse_err:
969
1056
  if attempt < max_retries - 1:
970
- self.log(f"Production correlation JSON parse failed for {alpha_id}, retrying...", "WARNING")
1057
+ self.log(f"Production correlation JSON parse failed for {alpha_id} (attempt {attempt + 1}), retrying in {retry_delay} seconds...", "WARNING")
971
1058
  await asyncio.sleep(retry_delay)
972
- retry_delay *= 1.5
973
1059
  continue
974
1060
  else:
975
- raise
1061
+ self.log(f"Production correlation JSON parse failed for {alpha_id} after {max_retries} attempts: {parse_err}", "WARNING")
1062
+ return {}
976
1063
 
977
- except requests.RequestException as e:
1064
+ except Exception as e:
978
1065
  if attempt < max_retries - 1:
979
- self.log(f"Failed to get production correlation for {alpha_id}, retrying: {e}", "WARNING")
1066
+ self.log(f"Failed to get production correlation for {alpha_id} (attempt {attempt + 1}), retrying in {retry_delay} seconds: {str(e)}", "WARNING")
980
1067
  await asyncio.sleep(retry_delay)
981
- retry_delay *= 1.5
982
1068
  continue
983
1069
  else:
1070
+ self.log(f"Failed to get production correlation for {alpha_id} after {max_retries} attempts: {str(e)}", "ERROR")
984
1071
  raise
985
1072
 
1073
+ # This should never be reached, but just in case
986
1074
  return {}
987
1075
 
988
1076
  async def get_self_correlation(self, alpha_id: str) -> Dict[str, Any]:
989
- """Get self correlation data for an alpha."""
1077
+ """Get self-correlation data for an alpha with retry logic."""
990
1078
  await self.ensure_authenticated()
991
1079
 
992
1080
  max_retries = 5
993
- retry_delay = 2
1081
+ retry_delay = 20 # seconds
994
1082
 
995
1083
  for attempt in range(max_retries):
996
1084
  try:
@@ -1011,36 +1099,38 @@ class BrainApiClient:
1011
1099
  return {}
1012
1100
 
1013
1101
  try:
1014
- corr_data = response.json()
1015
- if corr_data:
1016
- return corr_data
1102
+ correlation_data = response.json()
1103
+ if correlation_data:
1104
+ self.log(f"Successfully retrieved self correlation for alpha {alpha_id}", "SUCCESS")
1105
+ return correlation_data
1017
1106
  else:
1018
1107
  if attempt < max_retries - 1:
1019
- self.log(f"Empty self correlation JSON for {alpha_id}, retrying...", "WARNING")
1108
+ self.log(f"Empty self correlation JSON for {alpha_id}, retrying in {retry_delay} seconds...", "WARNING")
1020
1109
  await asyncio.sleep(retry_delay)
1021
- retry_delay *= 1.5
1022
1110
  continue
1023
1111
  else:
1112
+ self.log(f"Empty self correlation JSON after {max_retries} attempts for {alpha_id}", "WARNING")
1024
1113
  return {}
1025
1114
 
1026
- except json.JSONDecodeError:
1115
+ except Exception as parse_err:
1027
1116
  if attempt < max_retries - 1:
1028
- self.log(f"Self correlation JSON parse failed for {alpha_id}, retrying...", "WARNING")
1117
+ self.log(f"Self correlation JSON parse failed for {alpha_id} (attempt {attempt + 1}), retrying in {retry_delay} seconds...", "WARNING")
1029
1118
  await asyncio.sleep(retry_delay)
1030
- retry_delay *= 1.5
1031
1119
  continue
1032
1120
  else:
1033
- raise
1121
+ self.log(f"Self correlation JSON parse failed for {alpha_id} after {max_retries} attempts: {parse_err}", "WARNING")
1122
+ return {}
1034
1123
 
1035
- except requests.RequestException as e:
1124
+ except Exception as e:
1036
1125
  if attempt < max_retries - 1:
1037
- self.log(f"Failed to get self correlation for {alpha_id}, retrying: {e}", "WARNING")
1126
+ self.log(f"Failed to get self correlation for {alpha_id} (attempt {attempt + 1}), retrying in {retry_delay} seconds: {str(e)}", "WARNING")
1038
1127
  await asyncio.sleep(retry_delay)
1039
- retry_delay *= 1.5
1040
1128
  continue
1041
1129
  else:
1130
+ self.log(f"Failed to get self correlation for {alpha_id} after {max_retries} attempts: {str(e)}", "ERROR")
1042
1131
  raise
1043
1132
 
1133
+ # This should never be reached, but just in case
1044
1134
  return {}
1045
1135
 
1046
1136
  async def check_correlation(self, alpha_id: str, correlation_type: str = "both", threshold: float = 0.7) -> Dict[str, Any]:
@@ -1048,22 +1138,84 @@ class BrainApiClient:
1048
1138
  await self.ensure_authenticated()
1049
1139
 
1050
1140
  try:
1051
- results = {}
1052
- if correlation_type in ["production", "both"]:
1053
- results["production"] = await self.get_production_correlation(alpha_id)
1054
- if correlation_type in ["self", "both"]:
1055
- results["self"] = await self.get_self_correlation(alpha_id)
1056
-
1057
- # Add analysis based on threshold
1058
- for key, data in results.items():
1059
- if "correlation" in data and "sharpe" in data["correlation"]:
1060
- high_corr = [item for item in data["correlation"]["sharpe"] if abs(item.get("corr", 0)) > threshold]
1061
- data["analysis"] = {
1062
- "highly_correlated_count": len(high_corr),
1063
- "highly_correlated_alphas": high_corr
1064
- }
1141
+ results = {
1142
+ 'alpha_id': alpha_id,
1143
+ 'threshold': threshold,
1144
+ 'correlation_type': correlation_type,
1145
+ 'checks': {}
1146
+ }
1147
+
1148
+ # Determine which correlations to check
1149
+ check_types = []
1150
+ if correlation_type == "both":
1151
+ check_types = ["production", "self"]
1152
+ else:
1153
+ check_types = [correlation_type]
1154
+
1155
+ all_passed = True
1156
+
1157
+ for check_type in check_types:
1158
+ if check_type == "production":
1159
+ correlation_data = await self.get_production_correlation(alpha_id)
1160
+ elif check_type == "self":
1161
+ correlation_data = await self.get_self_correlation(alpha_id)
1162
+ else:
1163
+ continue
1164
+
1165
+ # Analyze correlation data (robust to schema/records format)
1166
+ if isinstance(correlation_data, dict):
1167
+ # Prefer strict access to schema.max or top-level max; otherwise error
1168
+ schema = correlation_data.get('schema') or {}
1169
+ if isinstance(schema, dict) and 'max' in schema:
1170
+ max_correlation = float(schema['max'])
1171
+ elif 'max' in correlation_data:
1172
+ # Some endpoints place max at top-level
1173
+ max_correlation = float(correlation_data['max'])
1174
+ else:
1175
+ # Attempt to derive from records; if none found, raise error instead of defaulting
1176
+ records = correlation_data.get('records') or []
1177
+ if isinstance(records, list) and records:
1178
+ candidate_max = None
1179
+ for row in records:
1180
+ if isinstance(row, (list, tuple)):
1181
+ for v in row:
1182
+ try:
1183
+ vf = float(v)
1184
+ if -1.0 <= vf <= 1.0:
1185
+ candidate_max = vf if candidate_max is None else max(candidate_max, vf)
1186
+ except Exception:
1187
+ continue
1188
+ elif isinstance(row, dict):
1189
+ for key in ('correlation', 'prodCorrelation', 'selfCorrelation', 'max'):
1190
+ try:
1191
+ vf = float(row.get(key))
1192
+ if -1.0 <= vf <= 1.0:
1193
+ candidate_max = vf if candidate_max is None else max(candidate_max, vf)
1194
+ except Exception:
1195
+ continue
1196
+ if candidate_max is None:
1197
+ raise ValueError("Unable to derive max correlation from records")
1198
+ max_correlation = float(candidate_max)
1199
+ else:
1200
+ raise KeyError("Correlation response missing 'schema.max' or top-level 'max' and no 'records' to derive from")
1201
+ else:
1202
+ raise TypeError("Correlation data is not a dictionary")
1065
1203
 
1204
+ passes_check = max_correlation < threshold
1205
+
1206
+ results['checks'][check_type] = {
1207
+ 'max_correlation': max_correlation,
1208
+ 'passes_check': passes_check,
1209
+ 'correlation_data': correlation_data
1210
+ }
1211
+
1212
+ if not passes_check:
1213
+ all_passed = False
1214
+
1215
+ results['all_passed'] = all_passed
1216
+
1066
1217
  return results
1218
+
1067
1219
  except Exception as e:
1068
1220
  self.log(f"Failed to check correlation: {str(e)}", "ERROR")
1069
1221
  raise
@@ -1073,20 +1225,22 @@ class BrainApiClient:
1073
1225
  await self.ensure_authenticated()
1074
1226
 
1075
1227
  try:
1076
- # This endpoint might not exist, so we simulate it by calling other functions
1077
- # In a real scenario, this would be a single API call
1228
+ # Get correlation checks using the unified function
1229
+ correlation_checks = await self.check_correlation(alpha_id, correlation_type="both")
1078
1230
 
1079
- pnl_data = await self.get_alpha_pnl(alpha_id)
1080
- yearly_stats = await self.get_alpha_yearly_stats(alpha_id)
1081
- correlation = await self.check_correlation(alpha_id)
1231
+ # Get alpha details for additional validation
1232
+ alpha_details = await self.get_alpha_details(alpha_id)
1082
1233
 
1083
- return {
1084
- "pnl_summary": pnl_data.get("pnlSummary", {}),
1085
- "yearly_stats": yearly_stats,
1086
- "correlation": correlation
1234
+ # Compile comprehensive check results
1235
+ checks = {
1236
+ 'correlation_checks': correlation_checks,
1237
+ 'alpha_details': alpha_details,
1238
+ 'all_passed': correlation_checks['all_passed']
1087
1239
  }
1240
+
1241
+ return checks
1088
1242
  except Exception as e:
1089
- self.log(f"Failed submission check: {str(e)}", "ERROR")
1243
+ self.log(f"Failed to get submission check: {str(e)}", "ERROR")
1090
1244
  raise
1091
1245
 
1092
1246
  async def set_alpha_properties(self, alpha_id: str, name: Optional[str] = None,
@@ -1096,18 +1250,19 @@ class BrainApiClient:
1096
1250
  await self.ensure_authenticated()
1097
1251
 
1098
1252
  try:
1099
- payload = {
1100
- "name": name,
1101
- "color": color,
1102
- "tags": tags,
1103
- "descriptions": {
1104
- "selection": selection_desc,
1105
- "combo": combo_desc
1106
- }
1107
- }
1108
- payload = {k: v for k, v in payload.items() if v is not None}
1253
+ data = {}
1254
+ if name:
1255
+ data['name'] = name
1256
+ if color:
1257
+ data['color'] = color
1258
+ if tags:
1259
+ data['tags'] = tags
1260
+ if selection_desc:
1261
+ data['selectionDesc'] = selection_desc
1262
+ if combo_desc:
1263
+ data['comboDesc'] = combo_desc
1109
1264
 
1110
- response = self.session.patch(f"{self.base_url}/alphas/{alpha_id}", json=payload)
1265
+ response = self.session.patch(f"{self.base_url}/alphas/{alpha_id}", json=data)
1111
1266
  response.raise_for_status()
1112
1267
  return response.json()
1113
1268
  except Exception as e:
@@ -1174,11 +1329,35 @@ class BrainApiClient:
1174
1329
  try:
1175
1330
  params = {}
1176
1331
  if start_date:
1177
- params["startDate"] = start_date
1332
+ params['startDate'] = start_date
1178
1333
  if end_date:
1179
- params["endDate"] = end_date
1180
-
1334
+ params['endDate'] = end_date
1335
+
1336
+ # Try the user-specific activities endpoint first (like pyramid-multipliers)
1181
1337
  response = self.session.get(f"{self.base_url}/users/self/activities/pyramid-alphas", params=params)
1338
+
1339
+ # If that fails, try alternative endpoints
1340
+ if response.status_code == 404:
1341
+ # Try alternative endpoint structure
1342
+ response = self.session.get(f"{self.base_url}/users/self/pyramid/alphas", params=params)
1343
+
1344
+ if response.status_code == 404:
1345
+ # Try yet another alternative
1346
+ response = self.session.get(f"{self.base_url}/activities/pyramid-alphas", params=params)
1347
+
1348
+ if response.status_code == 404:
1349
+ # Return an informative error with what we tried
1350
+ return {
1351
+ "error": "Pyramid alphas endpoint not found",
1352
+ "tried_endpoints": [
1353
+ "/users/self/activities/pyramid-alphas",
1354
+ "/users/self/pyramid/alphas",
1355
+ "/activities/pyramid-alphas",
1356
+ "/pyramid/alphas"
1357
+ ],
1358
+ "suggestion": "This endpoint may not be available in the current API version"
1359
+ }
1360
+
1182
1361
  response.raise_for_status()
1183
1362
  return response.json()
1184
1363
  except Exception as e:
@@ -1451,20 +1630,27 @@ async def manage_config(action: str = "get", settings: Optional[Dict[str, Any]]
1451
1630
  Returns:
1452
1631
  Current or updated configuration including authentication status
1453
1632
  """
1454
- config = load_config()
1633
+ if action == "get":
1634
+ config = load_config()
1635
+ auth_status = await brain_client.get_authentication_status()
1636
+
1637
+ return {
1638
+ "config": config,
1639
+ "auth_status": auth_status,
1640
+ "is_authenticated": await brain_client.is_authenticated()
1641
+ }
1455
1642
 
1456
- if action == "set" and settings:
1643
+ elif action == "set":
1644
+ if settings is None:
1645
+ return {"error": "Settings parameter is required when action='set'"}
1646
+
1647
+ config = load_config()
1457
1648
  config.update(settings)
1458
1649
  save_config(config)
1459
-
1460
- is_authed = await brain_client.is_authenticated()
1461
- config['isAuthenticated'] = is_authed
1650
+ return config
1462
1651
 
1463
- # Mask password for security
1464
- if 'password' in config:
1465
- config['password'] = '********'
1466
-
1467
- return config
1652
+ else:
1653
+ return {"error": f"Invalid action '{action}'. Use 'get' or 'set'."}
1468
1654
 
1469
1655
  # --- Simulation Tools ---
1470
1656