synth-ai 0.2.14__py3-none-any.whl → 0.2.17__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.

Potentially problematic release.


This version of synth-ai might be problematic. Click here for more details.

Files changed (354) hide show
  1. examples/README.md +1 -0
  2. examples/analyze_semantic_words.sh +2 -2
  3. examples/blog_posts/pokemon_vl/README.md +98 -0
  4. examples/blog_posts/pokemon_vl/configs/eval_qwen3_vl.toml +25 -0
  5. examples/blog_posts/pokemon_vl/configs/eval_rl_final.toml +24 -0
  6. examples/blog_posts/pokemon_vl/configs/filter_high_reward.toml +10 -0
  7. examples/blog_posts/pokemon_vl/configs/train_rl_from_sft.toml +42 -0
  8. examples/blog_posts/pokemon_vl/configs/train_sft_qwen4b_vl.toml +40 -0
  9. examples/blog_posts/warming_up_to_rl/README.md +158 -0
  10. examples/blog_posts/warming_up_to_rl/configs/eval_ft_qwen4b.toml +25 -0
  11. examples/blog_posts/warming_up_to_rl/configs/eval_groq_qwen32b.toml +25 -0
  12. examples/blog_posts/warming_up_to_rl/configs/eval_openai_gpt_oss_120b.toml +29 -0
  13. examples/blog_posts/warming_up_to_rl/configs/filter_high_reward_dataset.toml +10 -0
  14. examples/blog_posts/warming_up_to_rl/configs/train_rl_from_sft.toml +41 -0
  15. examples/blog_posts/warming_up_to_rl/configs/train_sft_qwen4b.toml +40 -0
  16. examples/dev/qwen3_32b_qlora_4xh100.toml +5 -0
  17. examples/multi_step/SFT_README.md +147 -0
  18. examples/multi_step/configs/crafter_rl_outcome.toml +1 -1
  19. examples/multi_step/configs/crafter_rl_stepwise_hosted_judge.toml +73 -115
  20. examples/multi_step/configs/crafter_rl_stepwise_shaped.toml +1 -1
  21. examples/multi_step/configs/crafter_rl_stepwise_simple.toml +1 -1
  22. examples/multi_step/configs/crafter_rl_stepwise_simple_NEW_FORMAT.toml +105 -0
  23. examples/multi_step/configs/crafter_sft_qwen30b_lora.toml +62 -0
  24. examples/multi_step/configs/verilog_rl_lora.toml +80 -123
  25. examples/multi_step/convert_traces_to_sft.py +84 -0
  26. examples/multi_step/run_sft_qwen30b.sh +45 -0
  27. examples/qwen_coder/configs/coder_lora_30b.toml +1 -2
  28. examples/qwen_coder/configs/coder_lora_4b.toml +5 -1
  29. examples/qwen_coder/configs/coder_lora_small.toml +1 -2
  30. examples/qwen_vl/BUGS_AND_FIXES.md +232 -0
  31. examples/qwen_vl/IMAGE_VALIDATION_COMPLETE.md +271 -0
  32. examples/qwen_vl/IMAGE_VALIDATION_SUMMARY.md +260 -0
  33. examples/qwen_vl/INFERENCE_SFT_TESTS.md +412 -0
  34. examples/qwen_vl/NEXT_STEPS_2B.md +325 -0
  35. examples/qwen_vl/QUICKSTART.md +327 -0
  36. examples/qwen_vl/QUICKSTART_RL_VISION.md +110 -0
  37. examples/qwen_vl/README.md +152 -0
  38. examples/qwen_vl/RL_VISION_COMPLETE.md +475 -0
  39. examples/qwen_vl/RL_VISION_TESTING.md +333 -0
  40. examples/qwen_vl/SDK_VISION_INTEGRATION.md +328 -0
  41. examples/qwen_vl/SETUP_COMPLETE.md +274 -0
  42. examples/qwen_vl/VISION_TESTS_COMPLETE.md +489 -0
  43. examples/qwen_vl/VLM_PIPELINE_COMPLETE.md +242 -0
  44. examples/qwen_vl/__init__.py +2 -0
  45. examples/qwen_vl/collect_data_via_cli.md +415 -0
  46. examples/qwen_vl/collect_vision_traces.py +368 -0
  47. examples/qwen_vl/configs/crafter_rl_vision_qwen3vl4b.toml +110 -0
  48. examples/qwen_vl/configs/crafter_vlm_sft_example.toml +59 -0
  49. examples/qwen_vl/configs/eval_gpt4o_mini_vision.toml +26 -0
  50. examples/qwen_vl/configs/eval_gpt4o_vision_proper.toml +29 -0
  51. examples/qwen_vl/configs/eval_gpt5nano_vision.toml +26 -0
  52. examples/qwen_vl/configs/eval_qwen3vl_vision.toml +26 -0
  53. examples/qwen_vl/configs/filter_qwen3vl_sft.toml +49 -0
  54. examples/qwen_vl/configs/filter_vision_sft.toml +52 -0
  55. examples/qwen_vl/configs/filter_vision_test.toml +8 -0
  56. examples/qwen_vl/configs/sft_qwen3_vl_2b_test.toml +54 -0
  57. examples/qwen_vl/crafter_gpt5nano_agent.py +308 -0
  58. examples/qwen_vl/crafter_qwen_vl_agent.py +300 -0
  59. examples/qwen_vl/run_vision_comparison.sh +61 -0
  60. examples/qwen_vl/run_vision_sft_pipeline.sh +175 -0
  61. examples/qwen_vl/test_image_validation.py +201 -0
  62. examples/qwen_vl/test_sft_vision_data.py +110 -0
  63. examples/rl/README.md +6 -6
  64. examples/rl/configs/eval_base_qwen.toml +17 -0
  65. examples/rl/configs/eval_rl_qwen.toml +13 -0
  66. examples/rl/configs/rl_from_base_qwen.toml +62 -0
  67. examples/rl/configs/rl_from_base_qwen17.toml +79 -0
  68. examples/rl/configs/rl_from_ft_qwen.toml +37 -0
  69. examples/rl/run_eval.py +436 -0
  70. examples/rl/run_rl_and_save.py +111 -0
  71. examples/rl/task_app/README.md +21 -0
  72. examples/rl/task_app/math_single_step.py +990 -0
  73. examples/rl/task_app/math_task_app.py +111 -0
  74. examples/run_crafter_demo.sh +2 -2
  75. examples/sft/README.md +6 -6
  76. examples/sft/configs/crafter_fft_qwen0p6b.toml +7 -2
  77. examples/sft/configs/crafter_lora_qwen0p6b.toml +7 -3
  78. examples/sft/evaluate.py +2 -4
  79. examples/sft/export_dataset.py +7 -4
  80. examples/swe/task_app/README.md +33 -3
  81. examples/swe/task_app/grpo_swe_mini.py +4 -1
  82. examples/swe/task_app/grpo_swe_mini_task_app.py +0 -12
  83. examples/swe/task_app/hosted/envs/crafter/react_agent.py +1 -1
  84. examples/swe/task_app/hosted/envs/mini_swe/environment.py +50 -23
  85. examples/swe/task_app/hosted/inference/openai_client.py +4 -4
  86. examples/swe/task_app/hosted/policy_routes.py +0 -2
  87. examples/swe/task_app/hosted/rollout.py +0 -8
  88. examples/swe/task_app/morph_backend.py +178 -0
  89. examples/task_apps/crafter/task_app/README.md +1 -1
  90. examples/task_apps/crafter/task_app/grpo_crafter.py +70 -10
  91. examples/task_apps/crafter/task_app/grpo_crafter_task_app.py +1 -1
  92. examples/task_apps/crafter/task_app/synth_envs_hosted/envs/crafter/policy.py +63 -27
  93. examples/task_apps/crafter/task_app/synth_envs_hosted/envs/crafter/react_agent.py +1 -2
  94. examples/task_apps/crafter/task_app/synth_envs_hosted/inference/openai_client.py +48 -50
  95. examples/task_apps/crafter/task_app/synth_envs_hosted/policy_routes.py +75 -36
  96. examples/task_apps/crafter/task_app/synth_envs_hosted/rollout.py +31 -15
  97. examples/task_apps/enron/__init__.py +1 -0
  98. examples/task_apps/enron/task_app/grpo_enron_task_app.py +1 -1
  99. examples/task_apps/math/README.md +1 -2
  100. examples/task_apps/pokemon_red/README.md +3 -4
  101. examples/task_apps/pokemon_red/eval_image_only_gpt4o.toml +6 -5
  102. examples/task_apps/pokemon_red/eval_pokemon_red_policy.py +1 -2
  103. examples/task_apps/pokemon_red/task_app.py +36 -5
  104. examples/task_apps/sokoban/README.md +2 -3
  105. examples/task_apps/verilog/eval_groq_qwen32b.toml +12 -14
  106. examples/task_apps/verilog/task_app/grpo_verilog_task_app.py +1 -1
  107. examples/vlm/README.md +3 -3
  108. examples/vlm/configs/crafter_vlm_gpt4o.toml +5 -0
  109. examples/vlm/crafter_openai_vlm_agent.py +3 -5
  110. examples/vlm/filter_image_rows.py +1 -1
  111. examples/vlm/run_crafter_vlm_benchmark.py +2 -2
  112. examples/warming_up_to_rl/_utils.py +92 -0
  113. examples/warming_up_to_rl/analyze_trace_db.py +1 -1
  114. examples/warming_up_to_rl/configs/crafter_fft.toml +5 -0
  115. examples/warming_up_to_rl/configs/eval_fft_qwen4b.toml +2 -0
  116. examples/warming_up_to_rl/configs/eval_groq_qwen32b.toml +2 -0
  117. examples/warming_up_to_rl/configs/eval_modal_qwen4b.toml +2 -1
  118. examples/warming_up_to_rl/configs/rl_from_base_qwen4b.toml +2 -1
  119. examples/warming_up_to_rl/configs/rl_from_ft.toml +2 -0
  120. examples/warming_up_to_rl/export_trace_sft.py +174 -60
  121. examples/warming_up_to_rl/readme.md +63 -132
  122. examples/warming_up_to_rl/run_fft_and_save.py +1 -1
  123. examples/warming_up_to_rl/run_local_rollout_traced.py +1 -1
  124. examples/warming_up_to_rl/run_rl_and_save.py +1 -1
  125. examples/warming_up_to_rl/task_app/README.md +42 -0
  126. examples/warming_up_to_rl/task_app/grpo_crafter.py +827 -0
  127. examples/warming_up_to_rl/task_app/grpo_crafter_task_app.py +135 -0
  128. examples/warming_up_to_rl/task_app/synth_envs_hosted/README.md +173 -0
  129. examples/warming_up_to_rl/task_app/synth_envs_hosted/__init__.py +5 -0
  130. examples/warming_up_to_rl/task_app/synth_envs_hosted/branching.py +143 -0
  131. examples/warming_up_to_rl/task_app/synth_envs_hosted/environment_routes.py +1226 -0
  132. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/__init__.py +1 -0
  133. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/__init__.py +6 -0
  134. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/app.py +1 -0
  135. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/environment.py +522 -0
  136. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/policy.py +454 -0
  137. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/react_agent.py +108 -0
  138. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/shared.py +305 -0
  139. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/tools.py +47 -0
  140. examples/warming_up_to_rl/task_app/synth_envs_hosted/hosted_app.py +204 -0
  141. examples/warming_up_to_rl/task_app/synth_envs_hosted/inference/__init__.py +5 -0
  142. examples/warming_up_to_rl/task_app/synth_envs_hosted/inference/openai_client.py +618 -0
  143. examples/warming_up_to_rl/task_app/synth_envs_hosted/main.py +100 -0
  144. examples/warming_up_to_rl/task_app/synth_envs_hosted/policy_routes.py +1084 -0
  145. examples/warming_up_to_rl/task_app/synth_envs_hosted/registry.py +195 -0
  146. examples/warming_up_to_rl/task_app/synth_envs_hosted/rollout.py +1861 -0
  147. examples/warming_up_to_rl/task_app/synth_envs_hosted/storage/__init__.py +5 -0
  148. examples/warming_up_to_rl/task_app/synth_envs_hosted/storage/volume.py +211 -0
  149. examples/warming_up_to_rl/task_app/synth_envs_hosted/test_agents.py +161 -0
  150. examples/warming_up_to_rl/task_app/synth_envs_hosted/test_service.py +137 -0
  151. examples/warming_up_to_rl/task_app/synth_envs_hosted/utils.py +62 -0
  152. examples/workflows/math_rl/configs/rl_from_base_qwen.toml +27 -0
  153. examples/workflows/math_rl/configs/rl_from_base_qwen17.toml +5 -0
  154. synth_ai/__init__.py +44 -30
  155. synth_ai/_utils/__init__.py +47 -0
  156. synth_ai/_utils/base_url.py +10 -0
  157. synth_ai/_utils/http.py +10 -0
  158. synth_ai/_utils/prompts.py +10 -0
  159. synth_ai/_utils/task_app_state.py +12 -0
  160. synth_ai/_utils/user_config.py +10 -0
  161. synth_ai/api/models/supported.py +144 -7
  162. synth_ai/api/train/__init__.py +13 -1
  163. synth_ai/api/train/builders.py +9 -3
  164. synth_ai/api/train/cli.py +155 -17
  165. synth_ai/api/train/config_finder.py +18 -11
  166. synth_ai/api/train/configs/__init__.py +8 -1
  167. synth_ai/api/train/configs/rl.py +32 -7
  168. synth_ai/api/train/configs/sft.py +6 -2
  169. synth_ai/api/train/configs/shared.py +59 -2
  170. synth_ai/api/train/env_resolver.py +13 -10
  171. synth_ai/auth/credentials.py +119 -0
  172. synth_ai/cli/__init__.py +61 -69
  173. synth_ai/cli/_modal_wrapper.py +7 -5
  174. synth_ai/cli/_typer_patch.py +0 -2
  175. synth_ai/cli/_validate_task_app.py +22 -4
  176. synth_ai/cli/commands/__init__.py +17 -0
  177. synth_ai/cli/commands/demo/__init__.py +6 -0
  178. synth_ai/cli/commands/demo/core.py +163 -0
  179. synth_ai/cli/commands/deploy/__init__.py +23 -0
  180. synth_ai/cli/commands/deploy/core.py +614 -0
  181. synth_ai/cli/commands/deploy/errors.py +72 -0
  182. synth_ai/cli/commands/deploy/validation.py +11 -0
  183. synth_ai/cli/commands/eval/__init__.py +19 -0
  184. synth_ai/cli/commands/eval/core.py +1109 -0
  185. synth_ai/cli/commands/eval/errors.py +81 -0
  186. synth_ai/cli/commands/eval/validation.py +133 -0
  187. synth_ai/cli/commands/filter/__init__.py +12 -0
  188. synth_ai/cli/commands/filter/core.py +388 -0
  189. synth_ai/cli/commands/filter/errors.py +55 -0
  190. synth_ai/cli/commands/filter/validation.py +77 -0
  191. synth_ai/cli/commands/help/__init__.py +177 -0
  192. synth_ai/cli/commands/help/core.py +73 -0
  193. synth_ai/cli/commands/status/__init__.py +64 -0
  194. synth_ai/cli/commands/status/client.py +192 -0
  195. synth_ai/cli/commands/status/config.py +92 -0
  196. synth_ai/cli/commands/status/errors.py +20 -0
  197. synth_ai/cli/commands/status/formatters.py +164 -0
  198. synth_ai/cli/commands/status/subcommands/__init__.py +9 -0
  199. synth_ai/cli/commands/status/subcommands/files.py +79 -0
  200. synth_ai/cli/commands/status/subcommands/jobs.py +334 -0
  201. synth_ai/cli/commands/status/subcommands/models.py +79 -0
  202. synth_ai/cli/commands/status/subcommands/runs.py +81 -0
  203. synth_ai/cli/commands/status/subcommands/summary.py +47 -0
  204. synth_ai/cli/commands/status/utils.py +114 -0
  205. synth_ai/cli/commands/train/__init__.py +53 -0
  206. synth_ai/cli/commands/train/core.py +21 -0
  207. synth_ai/cli/commands/train/errors.py +117 -0
  208. synth_ai/cli/commands/train/judge_schemas.py +199 -0
  209. synth_ai/cli/commands/train/judge_validation.py +304 -0
  210. synth_ai/cli/commands/train/validation.py +443 -0
  211. synth_ai/cli/demo.py +2 -162
  212. synth_ai/cli/deploy/__init__.py +28 -0
  213. synth_ai/cli/deploy/core.py +5 -0
  214. synth_ai/cli/deploy/errors.py +23 -0
  215. synth_ai/cli/deploy/validation.py +5 -0
  216. synth_ai/cli/eval/__init__.py +36 -0
  217. synth_ai/cli/eval/core.py +5 -0
  218. synth_ai/cli/eval/errors.py +31 -0
  219. synth_ai/cli/eval/validation.py +5 -0
  220. synth_ai/cli/filter/__init__.py +28 -0
  221. synth_ai/cli/filter/core.py +5 -0
  222. synth_ai/cli/filter/errors.py +23 -0
  223. synth_ai/cli/filter/validation.py +5 -0
  224. synth_ai/cli/legacy_root_backup.py +3 -1
  225. synth_ai/cli/lib/__init__.py +10 -0
  226. synth_ai/cli/lib/task_app_discovery.py +7 -0
  227. synth_ai/cli/lib/task_app_env.py +518 -0
  228. synth_ai/cli/modal_serve/__init__.py +12 -0
  229. synth_ai/cli/modal_serve/core.py +14 -0
  230. synth_ai/cli/modal_serve/errors.py +8 -0
  231. synth_ai/cli/modal_serve/validation.py +11 -0
  232. synth_ai/cli/recent.py +2 -1
  233. synth_ai/cli/serve/__init__.py +12 -0
  234. synth_ai/cli/serve/core.py +14 -0
  235. synth_ai/cli/serve/errors.py +8 -0
  236. synth_ai/cli/serve/validation.py +11 -0
  237. synth_ai/cli/setup.py +21 -0
  238. synth_ai/cli/status.py +7 -126
  239. synth_ai/cli/task_app_deploy.py +7 -0
  240. synth_ai/cli/task_app_list.py +25 -0
  241. synth_ai/cli/task_app_modal_serve.py +11 -0
  242. synth_ai/cli/task_app_serve.py +11 -0
  243. synth_ai/cli/task_apps.py +110 -1499
  244. synth_ai/cli/traces.py +1 -1
  245. synth_ai/cli/train/__init__.py +12 -0
  246. synth_ai/cli/train/core.py +21 -0
  247. synth_ai/cli/train/errors.py +8 -0
  248. synth_ai/cli/train/validation.py +24 -0
  249. synth_ai/cli/train.py +5 -0
  250. synth_ai/cli/turso.py +1 -1
  251. synth_ai/cli/watch.py +1 -1
  252. synth_ai/demos/__init__.py +10 -0
  253. synth_ai/demos/core/__init__.py +28 -1
  254. synth_ai/demos/crafter/__init__.py +1 -0
  255. synth_ai/demos/crafter/crafter_fft_4b.toml +55 -0
  256. synth_ai/demos/crafter/grpo_crafter_task_app.py +185 -0
  257. synth_ai/demos/crafter/rl_from_base_qwen4b.toml +74 -0
  258. synth_ai/demos/demo_registry.py +176 -0
  259. synth_ai/demos/demo_task_apps/crafter/grpo_crafter_task_app.py +1 -1
  260. synth_ai/demos/math/__init__.py +1 -0
  261. synth_ai/demos/math/_common.py +16 -0
  262. synth_ai/demos/math/app.py +38 -0
  263. synth_ai/demos/math/config.toml +76 -0
  264. synth_ai/demos/math/deploy_modal.py +54 -0
  265. synth_ai/demos/math/modal_task_app.py +702 -0
  266. synth_ai/demos/math/task_app_entry.py +51 -0
  267. synth_ai/environments/environment/core.py +7 -1
  268. synth_ai/environments/examples/bandit/engine.py +0 -1
  269. synth_ai/environments/examples/bandit/environment.py +0 -1
  270. synth_ai/environments/examples/red/engine.py +33 -12
  271. synth_ai/environments/examples/red/engine_helpers/reward_components.py +151 -179
  272. synth_ai/environments/examples/red/environment.py +26 -0
  273. synth_ai/environments/examples/red/trace_hooks_v3.py +168 -0
  274. synth_ai/environments/examples/wordle/environment.py +0 -1
  275. synth_ai/evals/base.py +16 -5
  276. synth_ai/evals/client.py +1 -1
  277. synth_ai/http.py +8 -22
  278. synth_ai/inference/client.py +1 -1
  279. synth_ai/judge_schemas.py +4 -5
  280. synth_ai/learning/client.py +1 -1
  281. synth_ai/learning/health.py +1 -1
  282. synth_ai/learning/jobs.py +1 -1
  283. synth_ai/learning/rl/client.py +4 -2
  284. synth_ai/learning/rl/env_keys.py +1 -1
  285. synth_ai/learning/rl/secrets.py +1 -1
  286. synth_ai/learning/sft/client.py +1 -1
  287. synth_ai/learning/sft/data.py +407 -4
  288. synth_ai/learning/validators.py +4 -1
  289. synth_ai/streaming/__init__.py +29 -0
  290. synth_ai/streaming/config.py +94 -0
  291. synth_ai/streaming/handlers.py +469 -0
  292. synth_ai/streaming/streamer.py +301 -0
  293. synth_ai/streaming/types.py +95 -0
  294. synth_ai/task/apps/__init__.py +4 -2
  295. synth_ai/task/config.py +6 -4
  296. synth_ai/task/rubrics/__init__.py +1 -2
  297. synth_ai/task/rubrics/loaders.py +14 -10
  298. synth_ai/task/rubrics.py +219 -0
  299. synth_ai/task/trace_correlation_helpers.py +24 -11
  300. synth_ai/task/tracing_utils.py +14 -3
  301. synth_ai/task/validators.py +0 -1
  302. synth_ai/tracing_v3/abstractions.py +3 -3
  303. synth_ai/tracing_v3/config.py +15 -13
  304. synth_ai/tracing_v3/constants.py +21 -0
  305. synth_ai/tracing_v3/db_config.py +3 -1
  306. synth_ai/tracing_v3/decorators.py +10 -7
  307. synth_ai/tracing_v3/llm_call_record_helpers.py +5 -5
  308. synth_ai/tracing_v3/migration_helper.py +1 -2
  309. synth_ai/tracing_v3/session_tracer.py +7 -7
  310. synth_ai/tracing_v3/storage/base.py +29 -29
  311. synth_ai/tracing_v3/storage/config.py +3 -3
  312. synth_ai/tracing_v3/turso/daemon.py +8 -9
  313. synth_ai/tracing_v3/turso/native_manager.py +80 -72
  314. synth_ai/tracing_v3/utils.py +2 -2
  315. synth_ai/utils/__init__.py +101 -0
  316. synth_ai/utils/base_url.py +94 -0
  317. synth_ai/utils/cli.py +131 -0
  318. synth_ai/utils/env.py +294 -0
  319. synth_ai/utils/http.py +172 -0
  320. synth_ai/utils/modal.py +308 -0
  321. synth_ai/utils/process.py +212 -0
  322. synth_ai/utils/prompts.py +39 -0
  323. synth_ai/utils/sqld.py +122 -0
  324. synth_ai/utils/task_app_discovery.py +882 -0
  325. synth_ai/utils/task_app_env.py +186 -0
  326. synth_ai/utils/task_app_state.py +318 -0
  327. synth_ai/utils/user_config.py +137 -0
  328. synth_ai/v0/config/__init__.py +1 -5
  329. synth_ai/v0/config/base_url.py +1 -7
  330. synth_ai/v0/tracing/config.py +1 -1
  331. synth_ai/v0/tracing/decorators.py +1 -1
  332. synth_ai/v0/tracing/upload.py +1 -1
  333. synth_ai/v0/tracing_v1/config.py +1 -1
  334. synth_ai/v0/tracing_v1/decorators.py +1 -1
  335. synth_ai/v0/tracing_v1/upload.py +1 -1
  336. {synth_ai-0.2.14.dist-info → synth_ai-0.2.17.dist-info}/METADATA +91 -32
  337. {synth_ai-0.2.14.dist-info → synth_ai-0.2.17.dist-info}/RECORD +341 -154
  338. synth_ai/cli/man.py +0 -106
  339. synth_ai/cli/tui.py +0 -57
  340. synth_ai/compound/cais.py +0 -0
  341. synth_ai/core/experiment.py +0 -13
  342. synth_ai/core/system.py +0 -15
  343. synth_ai/demo_registry.py +0 -295
  344. synth_ai/handshake.py +0 -109
  345. synth_ai/tui/__init__.py +0 -5
  346. synth_ai/tui/__main__.py +0 -13
  347. synth_ai/tui/cli/__init__.py +0 -1
  348. synth_ai/tui/cli/query_experiments.py +0 -164
  349. synth_ai/tui/cli/query_experiments_v3.py +0 -164
  350. synth_ai/tui/dashboard.py +0 -906
  351. {synth_ai-0.2.14.dist-info → synth_ai-0.2.17.dist-info}/WHEEL +0 -0
  352. {synth_ai-0.2.14.dist-info → synth_ai-0.2.17.dist-info}/entry_points.txt +0 -0
  353. {synth_ai-0.2.14.dist-info → synth_ai-0.2.17.dist-info}/licenses/LICENSE +0 -0
  354. {synth_ai-0.2.14.dist-info → synth_ai-0.2.17.dist-info}/top_level.txt +0 -0
@@ -1,11 +1,14 @@
1
1
  from __future__ import annotations
2
2
 
3
3
  import json
4
+ import logging
4
5
  from collections.abc import Iterable, Iterator, Sequence
5
6
  from dataclasses import dataclass, field
6
7
  from pathlib import Path
7
8
  from typing import Any
8
9
 
10
+ logger = logging.getLogger(__name__)
11
+
9
12
  SFTMessageContent = str | dict[str, Any] | list[Any] | None
10
13
 
11
14
 
@@ -37,6 +40,8 @@ class SFTMessage:
37
40
  tool_calls: list[SFTToolCall] = field(default_factory=list)
38
41
  tool_call_id: str | None = None
39
42
  name: str | None = None
43
+ reasoning: str | None = None # NEW: Explicit reasoning/thinking content
44
+ raw_content: str | None = None # NEW: Original unparsed content (before reasoning extraction)
40
45
  extra: dict[str, Any] = field(default_factory=dict)
41
46
 
42
47
 
@@ -86,9 +91,11 @@ def _coerce_tool_call(raw: Any, *, index: int) -> SFTToolCall:
86
91
  name: str | None = None
87
92
  arguments: Any = None
88
93
 
89
- if isinstance(raw.get("function"), dict):
90
- fn_payload = raw["function"]
91
- name = fn_payload.get("name") if isinstance(fn_payload.get("name"), str) else None
94
+ fn_obj = raw.get("function")
95
+ if isinstance(fn_obj, dict):
96
+ fn_payload = fn_obj
97
+ name_val = fn_payload.get("name")
98
+ name = name_val if isinstance(name_val, str) else None
92
99
  arguments = fn_payload.get("arguments")
93
100
  if name is None:
94
101
  maybe_name = raw.get("name")
@@ -143,11 +150,20 @@ def _coerce_message(raw: Any, *, index: int) -> SFTMessage:
143
150
  name = raw.get("name")
144
151
  if name is not None and not isinstance(name, str):
145
152
  raise SFTDataError(f"message {index} name must be a string if present")
153
+
154
+ # NEW: Extract reasoning and raw_content if present
155
+ reasoning = raw.get("reasoning")
156
+ if reasoning is not None and not isinstance(reasoning, str):
157
+ raise SFTDataError(f"message {index} reasoning must be a string if present")
158
+
159
+ raw_content = raw.get("raw_content")
160
+ if raw_content is not None and not isinstance(raw_content, str):
161
+ raise SFTDataError(f"message {index} raw_content must be a string if present")
146
162
 
147
163
  extra = {
148
164
  key: value
149
165
  for key, value in raw.items()
150
- if key not in {"role", "content", "tool_calls", "tool_call_id", "name"}
166
+ if key not in {"role", "content", "tool_calls", "tool_call_id", "name", "reasoning", "raw_content"}
151
167
  }
152
168
 
153
169
  return SFTMessage(
@@ -156,6 +172,8 @@ def _coerce_message(raw: Any, *, index: int) -> SFTMessage:
156
172
  tool_calls=tool_calls,
157
173
  tool_call_id=tool_call_id,
158
174
  name=name,
175
+ reasoning=reasoning,
176
+ raw_content=raw_content,
159
177
  extra=extra,
160
178
  )
161
179
 
@@ -280,6 +298,378 @@ def load_jsonl(path: Path, *, min_messages: int = 1) -> list[SFTExample]:
280
298
  return list(iter_sft_examples(fh, min_messages=min_messages))
281
299
 
282
300
 
301
+ # Reasoning/Thinking Utilities
302
+ # ============================================================================
303
+
304
+
305
+ def extract_reasoning(content: str, *, tag: str = "think") -> tuple[str | None, str]:
306
+ """Extract reasoning from content with <think> tags.
307
+
308
+ Args:
309
+ content: Raw content string
310
+ tag: Tag name to extract (default: "think")
311
+
312
+ Returns:
313
+ Tuple of (reasoning, clean_content)
314
+ - reasoning: Content inside tags, or None if no tags found
315
+ - clean_content: Content with tags removed
316
+
317
+ Examples:
318
+ >>> extract_reasoning("<think>Let me analyze...</think>The answer is 42")
319
+ ('Let me analyze...', 'The answer is 42')
320
+ >>> extract_reasoning("Just plain text")
321
+ (None, 'Just plain text')
322
+ """
323
+ import re
324
+
325
+ pattern = rf"<{tag}>(.*?)</{tag}>"
326
+ matches = re.findall(pattern, content, re.DOTALL)
327
+
328
+ if not matches:
329
+ return None, content
330
+
331
+ # Combine all reasoning blocks
332
+ reasoning = "\n\n".join(m.strip() for m in matches)
333
+
334
+ # Remove all reasoning blocks from content
335
+ clean_content = re.sub(pattern, "", content, flags=re.DOTALL).strip()
336
+
337
+ return reasoning, clean_content
338
+
339
+
340
+ def strip_reasoning(content: str, *, tag: str = "think") -> str:
341
+ """Remove reasoning tags from content.
342
+
343
+ Args:
344
+ content: Content with potential reasoning tags
345
+ tag: Tag name to strip (default: "think")
346
+
347
+ Returns:
348
+ Content with reasoning tags removed
349
+ """
350
+ _, clean = extract_reasoning(content, tag=tag)
351
+ return clean
352
+
353
+
354
+ def message_has_reasoning(message: SFTMessage) -> bool:
355
+ """Check if a message has explicit reasoning.
356
+
357
+ Args:
358
+ message: SFTMessage to check
359
+
360
+ Returns:
361
+ True if message has reasoning field or <think> tags in content
362
+ """
363
+ # Check explicit reasoning field
364
+ if message.reasoning:
365
+ return True
366
+
367
+ # Check for reasoning tags in content
368
+ if isinstance(message.content, str):
369
+ reasoning, _ = extract_reasoning(message.content)
370
+ return reasoning is not None
371
+
372
+ return False
373
+
374
+
375
+ def validate_message_content(
376
+ message: SFTMessage, *, require_content: bool = True
377
+ ) -> tuple[bool, str | None]:
378
+ """Validate that message has valid content combinations.
379
+
380
+ Rules:
381
+ - Must have at least one of: reasoning + tool_calls, reasoning + content,
382
+ content, raw_content, or tool_calls
383
+ - If raw_content present with reasoning + content, they should be consistent
384
+ - Cannot have neither reasoning, content, raw_content, nor tool_calls
385
+
386
+ Args:
387
+ message: SFTMessage to validate
388
+ require_content: If True, require some form of content (default: True)
389
+
390
+ Returns:
391
+ Tuple of (is_valid, error_message)
392
+ """
393
+ has_reasoning = bool(message.reasoning)
394
+ has_content = message.content is not None and message.content != ""
395
+ has_raw = bool(message.raw_content)
396
+ has_tools = len(message.tool_calls) > 0
397
+
398
+ # Check for completely empty message
399
+ if require_content and not (has_reasoning or has_content or has_raw or has_tools):
400
+ return False, "Message has no reasoning, content, raw_content, or tool_calls"
401
+
402
+ # Valid combinations:
403
+ # 1. reasoning + tool_calls (reasoning-based action)
404
+ if has_reasoning and has_tools:
405
+ return True, None
406
+
407
+ # 2. reasoning + content (reasoning then output)
408
+ if has_reasoning and has_content:
409
+ # If raw_content present, validate consistency
410
+ if has_raw and message.raw_content:
411
+ # Raw should contain both reasoning and content
412
+ reasoning_in_raw, content_in_raw = extract_reasoning(message.raw_content)
413
+ if message.reasoning and reasoning_in_raw != message.reasoning.strip():
414
+ logger.warning(
415
+ "raw_content reasoning doesn't match reasoning field"
416
+ )
417
+ # This is okay - just a warning, not an error
418
+ return True, None
419
+
420
+ # 3. content only (standard message)
421
+ if has_content and not has_reasoning:
422
+ return True, None
423
+
424
+ # 4. raw_content only (unparsed content)
425
+ if has_raw and not (has_reasoning and has_content):
426
+ return True, None
427
+
428
+ # 5. tool_calls only (action without reasoning/content - like OpenAI format)
429
+ if has_tools and not has_content:
430
+ return True, None
431
+
432
+ # 6. reasoning only (pure thinking turn)
433
+ if has_reasoning and not has_content and not has_tools:
434
+ return True, None
435
+
436
+ return True, None
437
+
438
+
439
+ # Vision/Multimodal Utilities
440
+ # ============================================================================
441
+
442
+
443
+ def has_image_content(content: SFTMessageContent) -> bool:
444
+ """Check if message content contains image data (OpenAI multimodal format).
445
+
446
+ Supports:
447
+ - List of content parts: [{"type": "text", ...}, {"type": "image_url", ...}]
448
+ - Single dict with type field: {"type": "image_url", "image_url": {...}}
449
+
450
+ Args:
451
+ content: Message content (can be str, list, dict, or None)
452
+
453
+ Returns:
454
+ True if content contains an image segment
455
+
456
+ Examples:
457
+ >>> has_image_content([{"type": "text", "text": "What's this?"},
458
+ ... {"type": "image_url", "image_url": {"url": "..."}}])
459
+ True
460
+ >>> has_image_content("Just text")
461
+ False
462
+ """
463
+ if isinstance(content, list):
464
+ return any(
465
+ isinstance(part, dict) and part.get("type") in {"image", "image_url"}
466
+ for part in content
467
+ )
468
+ elif isinstance(content, dict):
469
+ return content.get("type") in {"image", "image_url"}
470
+ return False
471
+
472
+
473
+ def message_has_image(message: SFTMessage) -> bool:
474
+ """Check if an SFTMessage contains image content.
475
+
476
+ Args:
477
+ message: SFTMessage to check
478
+
479
+ Returns:
480
+ True if the message contains image content
481
+ """
482
+ return has_image_content(message.content)
483
+
484
+
485
+ def example_has_image(example: SFTExample) -> bool:
486
+ """Check if an SFTExample contains any image content.
487
+
488
+ Args:
489
+ example: SFTExample to check
490
+
491
+ Returns:
492
+ True if any message in the example contains image content
493
+ """
494
+ return any(message_has_image(msg) for msg in example.messages)
495
+
496
+
497
+ def count_images_in_content(content: SFTMessageContent) -> int:
498
+ """Count the number of images in message content.
499
+
500
+ Args:
501
+ content: Message content to analyze
502
+
503
+ Returns:
504
+ Number of image segments found
505
+ """
506
+ if isinstance(content, list):
507
+ return sum(
508
+ 1 for part in content
509
+ if isinstance(part, dict) and part.get("type") in {"image", "image_url"}
510
+ )
511
+ elif isinstance(content, dict) and content.get("type") in {"image", "image_url"}:
512
+ return 1
513
+ return 0
514
+
515
+
516
+ def extract_image_urls(content: SFTMessageContent) -> list[str]:
517
+ """Extract all image URLs from message content.
518
+
519
+ Filters out invalid entries:
520
+ - Non-string URLs
521
+ - Empty strings
522
+ - Whitespace-only strings
523
+
524
+ Args:
525
+ content: Message content to extract from
526
+
527
+ Returns:
528
+ List of valid image URL strings (may be http(s):// URLs or data:image/... base64)
529
+ """
530
+ urls: list[str] = []
531
+
532
+ if isinstance(content, list):
533
+ for part in content:
534
+ if isinstance(part, dict) and part.get("type") in {"image", "image_url"}:
535
+ # Handle both formats:
536
+ # {"type": "image_url", "image_url": {"url": "..."}}
537
+ # {"type": "image", "image": "..."}
538
+ if "image_url" in part and isinstance(part["image_url"], dict):
539
+ url = part["image_url"].get("url")
540
+ if isinstance(url, str) and url.strip(): # Filter empty/whitespace
541
+ urls.append(url)
542
+ elif "image" in part and isinstance(part["image"], str):
543
+ if part["image"].strip(): # Filter empty/whitespace
544
+ urls.append(part["image"])
545
+ elif isinstance(content, dict) and content.get("type") in {"image", "image_url"}:
546
+ image_url_data = content.get("image_url")
547
+ if isinstance(image_url_data, dict):
548
+ url = image_url_data.get("url")
549
+ if isinstance(url, str) and url.strip(): # Filter empty/whitespace
550
+ urls.append(url)
551
+ else:
552
+ image_value = content.get("image")
553
+ if isinstance(image_value, str) and image_value.strip(): # Filter empty/whitespace
554
+ urls.append(image_value)
555
+
556
+ return urls
557
+
558
+
559
+ def validate_vision_example(
560
+ example: SFTExample, *, require_images: bool = True
561
+ ) -> tuple[bool, str | None]:
562
+ """Validate a vision SFT example.
563
+
564
+ Checks:
565
+ - If require_images is True, at least one message must contain an image
566
+ - All image URLs must be non-empty, non-whitespace strings
567
+ - Image entries must have valid URL data
568
+ - Messages must follow valid structure
569
+
570
+ Args:
571
+ example: SFTExample to validate
572
+ require_images: If True, fail if no images are present
573
+
574
+ Returns:
575
+ Tuple of (is_valid, error_message)
576
+ If valid, error_message is None
577
+ """
578
+ # Count actual valid URLs and detect any invalid entries
579
+ total_valid_urls = 0
580
+
581
+ # Validate image URLs in each message
582
+ for i, msg in enumerate(example.messages):
583
+ # Check if this message has image_url type entries
584
+ if not isinstance(msg.content, list | dict):
585
+ continue
586
+
587
+ # Count image_url type entries vs valid URLs
588
+ content_list = msg.content if isinstance(msg.content, list) else [msg.content]
589
+ image_type_count = sum(
590
+ 1 for item in content_list
591
+ if isinstance(item, dict) and item.get("type") in {"image", "image_url"}
592
+ )
593
+
594
+ if image_type_count > 0:
595
+ # Extract valid URLs (after filtering)
596
+ urls = extract_image_urls(msg.content)
597
+
598
+ # If we have image_url type entries but fewer valid URLs, some are invalid
599
+ if len(urls) < image_type_count:
600
+ return False, f"Message {i}: Has {image_type_count} image_url entries but only {len(urls)} valid URLs (some are empty, null, or missing)"
601
+
602
+ # Validate each URL (double-check, though extract_image_urls should have filtered)
603
+ for url in urls:
604
+ # extract_image_urls already filters for isinstance(url, str) and url.strip()
605
+ # but let's be defensive
606
+ if not isinstance(url, str):
607
+ return False, f"Message {i}: Image URL is not a string: {type(url)}"
608
+
609
+ if not url.strip():
610
+ return False, f"Message {i}: Invalid or empty image URL"
611
+
612
+ # Basic URL format check
613
+ if not url.startswith(("http://", "https://", "data:image/")):
614
+ logger.warning(
615
+ f"Message {i}: Image URL doesn't start with http://, https://, or data:image/ - "
616
+ f"this may cause issues during training. URL: {url[:100]}"
617
+ )
618
+
619
+ total_valid_urls += 1
620
+
621
+ # Final check: if images are required, ensure we found at least one valid URL
622
+ if require_images and total_valid_urls == 0:
623
+ return False, "No image content found in any message"
624
+
625
+ return True, None
626
+
627
+
628
+ def iter_vision_examples(
629
+ source: Iterable[str],
630
+ *,
631
+ min_messages: int = 1,
632
+ skip_empty: bool = True,
633
+ require_images: bool = True,
634
+ log_validation_errors: bool = False,
635
+ ) -> Iterator[SFTExample]:
636
+ """Iterate over vision SFT examples from JSONL source.
637
+
638
+ Similar to iter_sft_examples but with vision-specific validation.
639
+
640
+ Args:
641
+ source: Iterable of JSONL lines
642
+ min_messages: Minimum number of messages required
643
+ skip_empty: Skip empty lines
644
+ require_images: If True, skip examples without images
645
+ log_validation_errors: If True, log validation failures
646
+
647
+ Yields:
648
+ Valid vision SFTExample objects
649
+ """
650
+ for line in source:
651
+ if skip_empty and not line.strip():
652
+ continue
653
+
654
+ try:
655
+ example = parse_jsonl_line(line, min_messages=min_messages)
656
+
657
+ # Validate vision content if required
658
+ if require_images:
659
+ is_valid, error = validate_vision_example(example, require_images=True)
660
+ if not is_valid:
661
+ if log_validation_errors:
662
+ logger.warning(f"Skipping invalid vision example: {error}")
663
+ continue
664
+
665
+ yield example
666
+
667
+ except (json.JSONDecodeError, SFTDataError) as exc:
668
+ if log_validation_errors:
669
+ logger.warning(f"Failed to parse vision example: {exc}")
670
+ continue
671
+
672
+
283
673
  __all__ = [
284
674
  "SFTDataError",
285
675
  "SFTExample",
@@ -292,4 +682,17 @@ __all__ = [
292
682
  "load_jsonl",
293
683
  "parse_jsonl_line",
294
684
  "validate_jsonl_or_raise",
685
+ # Reasoning utilities
686
+ "extract_reasoning",
687
+ "strip_reasoning",
688
+ "message_has_reasoning",
689
+ "validate_message_content",
690
+ # Vision utilities
691
+ "has_image_content",
692
+ "message_has_image",
693
+ "example_has_image",
694
+ "count_images_in_content",
695
+ "extract_image_urls",
696
+ "validate_vision_example",
697
+ "iter_vision_examples",
295
698
  ]
@@ -37,7 +37,10 @@ def validate_training_jsonl(path: str | Path, *, sample_lines: int = 50) -> None
37
37
  def validate_task_app_url(url: str, *, name: str = "TASK_APP_BASE_URL") -> None:
38
38
  from synth_ai.task.validators import validate_task_app_url as _vt
39
39
 
40
- _vt(url, name=name)
40
+ try:
41
+ _vt(url)
42
+ except ValueError as exc:
43
+ raise ValueError(f"{name}: {exc}") from exc
41
44
 
42
45
 
43
46
  def validate_trainer_cfg_rl(trainer: dict[str, Any]) -> None:
@@ -0,0 +1,29 @@
1
+ from .config import StreamConfig
2
+ from .handlers import (
3
+ BufferedHandler,
4
+ CallbackHandler,
5
+ CLIHandler,
6
+ IntegrationTestHandler,
7
+ JSONHandler,
8
+ LossCurveHandler,
9
+ RichHandler,
10
+ StreamHandler,
11
+ )
12
+ from .streamer import JobStreamer, StreamEndpoints
13
+ from .types import StreamMessage, StreamType
14
+
15
+ __all__ = [
16
+ "BufferedHandler",
17
+ "CallbackHandler",
18
+ "CLIHandler",
19
+ "IntegrationTestHandler",
20
+ "JSONHandler",
21
+ "LossCurveHandler",
22
+ "JobStreamer",
23
+ "RichHandler",
24
+ "StreamEndpoints",
25
+ "StreamConfig",
26
+ "StreamHandler",
27
+ "StreamMessage",
28
+ "StreamType",
29
+ ]
@@ -0,0 +1,94 @@
1
+ from __future__ import annotations
2
+
3
+ from dataclasses import dataclass, field
4
+ from typing import Any
5
+
6
+ from .types import StreamType
7
+
8
+
9
+ @dataclass(slots=True)
10
+ class StreamConfig:
11
+ """Configuration describing which streams to consume and how to filter them."""
12
+
13
+ enabled_streams: set[StreamType] = field(default_factory=lambda: set(StreamType))
14
+ event_types: set[str] | None = None # Whitelist: only include these event types
15
+ event_types_exclude: set[str] | None = None # Blacklist: exclude these event types
16
+ event_levels: set[str] | None = None
17
+ metric_names: set[str] | None = None
18
+ metric_phases: set[str] | None = None
19
+ timeline_phases: set[str] | None = None
20
+ sample_rate: float = 1.0
21
+ max_events_per_poll: int | None = None
22
+ deduplicate: bool = True
23
+
24
+ @classmethod
25
+ def default(cls) -> StreamConfig:
26
+ """Return a configuration representing the default (all streams) view."""
27
+ return cls(
28
+ event_types_exclude={
29
+ # Filter out noisy events that just announce what metrics already show
30
+ "sft.progress", # Generic "Training progress" with no data
31
+ "sft.loss", # Generic "Loss update" with no data
32
+ "sft.upstream.status", # Very verbose status echo events
33
+ }
34
+ )
35
+
36
+ @classmethod
37
+ def minimal(cls) -> StreamConfig:
38
+ """Return a configuration streaming status updates only."""
39
+ return cls(enabled_streams={StreamType.STATUS})
40
+
41
+ @classmethod
42
+ def verbose(cls) -> StreamConfig:
43
+ """Return a configuration with all streams and events (no filters)."""
44
+ return cls()
45
+
46
+ @classmethod
47
+ def progress_only(cls) -> StreamConfig:
48
+ """Return a configuration tailored to show training progress."""
49
+ return cls(
50
+ enabled_streams={StreamType.STATUS, StreamType.EVENTS, StreamType.METRICS},
51
+ event_types={"sft.progress", "rl.train.step", "sft.validation.summary"},
52
+ metric_names={"train.loss", "eval.reward_mean"},
53
+ )
54
+
55
+ @classmethod
56
+ def errors_only(cls) -> StreamConfig:
57
+ """Return a configuration that focuses on heightened severity signals."""
58
+ return cls(
59
+ enabled_streams={StreamType.STATUS, StreamType.EVENTS},
60
+ event_levels={"error", "warning"},
61
+ )
62
+
63
+ def should_include_event(self, event: dict[str, Any]) -> bool:
64
+ """Determine whether an event message should be included."""
65
+ event_type = event.get("type")
66
+
67
+ # Apply blacklist first (takes precedence)
68
+ if self.event_types_exclude and event_type in self.event_types_exclude:
69
+ return False
70
+
71
+ # Then apply whitelist
72
+ if self.event_types and event_type not in self.event_types:
73
+ return False
74
+
75
+ if self.event_levels:
76
+ return event.get("level") in self.event_levels
77
+ return True
78
+
79
+ def should_include_metric(self, metric: dict[str, Any]) -> bool:
80
+ """Determine whether a metric point should be included."""
81
+ if self.metric_names and metric.get("name") not in self.metric_names:
82
+ return False
83
+ if self.metric_phases:
84
+ return metric.get("phase") in self.metric_phases
85
+ return True
86
+
87
+ def should_include_timeline(self, timeline_entry: dict[str, Any]) -> bool:
88
+ """Determine whether a timeline entry should be included."""
89
+ if self.timeline_phases:
90
+ return timeline_entry.get("phase") in self.timeline_phases
91
+ return True
92
+
93
+
94
+ __all__ = ["StreamConfig"]