synth-ai 0.2.9.dev11__py3-none-any.whl → 0.4.1__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 (909) hide show
  1. synth_ai/__init__.py +44 -45
  2. synth_ai/__main__.py +30 -3
  3. synth_ai/cli/__init__.py +104 -78
  4. synth_ai/cli/__main__.py +42 -0
  5. synth_ai/cli/_internal/__init__.py +5 -0
  6. synth_ai/cli/_internal/modal_wrapper.py +31 -0
  7. synth_ai/cli/_internal/storage.py +20 -0
  8. synth_ai/cli/_internal/typer_patch.py +47 -0
  9. synth_ai/cli/_internal/validate_task_app.py +29 -0
  10. synth_ai/cli/agents/__init__.py +17 -0
  11. synth_ai/cli/agents/claude.py +77 -0
  12. synth_ai/cli/agents/codex.py +265 -0
  13. synth_ai/cli/agents/opencode.py +253 -0
  14. synth_ai/cli/commands/__init__.py +18 -0
  15. synth_ai/cli/commands/artifacts/__init__.py +13 -0
  16. synth_ai/cli/commands/artifacts/client.py +119 -0
  17. synth_ai/cli/commands/artifacts/config.py +57 -0
  18. synth_ai/cli/commands/artifacts/core.py +24 -0
  19. synth_ai/cli/commands/artifacts/download.py +188 -0
  20. synth_ai/cli/commands/artifacts/export.py +186 -0
  21. synth_ai/cli/commands/artifacts/list.py +156 -0
  22. synth_ai/cli/commands/artifacts/parsing.py +250 -0
  23. synth_ai/cli/commands/artifacts/show.py +336 -0
  24. synth_ai/cli/commands/baseline/__init__.py +12 -0
  25. synth_ai/cli/commands/baseline/core.py +636 -0
  26. synth_ai/cli/commands/baseline/list.py +94 -0
  27. synth_ai/cli/commands/demo/__init__.py +3 -0
  28. synth_ai/cli/commands/demo/core.py +153 -0
  29. synth_ai/cli/commands/eval/__init__.py +19 -0
  30. synth_ai/cli/commands/eval/core.py +1113 -0
  31. synth_ai/cli/commands/eval/errors.py +81 -0
  32. synth_ai/cli/commands/eval/validation.py +133 -0
  33. synth_ai/cli/commands/filter/__init__.py +12 -0
  34. synth_ai/cli/commands/filter/core.py +424 -0
  35. synth_ai/cli/commands/filter/errors.py +55 -0
  36. synth_ai/cli/commands/filter/validation.py +77 -0
  37. synth_ai/cli/commands/help/__init__.py +185 -0
  38. synth_ai/cli/commands/help/core.py +72 -0
  39. synth_ai/cli/commands/scan/__init__.py +19 -0
  40. synth_ai/cli/commands/scan/cloudflare_scanner.py +403 -0
  41. synth_ai/cli/commands/scan/core.py +344 -0
  42. synth_ai/cli/commands/scan/health_checker.py +242 -0
  43. synth_ai/cli/commands/scan/local_scanner.py +278 -0
  44. synth_ai/cli/commands/scan/models.py +83 -0
  45. synth_ai/cli/commands/smoke/__init__.py +7 -0
  46. synth_ai/cli/commands/smoke/core.py +1438 -0
  47. synth_ai/cli/commands/status/__init__.py +66 -0
  48. synth_ai/cli/commands/status/client.py +192 -0
  49. synth_ai/cli/commands/status/config.py +92 -0
  50. synth_ai/cli/commands/status/errors.py +20 -0
  51. synth_ai/cli/commands/status/formatters.py +164 -0
  52. synth_ai/cli/commands/status/subcommands/__init__.py +9 -0
  53. synth_ai/cli/commands/status/subcommands/files.py +79 -0
  54. synth_ai/cli/commands/status/subcommands/jobs.py +334 -0
  55. synth_ai/cli/commands/status/subcommands/models.py +79 -0
  56. synth_ai/cli/commands/status/subcommands/pricing.py +23 -0
  57. synth_ai/cli/commands/status/subcommands/runs.py +81 -0
  58. synth_ai/cli/commands/status/subcommands/session.py +182 -0
  59. synth_ai/cli/commands/status/subcommands/summary.py +47 -0
  60. synth_ai/cli/commands/status/subcommands/usage.py +203 -0
  61. synth_ai/cli/commands/status/utils.py +114 -0
  62. synth_ai/cli/commands/train/__init__.py +53 -0
  63. synth_ai/cli/commands/train/core.py +22 -0
  64. synth_ai/cli/commands/train/errors.py +117 -0
  65. synth_ai/cli/commands/train/judge_schemas.py +201 -0
  66. synth_ai/cli/commands/train/judge_validation.py +305 -0
  67. synth_ai/cli/commands/train/prompt_learning_validation.py +633 -0
  68. synth_ai/cli/commands/train/validation.py +392 -0
  69. synth_ai/cli/demo_apps/__init__.py +10 -0
  70. synth_ai/cli/demo_apps/core/__init__.py +28 -0
  71. synth_ai/cli/demo_apps/core/cli.py +1735 -0
  72. synth_ai/cli/demo_apps/crafter/crafter_fft_4b.toml +55 -0
  73. synth_ai/cli/demo_apps/crafter/grpo_crafter_task_app.py +186 -0
  74. synth_ai/cli/demo_apps/crafter/rl_from_base_qwen4b.toml +74 -0
  75. synth_ai/cli/demo_apps/demo_registry.py +176 -0
  76. synth_ai/cli/demo_apps/demo_task_apps/core.py +440 -0
  77. synth_ai/cli/demo_apps/demo_task_apps/crafter/__init__.py +1 -0
  78. synth_ai/cli/demo_apps/demo_task_apps/crafter/grpo_crafter_task_app.py +185 -0
  79. synth_ai/cli/demo_apps/demo_task_apps/math/modal_task_app.py +742 -0
  80. synth_ai/cli/demo_apps/demo_task_apps/math/task_app_entry.py +39 -0
  81. synth_ai/cli/demo_apps/math/__init__.py +1 -0
  82. synth_ai/cli/demo_apps/math/_common.py +16 -0
  83. synth_ai/cli/demo_apps/math/app.py +38 -0
  84. synth_ai/cli/demo_apps/math/config.toml +76 -0
  85. synth_ai/cli/demo_apps/math/deploy_modal.py +54 -0
  86. synth_ai/cli/demo_apps/math/modal_task_app.py +702 -0
  87. synth_ai/cli/demo_apps/math/task_app_entry.py +53 -0
  88. synth_ai/cli/demo_apps/mipro/main.py +271 -0
  89. synth_ai/cli/demo_apps/mipro/task_app.py +933 -0
  90. synth_ai/cli/demo_apps/mipro/train_cfg.toml +92 -0
  91. synth_ai/cli/demos/__init__.py +12 -0
  92. synth_ai/cli/demos/demo.py +32 -0
  93. synth_ai/cli/demos/rl_demo.py +254 -0
  94. synth_ai/cli/deploy.py +216 -0
  95. synth_ai/cli/infra/__init__.py +14 -0
  96. synth_ai/cli/infra/balance.py +216 -0
  97. synth_ai/cli/infra/mcp.py +35 -0
  98. synth_ai/cli/infra/modal_app.py +36 -0
  99. synth_ai/cli/infra/setup.py +69 -0
  100. synth_ai/cli/infra/status.py +16 -0
  101. synth_ai/cli/infra/turso.py +77 -0
  102. synth_ai/cli/lib/__init__.py +10 -0
  103. synth_ai/cli/lib/agents.py +76 -0
  104. synth_ai/cli/lib/apps/modal_app.py +101 -0
  105. synth_ai/cli/lib/apps/task_app.py +643 -0
  106. synth_ai/cli/lib/bin.py +39 -0
  107. synth_ai/cli/lib/env.py +375 -0
  108. synth_ai/cli/lib/errors.py +85 -0
  109. synth_ai/cli/lib/modal.py +315 -0
  110. synth_ai/cli/lib/plotting.py +126 -0
  111. synth_ai/cli/lib/prompt_args.py +39 -0
  112. synth_ai/cli/lib/prompts.py +284 -0
  113. synth_ai/cli/lib/sqld.py +122 -0
  114. synth_ai/cli/lib/task_app_discovery.py +884 -0
  115. synth_ai/cli/lib/task_app_env.py +295 -0
  116. synth_ai/cli/lib/train_cfgs.py +300 -0
  117. synth_ai/cli/lib/tunnel_records.py +207 -0
  118. synth_ai/cli/local/__init__.py +14 -0
  119. synth_ai/cli/local/experiment_queue/__init__.py +72 -0
  120. synth_ai/cli/local/experiment_queue/api_schemas.py +221 -0
  121. synth_ai/cli/local/experiment_queue/celery_app.py +208 -0
  122. synth_ai/cli/local/experiment_queue/config.py +128 -0
  123. synth_ai/cli/local/experiment_queue/config_utils.py +272 -0
  124. synth_ai/cli/local/experiment_queue/database.py +175 -0
  125. synth_ai/cli/local/experiment_queue/dispatcher.py +119 -0
  126. synth_ai/cli/local/experiment_queue/models.py +231 -0
  127. synth_ai/cli/local/experiment_queue/progress_info.py +160 -0
  128. synth_ai/cli/local/experiment_queue/results.py +373 -0
  129. synth_ai/cli/local/experiment_queue/schemas.py +131 -0
  130. synth_ai/cli/local/experiment_queue/service.py +344 -0
  131. synth_ai/cli/local/experiment_queue/status.py +372 -0
  132. synth_ai/cli/local/experiment_queue/status_tracker.py +360 -0
  133. synth_ai/cli/local/experiment_queue/tasks.py +1984 -0
  134. synth_ai/cli/local/experiment_queue/trace_storage.py +65 -0
  135. synth_ai/cli/local/experiment_queue/validation.py +157 -0
  136. synth_ai/cli/local/session/__init__.py +92 -0
  137. synth_ai/cli/local/session/client.py +383 -0
  138. synth_ai/cli/local/session/constants.py +63 -0
  139. synth_ai/cli/local/session/exceptions.py +105 -0
  140. synth_ai/cli/local/session/manager.py +139 -0
  141. synth_ai/cli/local/session/models.py +89 -0
  142. synth_ai/cli/local/session/query.py +110 -0
  143. synth_ai/cli/root.py +30 -103
  144. synth_ai/cli/task_apps/__init__.py +26 -0
  145. synth_ai/cli/task_apps/commands.py +3153 -0
  146. synth_ai/cli/task_apps/deploy.py +7 -0
  147. synth_ai/cli/task_apps/list.py +26 -0
  148. synth_ai/cli/task_apps/main.py +36 -0
  149. synth_ai/cli/task_apps/modal_serve.py +11 -0
  150. synth_ai/cli/task_apps/serve.py +11 -0
  151. synth_ai/cli/training/__init__.py +8 -0
  152. synth_ai/cli/training/train.py +5 -0
  153. synth_ai/cli/training/train_cfg.py +34 -0
  154. synth_ai/cli/training/watch.py +506 -0
  155. synth_ai/cli/turso.py +34 -55
  156. synth_ai/cli/usage.py +159 -0
  157. synth_ai/cli/utils/__init__.py +8 -0
  158. synth_ai/cli/utils/experiments.py +235 -0
  159. synth_ai/cli/utils/queue.py +504 -0
  160. synth_ai/cli/utils/recent.py +133 -0
  161. synth_ai/cli/utils/traces.py +164 -0
  162. synth_ai/contracts/__init__.py +67 -0
  163. synth_ai/core/__init__.py +100 -0
  164. synth_ai/core/_utils/__init__.py +54 -0
  165. synth_ai/core/_utils/base_url.py +10 -0
  166. synth_ai/core/_utils/http.py +10 -0
  167. synth_ai/core/_utils/prompts.py +14 -0
  168. synth_ai/core/_utils/task_app_state.py +12 -0
  169. synth_ai/core/_utils/user_config.py +10 -0
  170. synth_ai/core/apps/common.py +116 -0
  171. synth_ai/core/auth.py +95 -0
  172. synth_ai/core/cfgs.py +240 -0
  173. synth_ai/core/config/__init__.py +16 -0
  174. synth_ai/core/config/base.py +168 -0
  175. synth_ai/core/config/resolver.py +89 -0
  176. synth_ai/core/env.py +220 -0
  177. synth_ai/core/errors.py +126 -0
  178. synth_ai/core/http.py +230 -0
  179. synth_ai/core/integrations/__init__.py +11 -0
  180. synth_ai/core/integrations/cloudflare.py +1710 -0
  181. synth_ai/core/integrations/mcp/__init__.py +6 -0
  182. synth_ai/core/integrations/mcp/__main__.py +8 -0
  183. synth_ai/core/integrations/mcp/claude.py +36 -0
  184. synth_ai/core/integrations/mcp/main.py +254 -0
  185. synth_ai/core/integrations/mcp/setup.py +100 -0
  186. synth_ai/core/integrations/modal.py +277 -0
  187. synth_ai/core/json.py +72 -0
  188. synth_ai/core/log_filter.py +99 -0
  189. synth_ai/core/logging.py +82 -0
  190. synth_ai/core/paths.py +107 -0
  191. synth_ai/core/pricing.py +109 -0
  192. synth_ai/core/process.py +233 -0
  193. synth_ai/core/ssl.py +25 -0
  194. synth_ai/core/storage/__init__.py +71 -0
  195. synth_ai/core/task_app_state.py +318 -0
  196. synth_ai/core/telemetry.py +282 -0
  197. synth_ai/core/tracing_v3/__init__.py +99 -0
  198. synth_ai/core/tracing_v3/config.py +229 -0
  199. synth_ai/core/tracing_v3/constants.py +21 -0
  200. synth_ai/core/tracing_v3/db_config.py +182 -0
  201. synth_ai/core/tracing_v3/decorators.py +401 -0
  202. synth_ai/core/tracing_v3/examples/basic_usage.py +194 -0
  203. synth_ai/core/tracing_v3/llm_call_record_helpers.py +437 -0
  204. synth_ai/core/tracing_v3/migration_helper.py +119 -0
  205. synth_ai/core/tracing_v3/replica_sync.py +262 -0
  206. synth_ai/core/tracing_v3/serialization.py +130 -0
  207. synth_ai/core/tracing_v3/session_tracer.py +542 -0
  208. synth_ai/core/tracing_v3/storage/base.py +211 -0
  209. synth_ai/core/tracing_v3/storage/config.py +109 -0
  210. synth_ai/core/tracing_v3/storage/factory.py +39 -0
  211. synth_ai/core/tracing_v3/storage/utils.py +206 -0
  212. synth_ai/core/tracing_v3/trace_utils.py +326 -0
  213. synth_ai/core/tracing_v3/turso/__init__.py +12 -0
  214. synth_ai/core/tracing_v3/turso/daemon.py +278 -0
  215. synth_ai/core/tracing_v3/turso/models.py +470 -0
  216. synth_ai/core/tracing_v3/turso/native_manager.py +1385 -0
  217. synth_ai/core/tracing_v3/utils.py +108 -0
  218. synth_ai/core/urls.py +18 -0
  219. synth_ai/core/user_config.py +137 -0
  220. synth_ai/core/uvicorn.py +222 -0
  221. synth_ai/data/__init__.py +110 -0
  222. synth_ai/data/enums.py +141 -0
  223. synth_ai/data/rewards.py +152 -0
  224. synth_ai/data/specs.py +36 -0
  225. synth_ai/data/traces.py +35 -0
  226. synth_ai/products/__init__.py +6 -0
  227. synth_ai/products/graph_evolve/__init__.py +46 -0
  228. synth_ai/products/graph_evolve/client.py +226 -0
  229. synth_ai/products/graph_evolve/config.py +591 -0
  230. synth_ai/products/graph_evolve/converters/__init__.py +42 -0
  231. synth_ai/products/graph_evolve/converters/openai_sft.py +484 -0
  232. synth_ai/products/graph_evolve/examples/hotpotqa/config.toml +109 -0
  233. synth_ai/products/graph_evolve/run.py +222 -0
  234. synth_ai/sdk/__init__.py +119 -0
  235. synth_ai/sdk/api/__init__.py +1 -0
  236. synth_ai/sdk/api/models/supported.py +514 -0
  237. synth_ai/sdk/api/research_agent/__init__.py +86 -0
  238. synth_ai/sdk/api/research_agent/cli.py +428 -0
  239. synth_ai/sdk/api/research_agent/config.py +357 -0
  240. synth_ai/sdk/api/research_agent/job.py +717 -0
  241. synth_ai/sdk/api/train/__init__.py +85 -0
  242. synth_ai/sdk/api/train/builders.py +895 -0
  243. synth_ai/sdk/api/train/cli.py +2188 -0
  244. synth_ai/sdk/api/train/config_finder.py +267 -0
  245. synth_ai/sdk/api/train/configs/__init__.py +65 -0
  246. synth_ai/sdk/api/train/configs/prompt_learning.py +1706 -0
  247. synth_ai/sdk/api/train/configs/rl.py +188 -0
  248. synth_ai/sdk/api/train/configs/sft.py +99 -0
  249. synth_ai/sdk/api/train/configs/shared.py +81 -0
  250. synth_ai/sdk/api/train/context_learning.py +312 -0
  251. synth_ai/sdk/api/train/env_resolver.py +418 -0
  252. synth_ai/sdk/api/train/graph_validators.py +216 -0
  253. synth_ai/sdk/api/train/graphgen.py +984 -0
  254. synth_ai/sdk/api/train/graphgen_models.py +823 -0
  255. synth_ai/sdk/api/train/graphgen_validators.py +109 -0
  256. synth_ai/sdk/api/train/pollers.py +124 -0
  257. synth_ai/sdk/api/train/progress/__init__.py +97 -0
  258. synth_ai/sdk/api/train/progress/dataclasses.py +569 -0
  259. synth_ai/sdk/api/train/progress/events.py +326 -0
  260. synth_ai/sdk/api/train/progress/results.py +428 -0
  261. synth_ai/sdk/api/train/progress/tracker.py +641 -0
  262. synth_ai/sdk/api/train/prompt_learning.py +470 -0
  263. synth_ai/sdk/api/train/rl.py +442 -0
  264. synth_ai/sdk/api/train/sft.py +396 -0
  265. synth_ai/sdk/api/train/summary.py +522 -0
  266. synth_ai/sdk/api/train/supported_algos.py +147 -0
  267. synth_ai/sdk/api/train/task_app.py +331 -0
  268. synth_ai/sdk/api/train/utils.py +279 -0
  269. synth_ai/sdk/api/train/validators.py +2424 -0
  270. synth_ai/sdk/baseline/__init__.py +25 -0
  271. synth_ai/sdk/baseline/config.py +209 -0
  272. synth_ai/sdk/baseline/discovery.py +216 -0
  273. synth_ai/sdk/baseline/execution.py +154 -0
  274. synth_ai/sdk/graphs/__init__.py +15 -0
  275. synth_ai/sdk/graphs/completions.py +570 -0
  276. synth_ai/sdk/inference/__init__.py +6 -0
  277. synth_ai/sdk/inference/client.py +128 -0
  278. synth_ai/sdk/jobs/__init__.py +16 -0
  279. synth_ai/sdk/jobs/client.py +371 -0
  280. synth_ai/sdk/judging/__init__.py +15 -0
  281. synth_ai/sdk/judging/base.py +24 -0
  282. synth_ai/sdk/judging/client.py +191 -0
  283. synth_ai/sdk/judging/schemas.py +222 -0
  284. synth_ai/sdk/judging/types.py +42 -0
  285. synth_ai/sdk/learning/__init__.py +69 -0
  286. synth_ai/sdk/learning/client.py +240 -0
  287. synth_ai/sdk/learning/ft_client.py +7 -0
  288. synth_ai/sdk/learning/health.py +49 -0
  289. synth_ai/sdk/learning/jobs.py +202 -0
  290. synth_ai/sdk/learning/prompt_extraction.py +334 -0
  291. synth_ai/sdk/learning/prompt_learning_client.py +455 -0
  292. synth_ai/sdk/learning/prompt_learning_types.py +185 -0
  293. synth_ai/sdk/learning/rl/client.py +268 -0
  294. synth_ai/sdk/learning/rl/contracts.py +27 -0
  295. synth_ai/sdk/learning/rl/env_keys.py +166 -0
  296. synth_ai/sdk/learning/rl/secrets.py +13 -0
  297. synth_ai/sdk/learning/sft/client.py +95 -0
  298. synth_ai/sdk/learning/sft/config.py +270 -0
  299. synth_ai/sdk/learning/sft/data.py +698 -0
  300. synth_ai/sdk/learning/validators.py +52 -0
  301. synth_ai/sdk/research_agent/__init__.py +34 -0
  302. synth_ai/sdk/research_agent/container_builder.py +328 -0
  303. synth_ai/sdk/research_agent/container_spec.py +198 -0
  304. synth_ai/sdk/research_agent/defaults.py +34 -0
  305. synth_ai/sdk/research_agent/results_collector.py +69 -0
  306. synth_ai/sdk/specs/__init__.py +46 -0
  307. synth_ai/sdk/specs/dataclasses.py +149 -0
  308. synth_ai/sdk/specs/loader.py +144 -0
  309. synth_ai/sdk/specs/serializer.py +199 -0
  310. synth_ai/sdk/specs/validation.py +250 -0
  311. synth_ai/sdk/streaming/__init__.py +35 -0
  312. synth_ai/sdk/streaming/config.py +94 -0
  313. synth_ai/sdk/streaming/handlers.py +1997 -0
  314. synth_ai/sdk/streaming/streamer.py +704 -0
  315. synth_ai/sdk/streaming/types.py +112 -0
  316. synth_ai/sdk/task/__init__.py +151 -0
  317. synth_ai/sdk/task/apps/__init__.py +133 -0
  318. synth_ai/sdk/task/config.py +261 -0
  319. synth_ai/sdk/task/contracts.py +298 -0
  320. synth_ai/sdk/task/datasets.py +108 -0
  321. synth_ai/sdk/task/in_process.py +1190 -0
  322. synth_ai/sdk/task/in_process_runner.py +309 -0
  323. synth_ai/sdk/task/inference_api.py +299 -0
  324. synth_ai/sdk/task/proxy.py +287 -0
  325. synth_ai/sdk/task/rubrics/__init__.py +55 -0
  326. synth_ai/sdk/task/rubrics/loaders.py +156 -0
  327. synth_ai/sdk/task/rubrics/models.py +57 -0
  328. synth_ai/sdk/task/rubrics/scoring.py +116 -0
  329. synth_ai/sdk/task/rubrics/strict.py +149 -0
  330. synth_ai/sdk/task/server.py +580 -0
  331. synth_ai/sdk/task/trace_correlation_helpers.py +506 -0
  332. synth_ai/sdk/task/tracing_utils.py +95 -0
  333. synth_ai/sdk/task/validators.py +456 -0
  334. synth_ai/sdk/tracing/__init__.py +39 -0
  335. synth_ai/sdk/training/__init__.py +102 -0
  336. synth_ai/sdk/usage/__init__.py +37 -0
  337. synth_ai/sdk/usage/client.py +171 -0
  338. synth_ai/sdk/usage/models.py +261 -0
  339. synth_ai/utils/__init__.py +213 -0
  340. synth_ai-0.4.1.dist-info/METADATA +195 -0
  341. synth_ai-0.4.1.dist-info/RECORD +379 -0
  342. synth_ai-0.4.1.dist-info/entry_points.txt +2 -0
  343. synth_ai-0.4.1.dist-info/top_level.txt +1 -0
  344. examples/__init__.py +0 -16
  345. examples/analyze_semantic_words.sh +0 -17
  346. examples/crafter_debug_render.py +0 -186
  347. examples/qwen_coder/README.md +0 -102
  348. examples/qwen_coder/_shared.py +0 -113
  349. examples/qwen_coder/configs/coder_lora_30b.toml +0 -61
  350. examples/qwen_coder/configs/coder_lora_4b.toml +0 -57
  351. examples/qwen_coder/configs/coder_lora_small.toml +0 -58
  352. examples/qwen_coder/generate_dataset.py +0 -98
  353. examples/qwen_coder/infer_ft_smoke.py +0 -64
  354. examples/qwen_coder/infer_prod_proxy.py +0 -73
  355. examples/qwen_coder/infer_via_synth.py +0 -87
  356. examples/qwen_coder/scripts/infer_coder.sh +0 -18
  357. examples/qwen_coder/scripts/train_coder_30b.sh +0 -21
  358. examples/qwen_coder/sft_full_17b.py +0 -103
  359. examples/qwen_coder/sft_lora_30b.py +0 -110
  360. examples/qwen_coder/subset_jsonl.py +0 -38
  361. examples/qwen_coder/validate_jsonl.py +0 -59
  362. examples/rl/README.md +0 -169
  363. examples/rl/configs/eval_base_qwen.toml +0 -15
  364. examples/rl/configs/eval_rl_qwen.toml +0 -11
  365. examples/rl/configs/rl_from_base_qwen.toml +0 -35
  366. examples/rl/configs/rl_from_base_qwen17.toml +0 -74
  367. examples/rl/configs/rl_from_ft_qwen.toml +0 -35
  368. examples/rl/download_dataset.py +0 -80
  369. examples/rl/run_eval.py +0 -436
  370. examples/rl/run_rl_and_save.py +0 -111
  371. examples/rl/task_app/README.md +0 -22
  372. examples/rl/task_app/math_single_step.py +0 -991
  373. examples/rl/task_app/math_task_app.py +0 -115
  374. examples/run_crafter_demo.sh +0 -10
  375. examples/sft/README.md +0 -139
  376. examples/sft/configs/crafter_fft_qwen0p6b.toml +0 -44
  377. examples/sft/configs/crafter_lora_qwen0p6b.toml +0 -45
  378. examples/sft/evaluate.py +0 -117
  379. examples/sft/export_dataset.py +0 -117
  380. examples/sft/generate_traces.py +0 -162
  381. examples/swe/__init__.py +0 -12
  382. examples/swe/task_app/README.md +0 -105
  383. examples/swe/task_app/__init__.py +0 -2
  384. examples/swe/task_app/grpo_swe_mini.py +0 -571
  385. examples/swe/task_app/grpo_swe_mini_task_app.py +0 -136
  386. examples/swe/task_app/hosted/README.md +0 -173
  387. examples/swe/task_app/hosted/__init__.py +0 -5
  388. examples/swe/task_app/hosted/branching.py +0 -143
  389. examples/swe/task_app/hosted/environment_routes.py +0 -1289
  390. examples/swe/task_app/hosted/envs/__init__.py +0 -1
  391. examples/swe/task_app/hosted/envs/crafter/__init__.py +0 -6
  392. examples/swe/task_app/hosted/envs/crafter/app.py +0 -1
  393. examples/swe/task_app/hosted/envs/crafter/environment.py +0 -522
  394. examples/swe/task_app/hosted/envs/crafter/policy.py +0 -478
  395. examples/swe/task_app/hosted/envs/crafter/react_agent.py +0 -108
  396. examples/swe/task_app/hosted/envs/crafter/shared.py +0 -305
  397. examples/swe/task_app/hosted/envs/crafter/tools.py +0 -47
  398. examples/swe/task_app/hosted/envs/mini_swe/__init__.py +0 -8
  399. examples/swe/task_app/hosted/envs/mini_swe/environment.py +0 -1164
  400. examples/swe/task_app/hosted/envs/mini_swe/policy.py +0 -355
  401. examples/swe/task_app/hosted/envs/mini_swe/shared.py +0 -83
  402. examples/swe/task_app/hosted/envs/mini_swe/tools.py +0 -96
  403. examples/swe/task_app/hosted/hosted_app.py +0 -204
  404. examples/swe/task_app/hosted/inference/__init__.py +0 -5
  405. examples/swe/task_app/hosted/inference/openai_client.py +0 -618
  406. examples/swe/task_app/hosted/main.py +0 -100
  407. examples/swe/task_app/hosted/policy_routes.py +0 -1079
  408. examples/swe/task_app/hosted/registry.py +0 -195
  409. examples/swe/task_app/hosted/rollout.py +0 -1869
  410. examples/swe/task_app/hosted/storage/__init__.py +0 -5
  411. examples/swe/task_app/hosted/storage/volume.py +0 -211
  412. examples/swe/task_app/hosted/test_agents.py +0 -161
  413. examples/swe/task_app/hosted/test_service.py +0 -137
  414. examples/swe/task_app/hosted/utils.py +0 -62
  415. examples/vlm/README.md +0 -68
  416. examples/vlm/configs/crafter_vlm_gpt4o.toml +0 -44
  417. examples/vlm/crafter_image_only_agent.py +0 -207
  418. examples/vlm/crafter_openai_vlm_agent.py +0 -277
  419. examples/vlm/filter_image_rows.py +0 -63
  420. examples/vlm/run_crafter_vlm_benchmark.py +0 -316
  421. examples/warming_up_to_rl/analyze_trace_db.py +0 -422
  422. examples/warming_up_to_rl/configs/crafter_fft.toml +0 -48
  423. examples/warming_up_to_rl/configs/crafter_fft_4b.toml +0 -54
  424. examples/warming_up_to_rl/configs/eval_fft_qwen4b.toml +0 -20
  425. examples/warming_up_to_rl/configs/eval_groq_qwen32b.toml +0 -13
  426. examples/warming_up_to_rl/configs/eval_modal_qwen4b.toml +0 -23
  427. examples/warming_up_to_rl/configs/rl_from_base_qwen4b.toml +0 -83
  428. examples/warming_up_to_rl/configs/rl_from_ft.toml +0 -56
  429. examples/warming_up_to_rl/export_trace_sft.py +0 -723
  430. examples/warming_up_to_rl/groq_test.py +0 -95
  431. examples/warming_up_to_rl/manage_secrets.py +0 -131
  432. examples/warming_up_to_rl/readme.md +0 -179
  433. examples/warming_up_to_rl/run_eval.py +0 -510
  434. examples/warming_up_to_rl/run_fft_and_save.py +0 -380
  435. examples/warming_up_to_rl/run_local_rollout.py +0 -237
  436. examples/warming_up_to_rl/run_local_rollout_modal.py +0 -246
  437. examples/warming_up_to_rl/run_local_rollout_parallel.py +0 -403
  438. examples/warming_up_to_rl/run_local_rollout_traced.py +0 -475
  439. examples/warming_up_to_rl/run_rl_and_save.py +0 -124
  440. examples/warming_up_to_rl/run_rollout_remote.py +0 -154
  441. examples/warming_up_to_rl/task_app/README.md +0 -42
  442. examples/warming_up_to_rl/task_app/grpo_crafter.py +0 -700
  443. examples/warming_up_to_rl/task_app/grpo_crafter_task_app.py +0 -146
  444. examples/warming_up_to_rl/task_app/synth_envs_hosted/README.md +0 -173
  445. examples/warming_up_to_rl/task_app/synth_envs_hosted/__init__.py +0 -5
  446. examples/warming_up_to_rl/task_app/synth_envs_hosted/branching.py +0 -143
  447. examples/warming_up_to_rl/task_app/synth_envs_hosted/environment_routes.py +0 -1226
  448. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/__init__.py +0 -1
  449. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/__init__.py +0 -6
  450. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/app.py +0 -1
  451. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/environment.py +0 -522
  452. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/policy.py +0 -478
  453. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/react_agent.py +0 -108
  454. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/shared.py +0 -305
  455. examples/warming_up_to_rl/task_app/synth_envs_hosted/envs/crafter/tools.py +0 -47
  456. examples/warming_up_to_rl/task_app/synth_envs_hosted/hosted_app.py +0 -204
  457. examples/warming_up_to_rl/task_app/synth_envs_hosted/inference/__init__.py +0 -5
  458. examples/warming_up_to_rl/task_app/synth_envs_hosted/inference/openai_client.py +0 -618
  459. examples/warming_up_to_rl/task_app/synth_envs_hosted/main.py +0 -100
  460. examples/warming_up_to_rl/task_app/synth_envs_hosted/policy_routes.py +0 -1083
  461. examples/warming_up_to_rl/task_app/synth_envs_hosted/registry.py +0 -195
  462. examples/warming_up_to_rl/task_app/synth_envs_hosted/rollout.py +0 -1869
  463. examples/warming_up_to_rl/task_app/synth_envs_hosted/storage/__init__.py +0 -5
  464. examples/warming_up_to_rl/task_app/synth_envs_hosted/storage/volume.py +0 -211
  465. examples/warming_up_to_rl/task_app/synth_envs_hosted/test_agents.py +0 -161
  466. examples/warming_up_to_rl/task_app/synth_envs_hosted/test_service.py +0 -137
  467. examples/warming_up_to_rl/task_app/synth_envs_hosted/utils.py +0 -62
  468. synth/__init__.py +0 -14
  469. synth_ai/api/models/supported.py +0 -376
  470. synth_ai/api/train/__init__.py +0 -5
  471. synth_ai/api/train/builders.py +0 -296
  472. synth_ai/api/train/cli.py +0 -606
  473. synth_ai/api/train/config_finder.py +0 -228
  474. synth_ai/api/train/env_resolver.py +0 -347
  475. synth_ai/api/train/pollers.py +0 -75
  476. synth_ai/api/train/supported_algos.py +0 -139
  477. synth_ai/api/train/task_app.py +0 -195
  478. synth_ai/api/train/utils.py +0 -217
  479. synth_ai/cli/_modal_wrapper.py +0 -28
  480. synth_ai/cli/_typer_patch.py +0 -49
  481. synth_ai/cli/balance.py +0 -203
  482. synth_ai/cli/calc.py +0 -69
  483. synth_ai/cli/demo.py +0 -159
  484. synth_ai/cli/legacy_root_backup.py +0 -470
  485. synth_ai/cli/man.py +0 -106
  486. synth_ai/cli/recent.py +0 -127
  487. synth_ai/cli/rl_demo.py +0 -274
  488. synth_ai/cli/status.py +0 -133
  489. synth_ai/cli/task_apps.py +0 -2782
  490. synth_ai/cli/traces.py +0 -163
  491. synth_ai/cli/watch.py +0 -505
  492. synth_ai/config/base_url.py +0 -107
  493. synth_ai/core/experiment.py +0 -13
  494. synth_ai/core/system.py +0 -15
  495. synth_ai/demo_registry.py +0 -295
  496. synth_ai/demos/core/__init__.py +0 -1
  497. synth_ai/demos/core/cli.py +0 -1756
  498. synth_ai/demos/demo_task_apps/core.py +0 -440
  499. synth_ai/demos/demo_task_apps/crafter/grpo_crafter_task_app.py +0 -172
  500. synth_ai/demos/demo_task_apps/math/deploy_task_app.sh +0 -22
  501. synth_ai/demos/demo_task_apps/math/modal_task_app.py +0 -739
  502. synth_ai/demos/demo_task_apps/math/task_app_entry.py +0 -37
  503. synth_ai/environments/__init__.py +0 -31
  504. synth_ai/environments/environment/__init__.py +0 -1
  505. synth_ai/environments/environment/artifacts/__init__.py +0 -1
  506. synth_ai/environments/environment/artifacts/base.py +0 -52
  507. synth_ai/environments/environment/core.py +0 -67
  508. synth_ai/environments/environment/db/__init__.py +0 -1
  509. synth_ai/environments/environment/db/sqlite.py +0 -45
  510. synth_ai/environments/environment/registry.py +0 -233
  511. synth_ai/environments/environment/resources/sqlite.py +0 -45
  512. synth_ai/environments/environment/results.py +0 -1
  513. synth_ai/environments/environment/rewards/__init__.py +0 -1
  514. synth_ai/environments/environment/rewards/core.py +0 -29
  515. synth_ai/environments/environment/shared_engine.py +0 -26
  516. synth_ai/environments/environment/tools/__init__.py +0 -200
  517. synth_ai/environments/examples/__init__.py +0 -1
  518. synth_ai/environments/examples/bandit/__init__.py +0 -33
  519. synth_ai/environments/examples/bandit/engine.py +0 -302
  520. synth_ai/environments/examples/bandit/environment.py +0 -194
  521. synth_ai/environments/examples/bandit/taskset.py +0 -200
  522. synth_ai/environments/examples/crafter_classic/__init__.py +0 -8
  523. synth_ai/environments/examples/crafter_classic/agent_demos/analyze_semantic_words_markdown.py +0 -250
  524. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_comprehensive_evaluation.py +0 -59
  525. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_evaluation_browser.py +0 -152
  526. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_evaluation_config.toml +0 -24
  527. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_evaluation_framework.py +0 -1194
  528. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/crafter_synth_config.toml +0 -56
  529. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/filter_config_modal.toml +0 -32
  530. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/kick_off_ft_modal.py +0 -384
  531. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/old/analyze_action_results.py +0 -53
  532. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/old/analyze_agent_actions.py +0 -178
  533. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/old/analyze_latest_run.py +0 -222
  534. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/old/analyze_lm_traces.py +0 -183
  535. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/old/analyze_no_rewards.py +0 -210
  536. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/old/analyze_trace_issue.py +0 -206
  537. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/old/check_db_schema.py +0 -49
  538. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/old/check_latest_results.py +0 -64
  539. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/old/debug_agent_responses.py +0 -88
  540. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_modal_ft/old/quick_trace_check.py +0 -77
  541. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/compare_experiments.py +0 -324
  542. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/kick_off_ft_oai.py +0 -362
  543. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/multi_model_config.toml +0 -49
  544. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/old/analyze_enhanced_hooks.py +0 -332
  545. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/old/analyze_hook_events.py +0 -97
  546. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/old/analyze_hook_results.py +0 -217
  547. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/old/check_hook_storage.py +0 -87
  548. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/old/check_seeds.py +0 -88
  549. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/old/compare_seed_performance.py +0 -195
  550. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/old/custom_eval_pipelines.py +0 -400
  551. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/old/plot_hook_frequency.py +0 -195
  552. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/old/seed_analysis_summary.py +0 -56
  553. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_openai_ft/run_rollouts_for_models_and_compare_v3.py +0 -858
  554. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_quick_evaluation.py +0 -52
  555. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_react_agent.py +0 -874
  556. synth_ai/environments/examples/crafter_classic/agent_demos/crafter_trace_evaluation.py +0 -1412
  557. synth_ai/environments/examples/crafter_classic/agent_demos/example_v3_usage.py +0 -216
  558. synth_ai/environments/examples/crafter_classic/agent_demos/old/compare_traces.py +0 -296
  559. synth_ai/environments/examples/crafter_classic/agent_demos/old/crafter_comprehensive_evaluation.py +0 -58
  560. synth_ai/environments/examples/crafter_classic/agent_demos/old/crafter_env_serialization.py +0 -464
  561. synth_ai/environments/examples/crafter_classic/agent_demos/old/crafter_evaluation_browser.py +0 -152
  562. synth_ai/environments/examples/crafter_classic/agent_demos/old/crafter_quick_evaluation.py +0 -51
  563. synth_ai/environments/examples/crafter_classic/agent_demos/old/crafter_trace_evaluation.py +0 -1412
  564. synth_ai/environments/examples/crafter_classic/agent_demos/old/debug_player_loss.py +0 -112
  565. synth_ai/environments/examples/crafter_classic/agent_demos/old/diagnose_service.py +0 -203
  566. synth_ai/environments/examples/crafter_classic/agent_demos/old/diagnose_slowness.py +0 -305
  567. synth_ai/environments/examples/crafter_classic/agent_demos/old/eval_by_difficulty.py +0 -126
  568. synth_ai/environments/examples/crafter_classic/agent_demos/old/eval_example.py +0 -94
  569. synth_ai/environments/examples/crafter_classic/agent_demos/old/explore_saved_states.py +0 -142
  570. synth_ai/environments/examples/crafter_classic/agent_demos/old/filter_traces_sft.py +0 -26
  571. synth_ai/environments/examples/crafter_classic/agent_demos/old/filter_traces_sft_OLD.py +0 -984
  572. synth_ai/environments/examples/crafter_classic/agent_demos/old/generate_ft_data_gemini.py +0 -724
  573. synth_ai/environments/examples/crafter_classic/agent_demos/old/generate_ft_data_modal.py +0 -386
  574. synth_ai/environments/examples/crafter_classic/agent_demos/old/generate_ft_metadata.py +0 -205
  575. synth_ai/environments/examples/crafter_classic/agent_demos/old/kick_off_ft_gemini.py +0 -150
  576. synth_ai/environments/examples/crafter_classic/agent_demos/old/kick_off_ft_modal.py +0 -283
  577. synth_ai/environments/examples/crafter_classic/agent_demos/old/prepare_vertex_ft.py +0 -280
  578. synth_ai/environments/examples/crafter_classic/agent_demos/old/profile_env_slowness.py +0 -456
  579. synth_ai/environments/examples/crafter_classic/agent_demos/old/replicate_issue.py +0 -166
  580. synth_ai/environments/examples/crafter_classic/agent_demos/old/run_and_eval.py +0 -102
  581. synth_ai/environments/examples/crafter_classic/agent_demos/old/run_comparison.py +0 -128
  582. synth_ai/environments/examples/crafter_classic/agent_demos/old/run_qwen_rollouts.py +0 -655
  583. synth_ai/environments/examples/crafter_classic/agent_demos/old/trace_eval_OLD.py +0 -202
  584. synth_ai/environments/examples/crafter_classic/agent_demos/old/validate_openai_format.py +0 -166
  585. synth_ai/environments/examples/crafter_classic/config_logging.py +0 -111
  586. synth_ai/environments/examples/crafter_classic/debug_translation.py +0 -0
  587. synth_ai/environments/examples/crafter_classic/engine.py +0 -579
  588. synth_ai/environments/examples/crafter_classic/engine_deterministic_patch.py +0 -64
  589. synth_ai/environments/examples/crafter_classic/engine_helpers/action_map.py +0 -6
  590. synth_ai/environments/examples/crafter_classic/engine_helpers/serialization.py +0 -75
  591. synth_ai/environments/examples/crafter_classic/engine_serialization_patch_v3.py +0 -267
  592. synth_ai/environments/examples/crafter_classic/environment.py +0 -479
  593. synth_ai/environments/examples/crafter_classic/taskset.py +0 -233
  594. synth_ai/environments/examples/crafter_classic/trace_hooks_v3.py +0 -228
  595. synth_ai/environments/examples/crafter_classic/world_config_patch_simple.py +0 -299
  596. synth_ai/environments/examples/crafter_custom/__init__.py +0 -4
  597. synth_ai/environments/examples/crafter_custom/agent_demos/__init__.py +0 -1
  598. synth_ai/environments/examples/crafter_custom/agent_demos/trace_eval.py +0 -202
  599. synth_ai/environments/examples/crafter_custom/crafter/__init__.py +0 -7
  600. synth_ai/environments/examples/crafter_custom/crafter/config.py +0 -182
  601. synth_ai/environments/examples/crafter_custom/crafter/constants.py +0 -8
  602. synth_ai/environments/examples/crafter_custom/crafter/engine.py +0 -269
  603. synth_ai/environments/examples/crafter_custom/crafter/env.py +0 -262
  604. synth_ai/environments/examples/crafter_custom/crafter/objects.py +0 -417
  605. synth_ai/environments/examples/crafter_custom/crafter/recorder.py +0 -187
  606. synth_ai/environments/examples/crafter_custom/crafter/worldgen.py +0 -118
  607. synth_ai/environments/examples/crafter_custom/dataset_builder.py +0 -373
  608. synth_ai/environments/examples/crafter_custom/environment.py +0 -312
  609. synth_ai/environments/examples/crafter_custom/old/analyze_diamond_issue.py +0 -159
  610. synth_ai/environments/examples/crafter_custom/old/analyze_diamond_spawning.py +0 -158
  611. synth_ai/environments/examples/crafter_custom/old/compare_worlds.py +0 -71
  612. synth_ai/environments/examples/crafter_custom/old/dataset_stats.py +0 -105
  613. synth_ai/environments/examples/crafter_custom/old/diamond_spawning_summary.py +0 -119
  614. synth_ai/environments/examples/crafter_custom/old/example_dataset_usage.py +0 -52
  615. synth_ai/environments/examples/crafter_custom/run_dataset.py +0 -305
  616. synth_ai/environments/examples/enron/art_helpers/email_search_tools.py +0 -156
  617. synth_ai/environments/examples/enron/art_helpers/local_email_db.py +0 -281
  618. synth_ai/environments/examples/enron/art_helpers/types_enron.py +0 -25
  619. synth_ai/environments/examples/enron/engine.py +0 -295
  620. synth_ai/environments/examples/enron/environment.py +0 -166
  621. synth_ai/environments/examples/enron/taskset.py +0 -112
  622. synth_ai/environments/examples/enron/units/keyword_stats.py +0 -112
  623. synth_ai/environments/examples/minigrid/__init__.py +0 -48
  624. synth_ai/environments/examples/minigrid/agent_demos/minigrid_evaluation_framework.py +0 -1188
  625. synth_ai/environments/examples/minigrid/agent_demos/minigrid_quick_evaluation.py +0 -48
  626. synth_ai/environments/examples/minigrid/agent_demos/minigrid_react_agent.py +0 -562
  627. synth_ai/environments/examples/minigrid/agent_demos/minigrid_trace_evaluation.py +0 -221
  628. synth_ai/environments/examples/minigrid/engine.py +0 -589
  629. synth_ai/environments/examples/minigrid/environment.py +0 -274
  630. synth_ai/environments/examples/minigrid/environment_mapping.py +0 -242
  631. synth_ai/environments/examples/minigrid/puzzle_loader.py +0 -417
  632. synth_ai/environments/examples/minigrid/taskset.py +0 -583
  633. synth_ai/environments/examples/nethack/__init__.py +0 -7
  634. synth_ai/environments/examples/nethack/achievements.py +0 -337
  635. synth_ai/environments/examples/nethack/agent_demos/nethack_evaluation_framework.py +0 -981
  636. synth_ai/environments/examples/nethack/agent_demos/nethack_quick_evaluation.py +0 -74
  637. synth_ai/environments/examples/nethack/agent_demos/nethack_react_agent.py +0 -831
  638. synth_ai/environments/examples/nethack/engine.py +0 -739
  639. synth_ai/environments/examples/nethack/environment.py +0 -256
  640. synth_ai/environments/examples/nethack/helpers/__init__.py +0 -41
  641. synth_ai/environments/examples/nethack/helpers/action_mapping.py +0 -301
  642. synth_ai/environments/examples/nethack/helpers/nle_wrapper.py +0 -402
  643. synth_ai/environments/examples/nethack/helpers/observation_utils.py +0 -433
  644. synth_ai/environments/examples/nethack/helpers/recording_wrapper.py +0 -200
  645. synth_ai/environments/examples/nethack/helpers/trajectory_recorder.py +0 -269
  646. synth_ai/environments/examples/nethack/helpers/visualization/replay_viewer.py +0 -308
  647. synth_ai/environments/examples/nethack/helpers/visualization/visualizer.py +0 -431
  648. synth_ai/environments/examples/nethack/taskset.py +0 -323
  649. synth_ai/environments/examples/red/__init__.py +0 -7
  650. synth_ai/environments/examples/red/agent_demos/__init__.py +0 -1
  651. synth_ai/environments/examples/red/config_logging.py +0 -110
  652. synth_ai/environments/examples/red/engine.py +0 -694
  653. synth_ai/environments/examples/red/engine_helpers/__init__.py +0 -1
  654. synth_ai/environments/examples/red/engine_helpers/memory_map.py +0 -28
  655. synth_ai/environments/examples/red/engine_helpers/reward_components.py +0 -276
  656. synth_ai/environments/examples/red/engine_helpers/reward_library/__init__.py +0 -142
  657. synth_ai/environments/examples/red/engine_helpers/reward_library/adaptive_rewards.py +0 -57
  658. synth_ai/environments/examples/red/engine_helpers/reward_library/battle_rewards.py +0 -284
  659. synth_ai/environments/examples/red/engine_helpers/reward_library/composite_rewards.py +0 -150
  660. synth_ai/environments/examples/red/engine_helpers/reward_library/economy_rewards.py +0 -138
  661. synth_ai/environments/examples/red/engine_helpers/reward_library/efficiency_rewards.py +0 -57
  662. synth_ai/environments/examples/red/engine_helpers/reward_library/exploration_rewards.py +0 -331
  663. synth_ai/environments/examples/red/engine_helpers/reward_library/novelty_rewards.py +0 -121
  664. synth_ai/environments/examples/red/engine_helpers/reward_library/pallet_town_rewards.py +0 -559
  665. synth_ai/environments/examples/red/engine_helpers/reward_library/pokemon_rewards.py +0 -313
  666. synth_ai/environments/examples/red/engine_helpers/reward_library/social_rewards.py +0 -148
  667. synth_ai/environments/examples/red/engine_helpers/reward_library/story_rewards.py +0 -247
  668. synth_ai/environments/examples/red/engine_helpers/screen_analysis.py +0 -368
  669. synth_ai/environments/examples/red/engine_helpers/state_extraction.py +0 -140
  670. synth_ai/environments/examples/red/environment.py +0 -238
  671. synth_ai/environments/examples/red/taskset.py +0 -79
  672. synth_ai/environments/examples/red/units/__init__.py +0 -1
  673. synth_ai/environments/examples/sokoban/__init__.py +0 -1
  674. synth_ai/environments/examples/sokoban/agent_demos/sokoban_full_eval.py +0 -899
  675. synth_ai/environments/examples/sokoban/engine.py +0 -678
  676. synth_ai/environments/examples/sokoban/engine_helpers/__init__.py +0 -1
  677. synth_ai/environments/examples/sokoban/engine_helpers/room_utils.py +0 -657
  678. synth_ai/environments/examples/sokoban/engine_helpers/vendored/__init__.py +0 -18
  679. synth_ai/environments/examples/sokoban/engine_helpers/vendored/envs/__init__.py +0 -3
  680. synth_ai/environments/examples/sokoban/engine_helpers/vendored/envs/boxoban_env.py +0 -131
  681. synth_ai/environments/examples/sokoban/engine_helpers/vendored/envs/render_utils.py +0 -370
  682. synth_ai/environments/examples/sokoban/engine_helpers/vendored/envs/room_utils.py +0 -332
  683. synth_ai/environments/examples/sokoban/engine_helpers/vendored/envs/sokoban_env.py +0 -306
  684. synth_ai/environments/examples/sokoban/engine_helpers/vendored/envs/sokoban_env_fixed_targets.py +0 -67
  685. synth_ai/environments/examples/sokoban/engine_helpers/vendored/envs/sokoban_env_pull.py +0 -115
  686. synth_ai/environments/examples/sokoban/engine_helpers/vendored/envs/sokoban_env_two_player.py +0 -123
  687. synth_ai/environments/examples/sokoban/engine_helpers/vendored/envs/sokoban_env_variations.py +0 -394
  688. synth_ai/environments/examples/sokoban/environment.py +0 -229
  689. synth_ai/environments/examples/sokoban/generate_verified_puzzles.py +0 -440
  690. synth_ai/environments/examples/sokoban/puzzle_loader.py +0 -312
  691. synth_ai/environments/examples/sokoban/taskset.py +0 -428
  692. synth_ai/environments/examples/tictactoe/__init__.py +0 -1
  693. synth_ai/environments/examples/tictactoe/engine.py +0 -368
  694. synth_ai/environments/examples/tictactoe/environment.py +0 -240
  695. synth_ai/environments/examples/tictactoe/taskset.py +0 -215
  696. synth_ai/environments/examples/verilog/__init__.py +0 -10
  697. synth_ai/environments/examples/verilog/engine.py +0 -329
  698. synth_ai/environments/examples/verilog/environment.py +0 -350
  699. synth_ai/environments/examples/verilog/taskset.py +0 -420
  700. synth_ai/environments/examples/wordle/__init__.py +0 -29
  701. synth_ai/environments/examples/wordle/engine.py +0 -398
  702. synth_ai/environments/examples/wordle/environment.py +0 -159
  703. synth_ai/environments/examples/wordle/helpers/generate_instances_wordfreq.py +0 -75
  704. synth_ai/environments/examples/wordle/taskset.py +0 -230
  705. synth_ai/environments/reproducibility/core.py +0 -42
  706. synth_ai/environments/reproducibility/helpers.py +0 -0
  707. synth_ai/environments/reproducibility/tree.py +0 -363
  708. synth_ai/environments/service/app.py +0 -97
  709. synth_ai/environments/service/core_routes.py +0 -1021
  710. synth_ai/environments/service/external_registry.py +0 -56
  711. synth_ai/environments/service/registry.py +0 -9
  712. synth_ai/environments/stateful/__init__.py +0 -1
  713. synth_ai/environments/stateful/core.py +0 -163
  714. synth_ai/environments/stateful/engine.py +0 -21
  715. synth_ai/environments/stateful/state.py +0 -7
  716. synth_ai/environments/tasks/api.py +0 -19
  717. synth_ai/environments/tasks/core.py +0 -81
  718. synth_ai/environments/tasks/filters.py +0 -40
  719. synth_ai/environments/tasks/utils.py +0 -90
  720. synth_ai/environments/v0_observability/history.py +0 -3
  721. synth_ai/environments/v0_observability/log.py +0 -2
  722. synth_ai/evals/base.py +0 -13
  723. synth_ai/handshake.py +0 -109
  724. synth_ai/http.py +0 -26
  725. synth_ai/http_client.py +0 -136
  726. synth_ai/inference/__init__.py +0 -5
  727. synth_ai/inference/client.py +0 -34
  728. synth_ai/jobs/client.py +0 -271
  729. synth_ai/learning/__init__.py +0 -59
  730. synth_ai/learning/client.py +0 -241
  731. synth_ai/learning/ft_client.py +0 -7
  732. synth_ai/learning/health.py +0 -49
  733. synth_ai/learning/jobs.py +0 -201
  734. synth_ai/learning/rl/client.py +0 -267
  735. synth_ai/learning/rl/contracts.py +0 -27
  736. synth_ai/learning/rl/env_keys.py +0 -166
  737. synth_ai/learning/rl/secrets.py +0 -13
  738. synth_ai/learning/sft/client.py +0 -68
  739. synth_ai/learning/sft/config.py +0 -270
  740. synth_ai/learning/sft/data.py +0 -295
  741. synth_ai/learning/validators.py +0 -49
  742. synth_ai/lm/__init__.py +0 -25
  743. synth_ai/main.py +0 -6
  744. synth_ai/task/__init__.py +0 -102
  745. synth_ai/task/apps/__init__.py +0 -128
  746. synth_ai/task/contracts.py +0 -137
  747. synth_ai/task/datasets.py +0 -108
  748. synth_ai/task/proxy.py +0 -259
  749. synth_ai/task/server.py +0 -424
  750. synth_ai/task/tracing_utils.py +0 -84
  751. synth_ai/task/validators.py +0 -11
  752. synth_ai/tracing_v3/__init__.py +0 -97
  753. synth_ai/tracing_v3/config.py +0 -84
  754. synth_ai/tracing_v3/db_config.py +0 -194
  755. synth_ai/tracing_v3/decorators.py +0 -369
  756. synth_ai/tracing_v3/examples/basic_usage.py +0 -189
  757. synth_ai/tracing_v3/llm_call_record_helpers.py +0 -337
  758. synth_ai/tracing_v3/migration_helper.py +0 -120
  759. synth_ai/tracing_v3/replica_sync.py +0 -258
  760. synth_ai/tracing_v3/session_tracer.py +0 -530
  761. synth_ai/tracing_v3/storage/base.py +0 -210
  762. synth_ai/tracing_v3/storage/config.py +0 -75
  763. synth_ai/tracing_v3/storage/factory.py +0 -39
  764. synth_ai/tracing_v3/storage/utils.py +0 -204
  765. synth_ai/tracing_v3/turso/daemon.py +0 -149
  766. synth_ai/tracing_v3/turso/models.py +0 -469
  767. synth_ai/tracing_v3/turso/native_manager.py +0 -1173
  768. synth_ai/tracing_v3/utils.py +0 -108
  769. synth_ai/v0/api/__init__.py +0 -8
  770. synth_ai/v0/api/models/__init__.py +0 -8
  771. synth_ai/v0/api/models/supported.py +0 -8
  772. synth_ai/v0/config/__init__.py +0 -15
  773. synth_ai/v0/config/base_url.py +0 -12
  774. synth_ai/v0/lm/__init__.py +0 -51
  775. synth_ai/v0/lm/caching/constants.py +0 -6
  776. synth_ai/v0/lm/caching/dbs.py +0 -0
  777. synth_ai/v0/lm/caching/ephemeral.py +0 -100
  778. synth_ai/v0/lm/caching/handler.py +0 -137
  779. synth_ai/v0/lm/caching/initialize.py +0 -11
  780. synth_ai/v0/lm/caching/persistent.py +0 -114
  781. synth_ai/v0/lm/config.py +0 -115
  782. synth_ai/v0/lm/constants.py +0 -32
  783. synth_ai/v0/lm/core/__init__.py +0 -8
  784. synth_ai/v0/lm/core/all.py +0 -73
  785. synth_ai/v0/lm/core/exceptions.py +0 -5
  786. synth_ai/v0/lm/core/main.py +0 -331
  787. synth_ai/v0/lm/core/main_v3.py +0 -594
  788. synth_ai/v0/lm/core/synth_models.py +0 -35
  789. synth_ai/v0/lm/core/vendor_clients.py +0 -190
  790. synth_ai/v0/lm/cost/__init__.py +0 -0
  791. synth_ai/v0/lm/cost/monitor.py +0 -1
  792. synth_ai/v0/lm/cost/statefulness.py +0 -1
  793. synth_ai/v0/lm/injection.py +0 -80
  794. synth_ai/v0/lm/overrides.py +0 -206
  795. synth_ai/v0/lm/provider_support/__init__.py +0 -8
  796. synth_ai/v0/lm/provider_support/anthropic.py +0 -972
  797. synth_ai/v0/lm/provider_support/openai.py +0 -1139
  798. synth_ai/v0/lm/provider_support/suppress_logging.py +0 -31
  799. synth_ai/v0/lm/structured_outputs/__init__.py +0 -0
  800. synth_ai/v0/lm/structured_outputs/handler.py +0 -440
  801. synth_ai/v0/lm/structured_outputs/inject.py +0 -297
  802. synth_ai/v0/lm/structured_outputs/rehabilitate.py +0 -185
  803. synth_ai/v0/lm/tools/__init__.py +0 -3
  804. synth_ai/v0/lm/tools/base.py +0 -172
  805. synth_ai/v0/lm/unified_interface.py +0 -202
  806. synth_ai/v0/lm/vendors/__init__.py +0 -0
  807. synth_ai/v0/lm/vendors/base.py +0 -81
  808. synth_ai/v0/lm/vendors/core/__init__.py +0 -0
  809. synth_ai/v0/lm/vendors/core/anthropic_api.py +0 -387
  810. synth_ai/v0/lm/vendors/core/gemini_api.py +0 -292
  811. synth_ai/v0/lm/vendors/core/mistral_api.py +0 -322
  812. synth_ai/v0/lm/vendors/core/openai_api.py +0 -227
  813. synth_ai/v0/lm/vendors/core/synth_dev_api.py +0 -0
  814. synth_ai/v0/lm/vendors/local/__init__.py +0 -0
  815. synth_ai/v0/lm/vendors/local/ollama.py +0 -0
  816. synth_ai/v0/lm/vendors/openai_standard.py +0 -782
  817. synth_ai/v0/lm/vendors/openai_standard_responses.py +0 -259
  818. synth_ai/v0/lm/vendors/retries.py +0 -22
  819. synth_ai/v0/lm/vendors/supported/__init__.py +0 -0
  820. synth_ai/v0/lm/vendors/supported/custom_endpoint.py +0 -415
  821. synth_ai/v0/lm/vendors/supported/deepseek.py +0 -69
  822. synth_ai/v0/lm/vendors/supported/grok.py +0 -75
  823. synth_ai/v0/lm/vendors/supported/groq.py +0 -16
  824. synth_ai/v0/lm/vendors/supported/ollama.py +0 -15
  825. synth_ai/v0/lm/vendors/supported/openrouter.py +0 -74
  826. synth_ai/v0/lm/vendors/supported/together.py +0 -11
  827. synth_ai/v0/lm/vendors/synth_client.py +0 -835
  828. synth_ai/v0/lm/warmup.py +0 -186
  829. synth_ai/v0/tracing/__init__.py +0 -0
  830. synth_ai/v0/tracing/abstractions.py +0 -224
  831. synth_ai/v0/tracing/base_client.py +0 -91
  832. synth_ai/v0/tracing/client_manager.py +0 -131
  833. synth_ai/v0/tracing/config.py +0 -142
  834. synth_ai/v0/tracing/context.py +0 -146
  835. synth_ai/v0/tracing/decorators.py +0 -682
  836. synth_ai/v0/tracing/events/__init__.py +0 -0
  837. synth_ai/v0/tracing/events/manage.py +0 -147
  838. synth_ai/v0/tracing/events/scope.py +0 -86
  839. synth_ai/v0/tracing/events/store.py +0 -228
  840. synth_ai/v0/tracing/immediate_client.py +0 -151
  841. synth_ai/v0/tracing/local.py +0 -18
  842. synth_ai/v0/tracing/log_client_base.py +0 -73
  843. synth_ai/v0/tracing/retry_queue.py +0 -186
  844. synth_ai/v0/tracing/trackers.py +0 -515
  845. synth_ai/v0/tracing/upload.py +0 -409
  846. synth_ai/v0/tracing/utils.py +0 -9
  847. synth_ai/v0/tracing_v1/__init__.py +0 -16
  848. synth_ai/v0/tracing_v1/abstractions.py +0 -224
  849. synth_ai/v0/tracing_v1/base_client.py +0 -91
  850. synth_ai/v0/tracing_v1/client_manager.py +0 -131
  851. synth_ai/v0/tracing_v1/config.py +0 -142
  852. synth_ai/v0/tracing_v1/context.py +0 -146
  853. synth_ai/v0/tracing_v1/decorators.py +0 -703
  854. synth_ai/v0/tracing_v1/events/__init__.py +0 -0
  855. synth_ai/v0/tracing_v1/events/manage.py +0 -147
  856. synth_ai/v0/tracing_v1/events/scope.py +0 -86
  857. synth_ai/v0/tracing_v1/events/store.py +0 -228
  858. synth_ai/v0/tracing_v1/immediate_client.py +0 -151
  859. synth_ai/v0/tracing_v1/local.py +0 -18
  860. synth_ai/v0/tracing_v1/log_client_base.py +0 -73
  861. synth_ai/v0/tracing_v1/retry_queue.py +0 -186
  862. synth_ai/v0/tracing_v1/trackers.py +0 -515
  863. synth_ai/v0/tracing_v1/upload.py +0 -527
  864. synth_ai/v0/tracing_v1/utils.py +0 -9
  865. synth_ai/v0/tracing_v3/__init__.py +0 -10
  866. synth_ai/v0/tracing_v3/abstractions.py +0 -3
  867. synth_ai/v0/tracing_v3/decorators.py +0 -3
  868. synth_ai/v0/tracing_v3/llm_call_record_helpers.py +0 -3
  869. synth_ai/v0/tracing_v3/session_tracer.py +0 -3
  870. synth_ai-0.2.9.dev11.dist-info/METADATA +0 -191
  871. synth_ai-0.2.9.dev11.dist-info/RECORD +0 -571
  872. synth_ai-0.2.9.dev11.dist-info/entry_points.txt +0 -3
  873. synth_ai-0.2.9.dev11.dist-info/top_level.txt +0 -3
  874. /synth_ai/{demos/demo_task_apps → cli/demo_apps}/crafter/__init__.py +0 -0
  875. /synth_ai/{demos → cli/demo_apps}/demo_task_apps/__init__.py +0 -0
  876. /synth_ai/{demos → cli/demo_apps}/demo_task_apps/crafter/configs/crafter_fft_4b.toml +0 -0
  877. /synth_ai/{demos → cli/demo_apps}/demo_task_apps/crafter/configs/rl_from_base_qwen4b.toml +0 -0
  878. /synth_ai/{demos → cli/demo_apps}/demo_task_apps/math/__init__.py +0 -0
  879. /synth_ai/{demos → cli/demo_apps}/demo_task_apps/math/_common.py +0 -0
  880. /synth_ai/{demos → cli/demo_apps}/demo_task_apps/math/app.py +0 -0
  881. /synth_ai/{demos → cli/demo_apps}/demo_task_apps/math/config.toml +0 -0
  882. /synth_ai/{demos → cli/demo_apps}/demo_task_apps/math/deploy_modal.py +0 -0
  883. /synth_ai/{v0/lm/caching → core/apps}/__init__.py +0 -0
  884. /synth_ai/{tracing_v3 → core/tracing_v3}/abstractions.py +0 -0
  885. /synth_ai/{tracing_v3 → core/tracing_v3}/hooks.py +0 -0
  886. /synth_ai/{tracing_v3 → core/tracing_v3}/lm_call_record_abstractions.py +0 -0
  887. /synth_ai/{tracing_v3 → core/tracing_v3}/storage/__init__.py +0 -0
  888. /synth_ai/{tracing_v3 → core/tracing_v3}/storage/exceptions.py +0 -0
  889. /synth_ai/{tracing_v3 → core/tracing_v3}/storage/types.py +0 -0
  890. /synth_ai/{compound/cais.py → py.typed} +0 -0
  891. /synth_ai/{learning → sdk/learning}/algorithms.py +0 -0
  892. /synth_ai/{learning → sdk/learning}/config.py +0 -0
  893. /synth_ai/{learning → sdk/learning}/constants.py +0 -0
  894. /synth_ai/{learning → sdk/learning}/core.py +0 -0
  895. /synth_ai/{learning → sdk/learning}/gateway.py +0 -0
  896. /synth_ai/{learning → sdk/learning}/rl/__init__.py +0 -0
  897. /synth_ai/{learning → sdk/learning}/rl/config.py +0 -0
  898. /synth_ai/{learning → sdk/learning}/rl_client.py +0 -0
  899. /synth_ai/{learning → sdk/learning}/sft/__init__.py +0 -0
  900. /synth_ai/{learning → sdk/learning}/sse.py +0 -0
  901. /synth_ai/{task → sdk/task}/auth.py +0 -0
  902. /synth_ai/{task → sdk/task}/client.py +0 -0
  903. /synth_ai/{task → sdk/task}/errors.py +0 -0
  904. /synth_ai/{task → sdk/task}/health.py +0 -0
  905. /synth_ai/{task → sdk/task}/json.py +0 -0
  906. /synth_ai/{task → sdk/task}/rubrics.py +0 -0
  907. /synth_ai/{task → sdk/task}/vendors.py +0 -0
  908. {synth_ai-0.2.9.dev11.dist-info → synth_ai-0.4.1.dist-info}/WHEEL +0 -0
  909. {synth_ai-0.2.9.dev11.dist-info → synth_ai-0.4.1.dist-info}/licenses/LICENSE +0 -0
@@ -1,991 +0,0 @@
1
- """Task app configuration for a single-step math reasoning environment."""
2
-
3
- from __future__ import annotations
4
-
5
- import contextlib
6
- import os
7
- import random
8
- import re
9
- import uuid
10
- from collections.abc import Iterable, Mapping, MutableMapping, Sequence
11
- from dataclasses import dataclass
12
- from pathlib import Path
13
- from typing import Any, cast
14
-
15
- import httpx
16
- from datasets import load_dataset
17
- from fastapi import APIRouter, HTTPException, Request
18
- from pydantic import BaseModel, Field
19
- from synth_ai.task.apps import ModalDeploymentConfig, TaskAppEntry, register_task_app
20
- from synth_ai.task.contracts import (
21
- RolloutMetrics,
22
- RolloutRequest,
23
- RolloutResponse,
24
- RolloutStep,
25
- RolloutTrajectory,
26
- TaskInfo,
27
- )
28
- from synth_ai.task.datasets import TaskDatasetRegistry, TaskDatasetSpec
29
- from synth_ai.task.errors import http_exception
30
- from synth_ai.task.rubrics import Rubric, load_rubric
31
- from synth_ai.task.server import ProxyConfig, RubricBundle, TaskAppConfig
32
- from synth_ai.task.tracing_utils import (
33
- build_tracer_factory,
34
- resolve_sft_output_dir,
35
- resolve_tracing_db_url,
36
- tracing_env_enabled,
37
- )
38
- from synth_ai.task.vendors import normalize_vendor_keys
39
- from synth_ai.tracing_v3.session_tracer import SessionTracer
40
-
41
- REPO_ROOT = Path(__file__).resolve().parents[3]
42
-
43
- _modal_volume_candidate = Path(
44
- os.getenv("MATH_MODAL_DATASET_DIR", "/modal_volumes/math_dataset")
45
- ).expanduser()
46
- _modal_volume_root: Path | None = None
47
- try:
48
- _modal_volume_candidate.mkdir(parents=True, exist_ok=True)
49
- _modal_volume_root = _modal_volume_candidate
50
- except Exception:
51
- if _modal_volume_candidate.exists():
52
- _modal_volume_root = _modal_volume_candidate
53
-
54
- if _modal_volume_root is not None:
55
- hf_cache_path = _modal_volume_root / "hf_cache"
56
- local_dataset_dir = _modal_volume_root / "jsonl"
57
- local_dataset_dir.mkdir(parents=True, exist_ok=True)
58
- os.environ.setdefault("MATH_DATASET_LOCAL_DIR", str(local_dataset_dir))
59
- else:
60
- hf_cache_path = Path(
61
- os.getenv("MATH_DATASET_CACHE_DIR", str(REPO_ROOT / ".cache" / "hf-datasets"))
62
- ).expanduser()
63
-
64
- hf_cache_path.mkdir(parents=True, exist_ok=True)
65
- os.environ.setdefault("MATH_DATASET_CACHE_DIR", str(hf_cache_path))
66
- os.environ.setdefault("HF_HOME", str(hf_cache_path))
67
- os.environ.setdefault("HF_DATASETS_CACHE", str(hf_cache_path))
68
- os.environ.setdefault("HUGGINGFACE_HUB_CACHE", str(hf_cache_path))
69
-
70
- HF_DATASETS_CACHE = hf_cache_path
71
- DATASET_NAME = os.getenv("MATH_DATASET_NAME", "nlile/hendrycks-MATH-benchmark")
72
- DATASET_CONFIG = os.getenv("MATH_DATASET_CONFIG", "")
73
- DEFAULT_SPLIT = os.getenv("MATH_DATASET_DEFAULT_SPLIT", "train")
74
- AVAILABLE_SPLITS: tuple[str, ...] = (
75
- DEFAULT_SPLIT,
76
- os.getenv("MATH_DATASET_VALIDATION_SPLIT", "test"),
77
- os.getenv("MATH_DATASET_TEST_SPLIT", "test"),
78
- )
79
- TOOL_NAME = "math_submit"
80
- PROBLEM_KEYS: tuple[str, ...] = ("problem", "question", "prompt", "query")
81
- SOLUTION_KEYS: tuple[str, ...] = ("solution", "answer", "final_answer", "solution_text")
82
-
83
- REWARD_POSITIVE = float(os.getenv("MATH_REWARD_POSITIVE", "1.0"))
84
- REWARD_NEGATIVE_NO_TOOL = float(os.getenv("MATH_REWARD_NEGATIVE_NO_TOOL", "-1.0"))
85
- REWARD_NEGATIVE_NO_ANSWER = float(os.getenv("MATH_REWARD_NEGATIVE_NO_ANSWER", "-0.5"))
86
-
87
- HF_TOKEN_ENV_KEYS: tuple[str, ...] = (
88
- "HF_DATASETS_TOKEN",
89
- "HUGGINGFACEHUB_API_TOKEN",
90
- "HUGGINGFACE_TOKEN",
91
- )
92
-
93
- ## Single-source dataset policy: use a single known-good HF dataset id by default.
94
-
95
- MATH_DATASET_SPEC = TaskDatasetSpec(
96
- id="math_single_step",
97
- name="MATH Single Step",
98
- version="1.0.0",
99
- splits=list(dict.fromkeys(split for split in AVAILABLE_SPLITS if split)),
100
- default_split=DEFAULT_SPLIT,
101
- description="Single-step math reasoning problems sourced from the Hendrycks MATH dataset.",
102
- )
103
-
104
-
105
- _BOXED_MARKERS: tuple[str, ...] = ("\\boxed", "boxed")
106
-
107
-
108
- def _extract_boxed(text: str) -> str | None:
109
- if not text:
110
- return None
111
- for marker in _BOXED_MARKERS:
112
- start = text.find(marker)
113
- if start == -1:
114
- continue
115
- brace_start = text.find("{", start)
116
- if brace_start == -1:
117
- continue
118
- depth = 1
119
- idx = brace_start + 1
120
- while idx < len(text) and depth > 0:
121
- ch = text[idx]
122
- if ch == "{":
123
- depth += 1
124
- elif ch == "}":
125
- depth -= 1
126
- idx += 1
127
- if depth == 0:
128
- return text[brace_start + 1 : idx - 1].strip()
129
- return None
130
-
131
-
132
- _FRAC_PATTERN = re.compile(r"\\?frac\{([^{}]+)\}\{([^{}]+)\}")
133
- _SQRT_PATTERN = re.compile(r"\\?sqrt\{([^{}]+)\}")
134
-
135
-
136
- def _normalize_final_answer(text: str) -> str:
137
- raw = str(text or "").strip()
138
- if not raw:
139
- return ""
140
- boxed = _extract_boxed(raw)
141
- if boxed:
142
- raw = boxed
143
- raw = raw.strip().strip("$")
144
- raw = raw.replace("\\left", "").replace("\\right", "")
145
- raw = raw.replace("\\!", "").replace("\\,", " ").replace("\\;", " ")
146
- raw = raw.replace("left", "").replace("right", "")
147
- raw = raw.replace("\\times", "*").replace("\\cdot", "*")
148
- raw = raw.replace("\\pi", "pi").replace("\\theta", "theta").replace("\\phi", "phi")
149
- raw = raw.replace("\\pm", "+/-").replace("\\mp", "-/+")
150
- raw = raw.replace("^{\\circ}", "deg").replace("^\\circ", "deg").replace("\\circ", "deg")
151
-
152
- def _frac_sub(match: re.Match[str]) -> str:
153
- num = match.group(1).strip()
154
- den = match.group(2).strip()
155
- return f"({num})/({den})"
156
-
157
- def _sqrt_sub(match: re.Match[str]) -> str:
158
- inner = match.group(1).strip()
159
- return f"sqrt({inner})"
160
-
161
- raw = _FRAC_PATTERN.sub(_frac_sub, raw)
162
- raw = _SQRT_PATTERN.sub(_sqrt_sub, raw)
163
- raw = raw.replace("\\", "")
164
- raw = raw.replace("{", "").replace("}", "")
165
- raw = raw.replace(" ", "")
166
- raw = raw.rstrip(".")
167
- return raw
168
-
169
-
170
- class MathDataset:
171
- """Lazy Hugging Face dataset loader for EleutherAI/math splits."""
172
-
173
- def __init__(self, *, name: str, config: str, splits: Sequence[str]) -> None:
174
- self.name = name
175
- self.config = config
176
- self.splits = [split for split in splits if split]
177
- self._cache: dict[str, Any] = {}
178
- self._local_dir = os.getenv("MATH_DATASET_LOCAL_DIR")
179
- self._hf_token: str | None = None
180
- for key in HF_TOKEN_ENV_KEYS:
181
- value = os.getenv(key)
182
- if value:
183
- trimmed = value.strip()
184
- if trimmed:
185
- self._hf_token = trimmed
186
- break
187
- # No multi-candidate fallback: enforce explicit dataset id
188
-
189
- def _local_file_for_split(self, split: str) -> Path | None:
190
- specific = os.getenv(f"MATH_DATASET_LOCAL_{split.upper()}_FILE")
191
- if specific:
192
- path = Path(specific).expanduser()
193
- if path.exists():
194
- return path
195
- if self._local_dir:
196
- candidate = Path(self._local_dir).expanduser() / f"{split}.jsonl"
197
- if candidate.exists():
198
- return candidate
199
- return None
200
-
201
- def _load_split(self, split: str):
202
- # Treat 'validation' as an alias for 'test' for datasets without a separate validation split
203
- if split not in self.splits and split.lower() == "validation":
204
- split = "test"
205
- if split not in self.splits:
206
- raise ValueError(f"Unknown split '{split}'. Available: {self.splits}")
207
- if split not in self._cache:
208
- local_file = self._local_file_for_split(split)
209
- if local_file is not None:
210
- dataset = load_dataset(
211
- "json", data_files=str(local_file), cache_dir=str(HF_DATASETS_CACHE)
212
- )
213
- self._cache[split] = dataset["train"]
214
- else:
215
- try:
216
- load_kwargs: dict[str, Any] = {"split": split}
217
- if self.config:
218
- load_kwargs["name"] = self.config
219
- if self._hf_token:
220
- load_kwargs["use_auth_token"] = self._hf_token
221
- ds = load_dataset(self.name, cache_dir=str(HF_DATASETS_CACHE), **load_kwargs)
222
- self._cache[split] = ds
223
- if self._local_dir:
224
- local_dir = Path(self._local_dir).expanduser()
225
- target = local_dir / f"{split}.jsonl"
226
- if not target.exists() and hasattr(ds, "to_json"):
227
- tmp_path = target.with_name(target.name + ".tmp")
228
- try:
229
- local_dir.mkdir(parents=True, exist_ok=True)
230
- ds.to_json(str(tmp_path))
231
- tmp_path.replace(target)
232
- except Exception:
233
- with contextlib.suppress(FileNotFoundError):
234
- tmp_path.unlink()
235
- except Exception as exc:
236
- hints = [
237
- "Failed to download MATH dataset from Hugging Face.",
238
- f"Dataset: {self.name} | Config: {self.config or 'none'} | Split: {split}",
239
- "If this persists, verify MATH_DATASET_NAME/MATH_DATASET_CONFIG or set MATH_DATASET_LOCAL_DIR to pre-downloaded JSONL files.",
240
- ]
241
- raise RuntimeError(" ".join(hints)) from exc
242
- return self._cache[split]
243
-
244
- def sample(self, *, split: str, index: int | None = None) -> dict[str, Any]:
245
- dataset = self._load_split(split)
246
- if len(dataset) == 0:
247
- raise RuntimeError(f"Dataset split '{split}' is empty")
248
- if index is None:
249
- index = random.randint(0, len(dataset) - 1)
250
- idx = int(index) % len(dataset)
251
- item = dataset[int(idx)]
252
-
253
- raw_problem = ""
254
- for key in PROBLEM_KEYS:
255
- value = item.get(key)
256
- if isinstance(value, str) and value.strip():
257
- raw_problem = value.strip()
258
- break
259
- if not raw_problem:
260
- raise RuntimeError(f"Sample missing problem field for split '{split}' index {idx}")
261
-
262
- solution_value: Any = None
263
- for key in SOLUTION_KEYS:
264
- if key in item:
265
- solution_value = item[key]
266
- break
267
- if solution_value is None:
268
- raise RuntimeError(f"Sample missing solution field for split '{split}' index {idx}")
269
-
270
- # Solutions can contain reasoning and final answer; take final line by convention
271
- if isinstance(solution_value, list):
272
- solution_text = "\n".join(str(part) for part in solution_value)
273
- else:
274
- solution_text = str(solution_value)
275
- lines = [line.strip() for line in solution_text.strip().splitlines() if line.strip()]
276
- final_line = ""
277
- for line in reversed(lines):
278
- lowered = line.lower()
279
- if "boxed" in lowered or "answer" in lowered:
280
- final_line = line
281
- break
282
- if not final_line and lines:
283
- final_line = lines[-1]
284
- candidate_answer = final_line or solution_text.strip()
285
- normalized_answer = _normalize_final_answer(candidate_answer)
286
- if not normalized_answer:
287
- normalized_answer = _normalize_final_answer(solution_text)
288
- return {
289
- "index": idx,
290
- "split": split,
291
- "problem": raw_problem,
292
- "answer": normalized_answer,
293
- "raw_solution": solution_text,
294
- }
295
-
296
- def size(self, split: str) -> int:
297
- dataset = self._load_split(split)
298
- return len(dataset)
299
-
300
- def ensure_ready(self, required_splits: Sequence[str]) -> None:
301
- errors: list[str] = []
302
- for split in required_splits:
303
- if not split:
304
- continue
305
- try:
306
- self._load_split(split)
307
- except Exception as exc:
308
- errors.append(f"{split}: {exc}")
309
- if errors:
310
- raise RuntimeError("Dataset preparation failed:\n" + "\n".join(errors))
311
-
312
-
313
- @dataclass
314
- class MathEnvState:
315
- env_id: str
316
- split: str
317
- index: int
318
- problem: str
319
- answer: str
320
- raw_solution: str
321
- done: bool = False
322
-
323
-
324
- class MathEnvironmentManager:
325
- """Stores in-flight environment state keyed by env_id."""
326
-
327
- def __init__(self, dataset: MathDataset) -> None:
328
- self.dataset = dataset
329
- self._states: dict[str, MathEnvState] = {}
330
-
331
- def create(self, *, split: str, index: int | None, seed: int | None) -> MathEnvState:
332
- if index is None and seed is not None:
333
- index = seed
334
- sample = self.dataset.sample(split=split, index=index)
335
- env_id = str(uuid.uuid4())
336
- state = MathEnvState(
337
- env_id=env_id,
338
- split=split,
339
- index=int(sample["index"]),
340
- problem=sample["problem"],
341
- answer=sample["answer"],
342
- raw_solution=sample["raw_solution"],
343
- )
344
- self._states[env_id] = state
345
- return state
346
-
347
- def get(self, env_id: str) -> MathEnvState:
348
- if env_id not in self._states:
349
- raise KeyError(f"Unknown env_id: {env_id}")
350
- return self._states[env_id]
351
-
352
- def terminate(self, env_id: str) -> None:
353
- self._states.pop(env_id, None)
354
-
355
-
356
- class InitializePayload(BaseModel):
357
- seed: int | None = None
358
- config: dict[str, Any] = Field(default_factory=dict)
359
-
360
-
361
- def _observation_from_state(state: MathEnvState) -> dict[str, Any]:
362
- return {
363
- "problem": state.problem,
364
- "split": state.split,
365
- "index": state.index,
366
- }
367
-
368
-
369
- def _score_submission(
370
- state: MathEnvState, tool_calls: Sequence[Mapping[str, Any]]
371
- ) -> tuple[float, str, bool]:
372
- if not tool_calls:
373
- return REWARD_NEGATIVE_NO_TOOL, "missing_tool_call", False
374
- call = tool_calls[0]
375
- tool_name = str(call.get("tool") or "").strip()
376
- if tool_name != TOOL_NAME:
377
- return REWARD_NEGATIVE_NO_TOOL, "wrong_tool", False
378
- args = call.get("args") or {}
379
- answer = _normalize_final_answer(str(args.get("answer") or ""))
380
- if not answer:
381
- return REWARD_NEGATIVE_NO_ANSWER, "blank_answer", False
382
- is_correct = answer == state.answer
383
- return (
384
- (REWARD_POSITIVE if is_correct else 0.0),
385
- ("correct" if is_correct else "incorrect"),
386
- is_correct,
387
- )
388
-
389
-
390
- math_router = APIRouter()
391
-
392
-
393
- def _preview_tool_calls(tool_calls: Sequence[Mapping[str, Any]]) -> list[dict[str, Any]]:
394
- """Return a compact, log-friendly preview of tool calls.
395
-
396
- Truncates long fields to avoid noisy logs and leaking excessive content.
397
- """
398
- preview: list[dict[str, Any]] = []
399
- for call in list(tool_calls or [])[:3]:
400
- args = dict(call.get("args") or {})
401
- answer = str(args.get("answer") or "")
402
- # Hard truncate to keep logs compact
403
- answer_short = answer[:120] + ("…" if len(answer) > 120 else "")
404
- preview.append(
405
- {
406
- "tool": call.get("tool"),
407
- "answer": answer_short,
408
- }
409
- )
410
- return preview
411
-
412
-
413
- def _event_and_outcome_components(
414
- tool_calls: Sequence[Mapping[str, Any]], *, correct: bool, reward: float
415
- ) -> dict[str, float]:
416
- """Approximate component-wise scores for RL-style logs.
417
-
418
- - env: task-level scalar reward (our single-step outcome)
419
- - rubric_event: 1.0 if a valid tool call with non-empty answer was made else 0.0
420
- - rubric_outcome: 1.0 if final answer was correct else 0.0
421
- """
422
- has_valid_tool = False
423
- if tool_calls:
424
- first = tool_calls[0] or {}
425
- if str(first.get("tool") or "") == TOOL_NAME:
426
- args = first.get("args") or {}
427
- ans = str(args.get("answer") or "").strip()
428
- has_valid_tool = bool(ans)
429
- return {
430
- "env": float(reward),
431
- "rubric_event": 1.0 if has_valid_tool else 0.0,
432
- "rubric_outcome": 1.0 if bool(correct) else 0.0,
433
- }
434
-
435
-
436
- @math_router.post("/env/math/initialize")
437
- async def initialize_env(request: Request, payload: InitializePayload) -> dict[str, Any]:
438
- manager: MathEnvironmentManager = request.app.state.math_env_manager
439
- split = str(payload.config.get("split") or DEFAULT_SPLIT)
440
- seed = payload.seed
441
- index = None
442
- if payload.config.get("index") is not None:
443
- index = int(payload.config["index"])
444
- state = manager.create(split=split, index=index, seed=seed)
445
- return {
446
- "env_id": state.env_id,
447
- "observation": _observation_from_state(state),
448
- "info": {"raw_solution": state.raw_solution},
449
- }
450
-
451
-
452
- @math_router.post("/env/math/step")
453
- async def step_env(request: Request, payload: dict[str, Any]) -> dict[str, Any]:
454
- manager: MathEnvironmentManager = request.app.state.math_env_manager
455
- env_id = str(payload.get("env_id") or "")
456
- if not env_id:
457
- raise HTTPException(status_code=400, detail="env_id required")
458
- try:
459
- state = manager.get(env_id)
460
- except KeyError as exc: # pragma: no cover - defensive
461
- raise HTTPException(status_code=404, detail=str(exc)) from exc
462
-
463
- action = payload.get("action") or {}
464
- tool_calls = action.get("tool_calls") or payload.get("tool_calls") or []
465
- reward, status, correct = _score_submission(state, tool_calls)
466
- with contextlib.suppress(Exception):
467
- print(
468
- "[MATH_STEP] env_id=",
469
- state.env_id,
470
- " split=",
471
- state.split,
472
- " index=",
473
- state.index,
474
- " calls=",
475
- _preview_tool_calls(tool_calls),
476
- " reward=",
477
- reward,
478
- " status=",
479
- status,
480
- " correct=",
481
- correct,
482
- " components=",
483
- _event_and_outcome_components(tool_calls, correct=correct, reward=reward),
484
- flush=True,
485
- )
486
- state.done = True
487
-
488
- observation = _observation_from_state(state)
489
- observation["status"] = status
490
- return {
491
- "observation": observation,
492
- "done": True,
493
- "reward": reward,
494
- "info": {
495
- "correct": correct,
496
- "expected_answer": state.answer,
497
- "raw_solution": state.raw_solution,
498
- },
499
- }
500
-
501
-
502
- @math_router.post("/env/math/terminate")
503
- async def terminate_env(request: Request, payload: dict[str, Any]) -> dict[str, Any]:
504
- manager: MathEnvironmentManager = request.app.state.math_env_manager
505
- env_id = str(payload.get("env_id") or "")
506
- if env_id:
507
- manager.terminate(env_id)
508
- return {"ok": True}
509
-
510
-
511
- def _resolve_inference_url(base_url: str) -> str:
512
- normalized = (base_url or "").rstrip("/")
513
- if not normalized:
514
- raise RuntimeError("policy.config.inference_url required")
515
- if normalized.endswith("/v1/chat/completions"):
516
- return normalized
517
- if normalized.endswith("/chat/completions"):
518
- return normalized
519
- if normalized.endswith("/v1"):
520
- return f"{normalized}/chat/completions"
521
- return f"{normalized}/v1/chat/completions"
522
-
523
-
524
- async def _call_inference(
525
- policy_config: Mapping[str, Any], observation: Mapping[str, Any]
526
- ) -> tuple[list[dict[str, Any]], dict[str, Any]]:
527
- inference_url = str(policy_config.get("inference_url") or "").rstrip("/")
528
- if not inference_url:
529
- raise RuntimeError("policy.config.inference_url required for rollout")
530
- model = policy_config.get("model")
531
- max_tokens = policy_config.get("max_tokens", 512)
532
- temperature = policy_config.get("temperature", 0.0)
533
- top_p = policy_config.get("top_p", 1.0)
534
-
535
- messages = [
536
- {
537
- "role": "system",
538
- "content": (
539
- "You are a math solver. Read the problem carefully and respond with a single"
540
- f" tool call using the function `{TOOL_NAME}`."
541
- "\nRules:\n"
542
- "- Do all reasoning internally.\n"
543
- "- The tool call must include ONLY the final numeric or simplified answer in the"
544
- " `answer` field.\n"
545
- "- DO NOT include explanations, units, or extra text in the answer."
546
- ),
547
- },
548
- {
549
- "role": "user",
550
- "content": (
551
- "Problem:\n"
552
- + str(observation.get("problem") or "")
553
- + "\nSubmit the final answer via the tool call."
554
- ),
555
- },
556
- ]
557
-
558
- payload: dict[str, Any] = {
559
- "model": model,
560
- "messages": messages,
561
- "tools": [
562
- {
563
- "type": "function",
564
- "function": {
565
- "name": TOOL_NAME,
566
- "description": "Submit the final answer for the math problem.",
567
- "parameters": {
568
- "type": "object",
569
- "properties": {
570
- "answer": {
571
- "type": "string",
572
- "description": "Final answer in simplest form",
573
- },
574
- "explanation": {
575
- "type": "string",
576
- "description": "Optional explanation of reasoning",
577
- },
578
- },
579
- "required": ["answer"],
580
- "additionalProperties": False,
581
- },
582
- },
583
- }
584
- ],
585
- "tool_choice": {"type": "function", "function": {"name": TOOL_NAME}},
586
- "temperature": temperature,
587
- "top_p": top_p,
588
- "max_tokens": max_tokens,
589
- }
590
-
591
- final_url = _resolve_inference_url(inference_url)
592
- async with httpx.AsyncClient(timeout=httpx.Timeout(60.0)) as client:
593
- response = await client.post(final_url, json=payload)
594
- try:
595
- data = response.json()
596
- except Exception as exc:
597
- raise http_exception(
598
- 502,
599
- "inference_invalid_response",
600
- "Inference provider returned invalid JSON",
601
- extra={"body": response.text[:800]},
602
- ) from exc
603
- if response.status_code >= 500:
604
- raise http_exception(
605
- 502,
606
- "inference_upstream_error",
607
- "Inference provider returned an error",
608
- extra={"status": response.status_code, "body": data},
609
- )
610
- if response.status_code >= 400:
611
- raise http_exception(
612
- 400,
613
- "inference_request_invalid",
614
- "Invalid inference request",
615
- extra={"status": response.status_code, "body": data},
616
- )
617
-
618
- tool_calls = []
619
- choices = data.get("choices") or []
620
- if choices:
621
- message = choices[0].get("message") or {}
622
- raw_calls = message.get("tool_calls") or []
623
- for call in raw_calls:
624
- function = call.get("function") or {}
625
- name = function.get("name")
626
- arguments = function.get("arguments")
627
- parsed_args: dict[str, Any]
628
- if isinstance(arguments, str):
629
- try:
630
- import json
631
-
632
- parsed_args = json.loads(arguments)
633
- except Exception:
634
- parsed_args = {}
635
- elif isinstance(arguments, MutableMapping):
636
- parsed_args = dict(arguments)
637
- else:
638
- parsed_args = {}
639
- tool_calls.append({"tool": name, "args": parsed_args})
640
- # Lightweight provider-side logging
641
- with contextlib.suppress(Exception):
642
- print(
643
- "[MATH_INFER] model=",
644
- model,
645
- " calls=",
646
- _preview_tool_calls(tool_calls),
647
- flush=True,
648
- )
649
- return tool_calls, data
650
-
651
-
652
- async def rollout_executor(request: RolloutRequest, fastapi_request: Request) -> RolloutResponse:
653
- dataset: MathDataset = fastapi_request.app.state.math_dataset
654
- split = str(((request.env.config or {}).get("split")) or DEFAULT_SPLIT)
655
- sample = dataset.sample(split=split, index=request.env.seed)
656
-
657
- observation = {
658
- "problem": sample["problem"],
659
- "split": sample["split"],
660
- "index": sample["index"],
661
- }
662
-
663
- tool_calls: list[dict[str, Any]] = []
664
- inference_payload: dict[str, Any] | None = None
665
- error_info: dict[str, Any] = {}
666
- try:
667
- tool_calls, inference_payload = await _call_inference(
668
- request.policy.config or {}, observation
669
- )
670
- except HTTPException as http_err:
671
- tool_calls = []
672
- error_info = {"error": http_err.detail, "code": http_err.status_code}
673
- except Exception as exc:
674
- tool_calls = []
675
- error_info = {"error": str(exc)}
676
-
677
- reward, status, correct = _score_submission(
678
- MathEnvState(
679
- env_id="rollout",
680
- split=sample["split"],
681
- index=sample["index"],
682
- problem=sample["problem"],
683
- answer=sample["answer"],
684
- raw_solution=sample["raw_solution"],
685
- ),
686
- tool_calls,
687
- )
688
-
689
- # Log a concise summary so we can debug reward=0 issues in production
690
- with contextlib.suppress(Exception):
691
- print(
692
- "[MATH_ROLLOUT] run=",
693
- request.run_id,
694
- " split=",
695
- sample["split"],
696
- " index=",
697
- sample["index"],
698
- " calls=",
699
- _preview_tool_calls(tool_calls),
700
- " reward=",
701
- reward,
702
- " status=",
703
- status,
704
- " correct=",
705
- correct,
706
- " components=",
707
- _event_and_outcome_components(tool_calls, correct=correct, reward=reward),
708
- flush=True,
709
- )
710
-
711
- step = RolloutStep(
712
- obs=observation,
713
- tool_calls=tool_calls,
714
- reward=reward,
715
- done=True,
716
- info={
717
- "expected_answer": sample["answer"],
718
- "status": status,
719
- "correct": correct,
720
- "raw_solution": sample["raw_solution"],
721
- "tool_call_preview": _preview_tool_calls(tool_calls),
722
- **error_info,
723
- },
724
- )
725
-
726
- trajectory = RolloutTrajectory(
727
- env_id=f"math::{sample['split']}::{sample['index']}",
728
- policy_id=request.policy.policy_id or "policy",
729
- steps=[step],
730
- final={
731
- "observation": {**observation, "status": status},
732
- "reward": reward,
733
- },
734
- length=1,
735
- )
736
- metrics = RolloutMetrics(
737
- episode_returns=[reward],
738
- mean_return=reward,
739
- num_steps=1,
740
- num_episodes=1,
741
- outcome_score=reward,
742
- events_score=reward,
743
- details={"status": status, "correct": correct},
744
- )
745
-
746
- # Include a minimal trace when requested or tracing is enabled via env
747
- include_trace = bool(
748
- (request.record and getattr(request.record, "return_trace", False))
749
- or os.getenv("TASKAPP_TRACING_ENABLED")
750
- )
751
- trace_payload = None
752
- if include_trace:
753
- try:
754
- # Minimal structured trace for assertions
755
- trace_payload = {
756
- "session_id": str(uuid.uuid4()),
757
- "events_count": 1,
758
- "decision_rewards": [reward],
759
- "lm_calls": (
760
- [{"prompt": str(observation.get("problem", "")), "response": str(tool_calls)}]
761
- if tool_calls
762
- else []
763
- ),
764
- "metadata": {
765
- "env": "math_single_step",
766
- "split": sample["split"],
767
- "index": sample["index"],
768
- "status": status,
769
- },
770
- }
771
- except Exception:
772
- trace_payload = None
773
-
774
- return RolloutResponse(
775
- run_id=request.run_id,
776
- trajectories=[trajectory],
777
- branches={},
778
- metrics=metrics,
779
- aborted=False,
780
- ops_executed=2,
781
- trace=trace_payload,
782
- )
783
-
784
-
785
- def build_dataset() -> tuple[TaskDatasetRegistry, MathDataset]:
786
- registry = TaskDatasetRegistry()
787
- dataset = MathDataset(name=DATASET_NAME, config=DATASET_CONFIG, splits=AVAILABLE_SPLITS)
788
- # Ensure default split is available when the task app boots
789
- try:
790
- dataset.ensure_ready([DEFAULT_SPLIT])
791
- except Exception as exc:
792
- raise RuntimeError(
793
- "Failed to initialise math dataset. Set MATH_DATASET_LOCAL_DIR or ensure network access.\n"
794
- f"Underlying error: {exc}"
795
- ) from exc
796
- registry.register(MATH_DATASET_SPEC, lambda _spec: dataset, cache=True)
797
- return registry, dataset
798
-
799
-
800
- def _base_task_info() -> TaskInfo:
801
- return TaskInfo(
802
- task={"id": "math_single_step", "name": "Math Single Step", "version": "1.0.0"},
803
- environments=["math"],
804
- action_space={
805
- "type": "tool_call",
806
- "tools": [
807
- {
808
- "name": TOOL_NAME,
809
- "description": "Submit the final answer for the math problem.",
810
- "schema": {"answer": "string"},
811
- }
812
- ],
813
- "max_calls": 1,
814
- },
815
- observation={
816
- "summary": "Single math word problem presented as plain text.",
817
- "keys": ["problem"],
818
- },
819
- dataset={
820
- **MATH_DATASET_SPEC.model_dump(),
821
- "hf_dataset": DATASET_NAME,
822
- "hf_config": DATASET_CONFIG,
823
- },
824
- rubric={
825
- "version": "1",
826
- "criteria_count": 1,
827
- "source": "inline",
828
- },
829
- inference={
830
- "supports_proxy": True,
831
- "tool": {"name": TOOL_NAME, "parallel_tool_calls": False},
832
- },
833
- capabilities={
834
- "supports_rollout": True,
835
- "supports_env_lifecycle": True,
836
- "requires_api_key_header": True,
837
- },
838
- limits={"max_turns": 1},
839
- )
840
-
841
-
842
- OUTCOME_RUBRIC: Rubric = cast(
843
- Rubric,
844
- load_rubric(
845
- {
846
- "version": "1",
847
- "goal_text": "Encourage correct single-step math answers via tool calls.",
848
- "aggregation": "weighted_sum",
849
- "criteria": [
850
- {
851
- "id": "correct_answer",
852
- "description": "Submit the correct final answer using the math_submit tool.",
853
- "weight": 1.0,
854
- }
855
- ],
856
- }
857
- ),
858
- )
859
-
860
- EVENTS_RUBRIC: Rubric = cast(
861
- Rubric,
862
- load_rubric(
863
- {
864
- "version": "1",
865
- "goal_text": "Penalize missing or malformed tool calls.",
866
- "aggregation": "weighted_sum",
867
- "criteria": [
868
- {
869
- "id": "tool_usage",
870
- "description": "Make exactly one tool call with an answer string.",
871
- "weight": 1.0,
872
- }
873
- ],
874
- }
875
- ),
876
- )
877
-
878
-
879
- def describe_taskset(dataset: MathDataset) -> dict[str, Any]:
880
- return {
881
- **MATH_DATASET_SPEC.model_dump(),
882
- "hf_dataset": DATASET_NAME,
883
- "hf_config": DATASET_CONFIG,
884
- "sizes": {split: dataset.size(split) for split in dataset.splits},
885
- }
886
-
887
-
888
- def provide_task_instances(dataset: MathDataset, seeds: Sequence[int]) -> Iterable[TaskInfo]:
889
- info = _base_task_info()
890
- for seed in seeds:
891
- sample = dataset.sample(split=DEFAULT_SPLIT, index=seed)
892
- yield TaskInfo(
893
- task=info.task,
894
- environments=info.environments,
895
- action_space=info.action_space,
896
- observation={**info.observation, "sample_index": sample["index"]},
897
- dataset={
898
- **info.dataset,
899
- "split": sample["split"],
900
- "index": sample["index"],
901
- },
902
- rubric=info.rubric,
903
- inference=info.inference,
904
- capabilities=info.capabilities,
905
- limits=info.limits,
906
- )
907
-
908
-
909
- def build_config() -> TaskAppConfig:
910
- registry, dataset = build_dataset()
911
- base_info = _base_task_info()
912
-
913
- tracing_enabled = tracing_env_enabled()
914
- tracing_db_url = resolve_tracing_db_url()
915
- tracer_factory = build_tracer_factory(
916
- SessionTracer, enabled=tracing_enabled, db_url=tracing_db_url
917
- )
918
- sft_output_dir = resolve_sft_output_dir()
919
-
920
- app_state: dict[str, Any] = {
921
- "math_dataset": dataset,
922
- "math_env_manager": MathEnvironmentManager(dataset),
923
- "tracing_enabled": tracing_enabled,
924
- }
925
- if tracer_factory is not None:
926
- app_state["session_tracer_factory"] = tracer_factory
927
- if sft_output_dir:
928
- app_state["sft_output_dir"] = sft_output_dir
929
-
930
- proxy_keys = normalize_vendor_keys()
931
- openai_key = proxy_keys.get("OPENAI_API_KEY")
932
- groq_key = proxy_keys.get("GROQ_API_KEY")
933
- proxy_config = ProxyConfig(
934
- enable_openai=openai_key is not None,
935
- enable_groq=groq_key is not None,
936
- system_hint=(
937
- "You must respond with a single math_submit tool call containing only the final answer."
938
- ),
939
- )
940
-
941
- config = TaskAppConfig(
942
- app_id="math-single-step",
943
- name="Math Single Step Task",
944
- description="Single-step math reasoning environment built on the MATH dataset.",
945
- base_task_info=base_info,
946
- describe_taskset=lambda: describe_taskset(dataset),
947
- provide_task_instances=lambda seeds: provide_task_instances(dataset, seeds),
948
- rollout=rollout_executor,
949
- dataset_registry=registry,
950
- rubrics=RubricBundle(outcome=OUTCOME_RUBRIC, events=EVENTS_RUBRIC),
951
- proxy=proxy_config,
952
- routers=(math_router,),
953
- app_state=app_state,
954
- cors_origins=["*"],
955
- )
956
- return config
957
-
958
-
959
- register_task_app(
960
- entry=TaskAppEntry(
961
- app_id="math-single-step",
962
- description="Single-step math reasoning task app using EleutherAI/math dataset.",
963
- config_factory=build_config,
964
- aliases=("math-rl",),
965
- env_files=("examples/rl/.env",),
966
- modal=ModalDeploymentConfig(
967
- app_name="synth-math-single-step",
968
- pip_packages=(
969
- "datasets>=4.0.0",
970
- "fastapi>=0.115.0",
971
- "pydantic>=2.0.0",
972
- "httpx>=0.26.0",
973
- "requests>=2.32.0",
974
- "python-dotenv>=1.0.0",
975
- "diskcache>=5.6.3",
976
- "duckdb>=1.0.0",
977
- "ty>=0.0.1a5",
978
- "toml>=0.10.2",
979
- "aiosqlite>=0.21.0",
980
- "libsql>=0.1.8",
981
- "pynacl>=1.5.0",
982
- "sqlalchemy>=2.0.42",
983
- ),
984
- extra_local_dirs=(
985
- (str(REPO_ROOT / "synth_ai"), "/opt/synth_ai_repo/synth_ai"),
986
- (str(REPO_ROOT / "examples" / "rl"), "/opt/synth_ai_repo/examples/rl"),
987
- ),
988
- volume_mounts=(("math-dataset-cache", "/modal_volumes/math_dataset"),),
989
- ),
990
- )
991
- )