flock-core 0.5.0b28__py3-none-any.whl → 0.5.0b51__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 flock-core might be problematic. Click here for more details.

Files changed (469) hide show
  1. flock/__init__.py +12 -217
  2. flock/agent.py +678 -0
  3. flock/api/themes.py +71 -0
  4. flock/artifacts.py +79 -0
  5. flock/cli.py +75 -0
  6. flock/components.py +173 -0
  7. flock/dashboard/__init__.py +28 -0
  8. flock/dashboard/collector.py +283 -0
  9. flock/dashboard/events.py +182 -0
  10. flock/dashboard/launcher.py +230 -0
  11. flock/dashboard/service.py +537 -0
  12. flock/dashboard/websocket.py +235 -0
  13. flock/engines/__init__.py +6 -0
  14. flock/engines/dspy_engine.py +856 -0
  15. flock/examples.py +128 -0
  16. flock/frontend/README.md +678 -0
  17. flock/frontend/docs/DESIGN_SYSTEM.md +1980 -0
  18. flock/frontend/index.html +12 -0
  19. flock/frontend/package-lock.json +4347 -0
  20. flock/frontend/package.json +48 -0
  21. flock/frontend/src/App.tsx +79 -0
  22. flock/frontend/src/__tests__/e2e/critical-scenarios.test.tsx +587 -0
  23. flock/frontend/src/__tests__/integration/filtering-e2e.test.tsx +387 -0
  24. flock/frontend/src/__tests__/integration/graph-rendering.test.tsx +640 -0
  25. flock/frontend/src/__tests__/integration/indexeddb-persistence.test.tsx +699 -0
  26. flock/frontend/src/components/common/BuildInfo.tsx +39 -0
  27. flock/frontend/src/components/common/EmptyState.module.css +115 -0
  28. flock/frontend/src/components/common/EmptyState.tsx +128 -0
  29. flock/frontend/src/components/common/ErrorBoundary.module.css +169 -0
  30. flock/frontend/src/components/common/ErrorBoundary.tsx +118 -0
  31. flock/frontend/src/components/common/KeyboardShortcutsDialog.css +251 -0
  32. flock/frontend/src/components/common/KeyboardShortcutsDialog.tsx +151 -0
  33. flock/frontend/src/components/common/LoadingSpinner.module.css +97 -0
  34. flock/frontend/src/components/common/LoadingSpinner.tsx +29 -0
  35. flock/frontend/src/components/controls/PublishControl.css +547 -0
  36. flock/frontend/src/components/controls/PublishControl.test.tsx +543 -0
  37. flock/frontend/src/components/controls/PublishControl.tsx +432 -0
  38. flock/frontend/src/components/details/DetailWindowContainer.tsx +62 -0
  39. flock/frontend/src/components/details/LiveOutputTab.test.tsx +792 -0
  40. flock/frontend/src/components/details/LiveOutputTab.tsx +220 -0
  41. flock/frontend/src/components/details/MessageHistoryTab.tsx +299 -0
  42. flock/frontend/src/components/details/NodeDetailWindow.test.tsx +501 -0
  43. flock/frontend/src/components/details/NodeDetailWindow.tsx +218 -0
  44. flock/frontend/src/components/details/RunStatusTab.tsx +307 -0
  45. flock/frontend/src/components/details/tabs.test.tsx +1015 -0
  46. flock/frontend/src/components/filters/CorrelationIDFilter.module.css +102 -0
  47. flock/frontend/src/components/filters/CorrelationIDFilter.test.tsx +197 -0
  48. flock/frontend/src/components/filters/CorrelationIDFilter.tsx +121 -0
  49. flock/frontend/src/components/filters/FilterBar.module.css +29 -0
  50. flock/frontend/src/components/filters/FilterBar.test.tsx +133 -0
  51. flock/frontend/src/components/filters/FilterBar.tsx +33 -0
  52. flock/frontend/src/components/filters/FilterPills.module.css +79 -0
  53. flock/frontend/src/components/filters/FilterPills.test.tsx +173 -0
  54. flock/frontend/src/components/filters/FilterPills.tsx +67 -0
  55. flock/frontend/src/components/filters/TimeRangeFilter.module.css +91 -0
  56. flock/frontend/src/components/filters/TimeRangeFilter.test.tsx +154 -0
  57. flock/frontend/src/components/filters/TimeRangeFilter.tsx +105 -0
  58. flock/frontend/src/components/graph/AgentNode.test.tsx +75 -0
  59. flock/frontend/src/components/graph/AgentNode.tsx +322 -0
  60. flock/frontend/src/components/graph/GraphCanvas.tsx +406 -0
  61. flock/frontend/src/components/graph/MessageFlowEdge.tsx +128 -0
  62. flock/frontend/src/components/graph/MessageNode.test.tsx +62 -0
  63. flock/frontend/src/components/graph/MessageNode.tsx +116 -0
  64. flock/frontend/src/components/graph/MiniMap.tsx +47 -0
  65. flock/frontend/src/components/graph/TransformEdge.tsx +123 -0
  66. flock/frontend/src/components/layout/DashboardLayout.css +407 -0
  67. flock/frontend/src/components/layout/DashboardLayout.tsx +300 -0
  68. flock/frontend/src/components/layout/Header.module.css +88 -0
  69. flock/frontend/src/components/layout/Header.tsx +52 -0
  70. flock/frontend/src/components/modules/EventLogModule.test.tsx +401 -0
  71. flock/frontend/src/components/modules/EventLogModule.tsx +396 -0
  72. flock/frontend/src/components/modules/EventLogModuleWrapper.tsx +17 -0
  73. flock/frontend/src/components/modules/ModuleRegistry.test.ts +333 -0
  74. flock/frontend/src/components/modules/ModuleRegistry.ts +85 -0
  75. flock/frontend/src/components/modules/ModuleWindow.tsx +155 -0
  76. flock/frontend/src/components/modules/registerModules.ts +20 -0
  77. flock/frontend/src/components/settings/AdvancedSettings.tsx +175 -0
  78. flock/frontend/src/components/settings/AppearanceSettings.tsx +185 -0
  79. flock/frontend/src/components/settings/GraphSettings.tsx +110 -0
  80. flock/frontend/src/components/settings/SettingsPanel.css +327 -0
  81. flock/frontend/src/components/settings/SettingsPanel.tsx +131 -0
  82. flock/frontend/src/components/settings/ThemeSelector.tsx +298 -0
  83. flock/frontend/src/hooks/useKeyboardShortcuts.ts +148 -0
  84. flock/frontend/src/hooks/useModulePersistence.test.ts +442 -0
  85. flock/frontend/src/hooks/useModulePersistence.ts +154 -0
  86. flock/frontend/src/hooks/useModules.ts +139 -0
  87. flock/frontend/src/hooks/usePersistence.ts +139 -0
  88. flock/frontend/src/main.tsx +13 -0
  89. flock/frontend/src/services/api.ts +213 -0
  90. flock/frontend/src/services/indexeddb.test.ts +793 -0
  91. flock/frontend/src/services/indexeddb.ts +794 -0
  92. flock/frontend/src/services/layout.test.ts +437 -0
  93. flock/frontend/src/services/layout.ts +146 -0
  94. flock/frontend/src/services/themeApplicator.ts +140 -0
  95. flock/frontend/src/services/themeService.ts +77 -0
  96. flock/frontend/src/services/websocket.test.ts +595 -0
  97. flock/frontend/src/services/websocket.ts +685 -0
  98. flock/frontend/src/store/filterStore.test.ts +242 -0
  99. flock/frontend/src/store/filterStore.ts +103 -0
  100. flock/frontend/src/store/graphStore.test.ts +186 -0
  101. flock/frontend/src/store/graphStore.ts +414 -0
  102. flock/frontend/src/store/moduleStore.test.ts +253 -0
  103. flock/frontend/src/store/moduleStore.ts +57 -0
  104. flock/frontend/src/store/settingsStore.ts +188 -0
  105. flock/frontend/src/store/streamStore.ts +68 -0
  106. flock/frontend/src/store/uiStore.test.ts +54 -0
  107. flock/frontend/src/store/uiStore.ts +110 -0
  108. flock/frontend/src/store/wsStore.ts +34 -0
  109. flock/frontend/src/styles/index.css +15 -0
  110. flock/frontend/src/styles/scrollbar.css +47 -0
  111. flock/frontend/src/styles/variables.css +488 -0
  112. flock/frontend/src/test/setup.ts +1 -0
  113. flock/frontend/src/types/filters.ts +14 -0
  114. flock/frontend/src/types/graph.ts +55 -0
  115. flock/frontend/src/types/modules.ts +7 -0
  116. flock/frontend/src/types/theme.ts +55 -0
  117. flock/frontend/src/utils/mockData.ts +85 -0
  118. flock/frontend/src/utils/performance.ts +16 -0
  119. flock/frontend/src/utils/transforms.test.ts +860 -0
  120. flock/frontend/src/utils/transforms.ts +323 -0
  121. flock/frontend/src/vite-env.d.ts +17 -0
  122. flock/frontend/tsconfig.json +27 -0
  123. flock/frontend/tsconfig.node.json +11 -0
  124. flock/frontend/vite.config.ts +25 -0
  125. flock/frontend/vitest.config.ts +11 -0
  126. flock/{core/util → helper}/cli_helper.py +4 -3
  127. flock/{core/logging → logging}/__init__.py +2 -3
  128. flock/{core/logging → logging}/formatters/enum_builder.py +3 -4
  129. flock/{core/logging → logging}/formatters/theme_builder.py +19 -44
  130. flock/{core/logging → logging}/formatters/themed_formatter.py +69 -115
  131. flock/{core/logging → logging}/logging.py +77 -61
  132. flock/{core/logging → logging}/telemetry.py +20 -26
  133. flock/{core/logging → logging}/telemetry_exporter/base_exporter.py +2 -2
  134. flock/{core/logging → logging}/telemetry_exporter/file_exporter.py +6 -9
  135. flock/{core/logging → logging}/telemetry_exporter/sqlite_exporter.py +2 -3
  136. flock/{core/logging → logging}/trace_and_logged.py +20 -24
  137. flock/mcp/__init__.py +91 -0
  138. flock/{core/mcp/mcp_client.py → mcp/client.py} +103 -154
  139. flock/{core/mcp/mcp_config.py → mcp/config.py} +62 -117
  140. flock/mcp/manager.py +255 -0
  141. flock/mcp/servers/sse/__init__.py +1 -1
  142. flock/mcp/servers/sse/flock_sse_server.py +11 -53
  143. flock/mcp/servers/stdio/__init__.py +1 -1
  144. flock/mcp/servers/stdio/flock_stdio_server.py +8 -48
  145. flock/mcp/servers/streamable_http/flock_streamable_http_server.py +17 -62
  146. flock/mcp/servers/websockets/flock_websocket_server.py +7 -40
  147. flock/{core/mcp/flock_mcp_tool.py → mcp/tool.py} +16 -26
  148. flock/mcp/types/__init__.py +42 -0
  149. flock/{core/mcp → mcp}/types/callbacks.py +9 -15
  150. flock/{core/mcp → mcp}/types/factories.py +7 -6
  151. flock/{core/mcp → mcp}/types/handlers.py +13 -18
  152. flock/{core/mcp → mcp}/types/types.py +70 -74
  153. flock/{core/mcp → mcp}/util/helpers.py +1 -1
  154. flock/orchestrator.py +645 -0
  155. flock/registry.py +148 -0
  156. flock/runtime.py +262 -0
  157. flock/service.py +140 -0
  158. flock/store.py +69 -0
  159. flock/subscription.py +111 -0
  160. flock/themes/andromeda.toml +1 -1
  161. flock/themes/apple-system-colors.toml +1 -1
  162. flock/themes/arcoiris.toml +1 -1
  163. flock/themes/atomonelight.toml +1 -1
  164. flock/themes/ayu copy.toml +1 -1
  165. flock/themes/ayu-light.toml +1 -1
  166. flock/themes/belafonte-day.toml +1 -1
  167. flock/themes/belafonte-night.toml +1 -1
  168. flock/themes/blulocodark.toml +1 -1
  169. flock/themes/breeze.toml +1 -1
  170. flock/themes/broadcast.toml +1 -1
  171. flock/themes/brogrammer.toml +1 -1
  172. flock/themes/builtin-dark.toml +1 -1
  173. flock/themes/builtin-pastel-dark.toml +1 -1
  174. flock/themes/catppuccin-latte.toml +1 -1
  175. flock/themes/catppuccin-macchiato.toml +1 -1
  176. flock/themes/catppuccin-mocha.toml +1 -1
  177. flock/themes/cga.toml +1 -1
  178. flock/themes/chalk.toml +1 -1
  179. flock/themes/ciapre.toml +1 -1
  180. flock/themes/coffee-theme.toml +1 -1
  181. flock/themes/cyberpunkscarletprotocol.toml +1 -1
  182. flock/themes/dark+.toml +1 -1
  183. flock/themes/darkermatrix.toml +1 -1
  184. flock/themes/darkside.toml +1 -1
  185. flock/themes/desert.toml +1 -1
  186. flock/themes/django.toml +1 -1
  187. flock/themes/djangosmooth.toml +1 -1
  188. flock/themes/doomone.toml +1 -1
  189. flock/themes/dotgov.toml +1 -1
  190. flock/themes/dracula+.toml +1 -1
  191. flock/themes/duckbones.toml +1 -1
  192. flock/themes/encom.toml +1 -1
  193. flock/themes/espresso.toml +1 -1
  194. flock/themes/everblush.toml +1 -1
  195. flock/themes/fairyfloss.toml +1 -1
  196. flock/themes/fideloper.toml +1 -1
  197. flock/themes/fishtank.toml +1 -1
  198. flock/themes/flexoki-light.toml +1 -1
  199. flock/themes/floraverse.toml +1 -1
  200. flock/themes/framer.toml +1 -1
  201. flock/themes/galizur.toml +1 -1
  202. flock/themes/github.toml +1 -1
  203. flock/themes/grass.toml +1 -1
  204. flock/themes/grey-green.toml +1 -1
  205. flock/themes/gruvboxlight.toml +1 -1
  206. flock/themes/guezwhoz.toml +1 -1
  207. flock/themes/harper.toml +1 -1
  208. flock/themes/hax0r-blue.toml +1 -1
  209. flock/themes/hopscotch.256.toml +1 -1
  210. flock/themes/ic-green-ppl.toml +1 -1
  211. flock/themes/iceberg-dark.toml +1 -1
  212. flock/themes/japanesque.toml +1 -1
  213. flock/themes/jubi.toml +1 -1
  214. flock/themes/kibble.toml +1 -1
  215. flock/themes/kolorit.toml +1 -1
  216. flock/themes/kurokula.toml +1 -1
  217. flock/themes/materialdesigncolors.toml +1 -1
  218. flock/themes/matrix.toml +1 -1
  219. flock/themes/mellifluous.toml +1 -1
  220. flock/themes/midnight-in-mojave.toml +1 -1
  221. flock/themes/monokai-remastered.toml +1 -1
  222. flock/themes/monokai-soda.toml +1 -1
  223. flock/themes/neon.toml +1 -1
  224. flock/themes/neopolitan.toml +1 -1
  225. flock/themes/nord-light.toml +1 -1
  226. flock/themes/ocean.toml +1 -1
  227. flock/themes/onehalfdark.toml +1 -1
  228. flock/themes/onehalflight.toml +1 -1
  229. flock/themes/palenighthc.toml +1 -1
  230. flock/themes/paulmillr.toml +1 -1
  231. flock/themes/pencildark.toml +1 -1
  232. flock/themes/pnevma.toml +1 -1
  233. flock/themes/purple-rain.toml +1 -1
  234. flock/themes/purplepeter.toml +1 -1
  235. flock/themes/raycast-dark.toml +1 -1
  236. flock/themes/red-sands.toml +1 -1
  237. flock/themes/relaxed.toml +1 -1
  238. flock/themes/retro.toml +1 -1
  239. flock/themes/rose-pine.toml +1 -1
  240. flock/themes/royal.toml +1 -1
  241. flock/themes/ryuuko.toml +1 -1
  242. flock/themes/sakura.toml +1 -1
  243. flock/themes/scarlet-protocol.toml +1 -1
  244. flock/themes/seoulbones-dark.toml +1 -1
  245. flock/themes/shades-of-purple.toml +1 -1
  246. flock/themes/smyck.toml +1 -1
  247. flock/themes/softserver.toml +1 -1
  248. flock/themes/solarized-darcula.toml +1 -1
  249. flock/themes/square.toml +1 -1
  250. flock/themes/sugarplum.toml +1 -1
  251. flock/themes/thayer-bright.toml +1 -1
  252. flock/themes/tokyonight.toml +1 -1
  253. flock/themes/tomorrow.toml +1 -1
  254. flock/themes/ubuntu.toml +1 -1
  255. flock/themes/ultradark.toml +1 -1
  256. flock/themes/ultraviolent.toml +1 -1
  257. flock/themes/unikitty.toml +1 -1
  258. flock/themes/urple.toml +1 -1
  259. flock/themes/vesper.toml +1 -1
  260. flock/themes/vimbones.toml +1 -1
  261. flock/themes/wildcherry.toml +1 -1
  262. flock/themes/wilmersdorf.toml +1 -1
  263. flock/themes/wryan.toml +1 -1
  264. flock/themes/xcodedarkhc.toml +1 -1
  265. flock/themes/xcodelight.toml +1 -1
  266. flock/themes/zenbones-light.toml +1 -1
  267. flock/themes/zenwritten-dark.toml +1 -1
  268. flock/utilities.py +301 -0
  269. flock/{components/utility → utility}/output_utility_component.py +68 -53
  270. flock/visibility.py +107 -0
  271. flock_core-0.5.0b51.dist-info/METADATA +747 -0
  272. flock_core-0.5.0b51.dist-info/RECORD +508 -0
  273. flock_core-0.5.0b51.dist-info/entry_points.txt +2 -0
  274. {flock_core-0.5.0b28.dist-info → flock_core-0.5.0b51.dist-info}/licenses/LICENSE +1 -1
  275. flock/adapter/__init__.py +0 -14
  276. flock/adapter/azure_adapter.py +0 -68
  277. flock/adapter/chroma_adapter.py +0 -73
  278. flock/adapter/faiss_adapter.py +0 -97
  279. flock/adapter/pinecone_adapter.py +0 -51
  280. flock/adapter/vector_base.py +0 -47
  281. flock/cli/assets/release_notes.md +0 -140
  282. flock/cli/config.py +0 -8
  283. flock/cli/constants.py +0 -36
  284. flock/cli/create_agent.py +0 -1
  285. flock/cli/create_flock.py +0 -280
  286. flock/cli/execute_flock.py +0 -620
  287. flock/cli/load_agent.py +0 -1
  288. flock/cli/load_examples.py +0 -1
  289. flock/cli/load_flock.py +0 -192
  290. flock/cli/load_release_notes.py +0 -20
  291. flock/cli/loaded_flock_cli.py +0 -254
  292. flock/cli/manage_agents.py +0 -459
  293. flock/cli/registry_management.py +0 -889
  294. flock/cli/runner.py +0 -41
  295. flock/cli/settings.py +0 -857
  296. flock/cli/utils.py +0 -135
  297. flock/cli/view_results.py +0 -29
  298. flock/cli/yaml_editor.py +0 -396
  299. flock/components/__init__.py +0 -30
  300. flock/components/evaluation/__init__.py +0 -9
  301. flock/components/evaluation/declarative_evaluation_component.py +0 -606
  302. flock/components/routing/__init__.py +0 -15
  303. flock/components/routing/conditional_routing_component.py +0 -494
  304. flock/components/routing/default_routing_component.py +0 -103
  305. flock/components/routing/llm_routing_component.py +0 -206
  306. flock/components/utility/__init__.py +0 -22
  307. flock/components/utility/example_utility_component.py +0 -250
  308. flock/components/utility/feedback_utility_component.py +0 -206
  309. flock/components/utility/memory_utility_component.py +0 -550
  310. flock/components/utility/metrics_utility_component.py +0 -700
  311. flock/config.py +0 -61
  312. flock/core/__init__.py +0 -110
  313. flock/core/agent/__init__.py +0 -16
  314. flock/core/agent/default_agent.py +0 -216
  315. flock/core/agent/flock_agent_components.py +0 -104
  316. flock/core/agent/flock_agent_execution.py +0 -101
  317. flock/core/agent/flock_agent_integration.py +0 -260
  318. flock/core/agent/flock_agent_lifecycle.py +0 -186
  319. flock/core/agent/flock_agent_serialization.py +0 -381
  320. flock/core/api/__init__.py +0 -10
  321. flock/core/api/custom_endpoint.py +0 -45
  322. flock/core/api/endpoints.py +0 -254
  323. flock/core/api/main.py +0 -162
  324. flock/core/api/models.py +0 -97
  325. flock/core/api/run_store.py +0 -224
  326. flock/core/api/runner.py +0 -44
  327. flock/core/api/service.py +0 -214
  328. flock/core/component/__init__.py +0 -15
  329. flock/core/component/agent_component_base.py +0 -309
  330. flock/core/component/evaluation_component.py +0 -62
  331. flock/core/component/routing_component.py +0 -74
  332. flock/core/component/utility_component.py +0 -69
  333. flock/core/config/flock_agent_config.py +0 -58
  334. flock/core/config/scheduled_agent_config.py +0 -40
  335. flock/core/context/context.py +0 -213
  336. flock/core/context/context_manager.py +0 -37
  337. flock/core/context/context_vars.py +0 -10
  338. flock/core/evaluation/utils.py +0 -396
  339. flock/core/execution/batch_executor.py +0 -369
  340. flock/core/execution/evaluation_executor.py +0 -438
  341. flock/core/execution/local_executor.py +0 -31
  342. flock/core/execution/opik_executor.py +0 -103
  343. flock/core/execution/temporal_executor.py +0 -164
  344. flock/core/flock.py +0 -634
  345. flock/core/flock_agent.py +0 -336
  346. flock/core/flock_factory.py +0 -613
  347. flock/core/flock_scheduler.py +0 -166
  348. flock/core/flock_server_manager.py +0 -136
  349. flock/core/interpreter/python_interpreter.py +0 -689
  350. flock/core/mcp/__init__.py +0 -1
  351. flock/core/mcp/flock_mcp_server.py +0 -680
  352. flock/core/mcp/mcp_client_manager.py +0 -201
  353. flock/core/mcp/types/__init__.py +0 -1
  354. flock/core/mixin/dspy_integration.py +0 -403
  355. flock/core/mixin/prompt_parser.py +0 -125
  356. flock/core/orchestration/__init__.py +0 -15
  357. flock/core/orchestration/flock_batch_processor.py +0 -94
  358. flock/core/orchestration/flock_evaluator.py +0 -113
  359. flock/core/orchestration/flock_execution.py +0 -295
  360. flock/core/orchestration/flock_initialization.py +0 -149
  361. flock/core/orchestration/flock_server_manager.py +0 -67
  362. flock/core/orchestration/flock_web_server.py +0 -117
  363. flock/core/registry/__init__.py +0 -45
  364. flock/core/registry/agent_registry.py +0 -69
  365. flock/core/registry/callable_registry.py +0 -139
  366. flock/core/registry/component_discovery.py +0 -142
  367. flock/core/registry/component_registry.py +0 -64
  368. flock/core/registry/config_mapping.py +0 -64
  369. flock/core/registry/decorators.py +0 -137
  370. flock/core/registry/registry_hub.py +0 -205
  371. flock/core/registry/server_registry.py +0 -57
  372. flock/core/registry/type_registry.py +0 -86
  373. flock/core/serialization/__init__.py +0 -13
  374. flock/core/serialization/callable_registry.py +0 -52
  375. flock/core/serialization/flock_serializer.py +0 -832
  376. flock/core/serialization/json_encoder.py +0 -41
  377. flock/core/serialization/secure_serializer.py +0 -175
  378. flock/core/serialization/serializable.py +0 -342
  379. flock/core/serialization/serialization_utils.py +0 -412
  380. flock/core/util/file_path_utils.py +0 -223
  381. flock/core/util/hydrator.py +0 -309
  382. flock/core/util/input_resolver.py +0 -164
  383. flock/core/util/loader.py +0 -59
  384. flock/core/util/splitter.py +0 -219
  385. flock/di.py +0 -27
  386. flock/platform/docker_tools.py +0 -49
  387. flock/platform/jaeger_install.py +0 -86
  388. flock/webapp/__init__.py +0 -1
  389. flock/webapp/app/__init__.py +0 -0
  390. flock/webapp/app/api/__init__.py +0 -0
  391. flock/webapp/app/api/agent_management.py +0 -241
  392. flock/webapp/app/api/execution.py +0 -709
  393. flock/webapp/app/api/flock_management.py +0 -129
  394. flock/webapp/app/api/registry_viewer.py +0 -30
  395. flock/webapp/app/chat.py +0 -665
  396. flock/webapp/app/config.py +0 -104
  397. flock/webapp/app/dependencies.py +0 -117
  398. flock/webapp/app/main.py +0 -1070
  399. flock/webapp/app/middleware.py +0 -113
  400. flock/webapp/app/models_ui.py +0 -7
  401. flock/webapp/app/services/__init__.py +0 -0
  402. flock/webapp/app/services/feedback_file_service.py +0 -363
  403. flock/webapp/app/services/flock_service.py +0 -337
  404. flock/webapp/app/services/sharing_models.py +0 -81
  405. flock/webapp/app/services/sharing_store.py +0 -762
  406. flock/webapp/app/templates/theme_mapper.html +0 -326
  407. flock/webapp/app/theme_mapper.py +0 -812
  408. flock/webapp/app/utils.py +0 -85
  409. flock/webapp/run.py +0 -215
  410. flock/webapp/static/css/chat.css +0 -301
  411. flock/webapp/static/css/components.css +0 -167
  412. flock/webapp/static/css/header.css +0 -39
  413. flock/webapp/static/css/layout.css +0 -46
  414. flock/webapp/static/css/sidebar.css +0 -127
  415. flock/webapp/static/css/two-pane.css +0 -48
  416. flock/webapp/templates/base.html +0 -200
  417. flock/webapp/templates/chat.html +0 -152
  418. flock/webapp/templates/chat_settings.html +0 -19
  419. flock/webapp/templates/flock_editor.html +0 -16
  420. flock/webapp/templates/index.html +0 -12
  421. flock/webapp/templates/partials/_agent_detail_form.html +0 -93
  422. flock/webapp/templates/partials/_agent_list.html +0 -18
  423. flock/webapp/templates/partials/_agent_manager_view.html +0 -51
  424. flock/webapp/templates/partials/_agent_tools_checklist.html +0 -14
  425. flock/webapp/templates/partials/_chat_container.html +0 -15
  426. flock/webapp/templates/partials/_chat_messages.html +0 -57
  427. flock/webapp/templates/partials/_chat_settings_form.html +0 -85
  428. flock/webapp/templates/partials/_create_flock_form.html +0 -50
  429. flock/webapp/templates/partials/_dashboard_flock_detail.html +0 -17
  430. flock/webapp/templates/partials/_dashboard_flock_file_list.html +0 -16
  431. flock/webapp/templates/partials/_dashboard_flock_properties_preview.html +0 -28
  432. flock/webapp/templates/partials/_dashboard_upload_flock_form.html +0 -16
  433. flock/webapp/templates/partials/_dynamic_input_form_content.html +0 -22
  434. flock/webapp/templates/partials/_env_vars_table.html +0 -23
  435. flock/webapp/templates/partials/_execution_form.html +0 -118
  436. flock/webapp/templates/partials/_execution_view_container.html +0 -28
  437. flock/webapp/templates/partials/_flock_file_list.html +0 -23
  438. flock/webapp/templates/partials/_flock_properties_form.html +0 -52
  439. flock/webapp/templates/partials/_flock_upload_form.html +0 -16
  440. flock/webapp/templates/partials/_header_flock_status.html +0 -5
  441. flock/webapp/templates/partials/_load_manager_view.html +0 -49
  442. flock/webapp/templates/partials/_registry_table.html +0 -25
  443. flock/webapp/templates/partials/_registry_viewer_content.html +0 -70
  444. flock/webapp/templates/partials/_results_display.html +0 -78
  445. flock/webapp/templates/partials/_settings_env_content.html +0 -9
  446. flock/webapp/templates/partials/_settings_theme_content.html +0 -14
  447. flock/webapp/templates/partials/_settings_view.html +0 -36
  448. flock/webapp/templates/partials/_share_chat_link_snippet.html +0 -11
  449. flock/webapp/templates/partials/_share_link_snippet.html +0 -35
  450. flock/webapp/templates/partials/_sidebar.html +0 -74
  451. flock/webapp/templates/partials/_streaming_results_container.html +0 -195
  452. flock/webapp/templates/partials/_structured_data_view.html +0 -40
  453. flock/webapp/templates/partials/_theme_preview.html +0 -36
  454. flock/webapp/templates/registry_viewer.html +0 -84
  455. flock/webapp/templates/shared_run_page.html +0 -140
  456. flock/workflow/__init__.py +0 -0
  457. flock/workflow/activities.py +0 -196
  458. flock/workflow/agent_activities.py +0 -24
  459. flock/workflow/agent_execution_activity.py +0 -202
  460. flock/workflow/flock_workflow.py +0 -214
  461. flock/workflow/temporal_config.py +0 -96
  462. flock/workflow/temporal_setup.py +0 -68
  463. flock_core-0.5.0b28.dist-info/METADATA +0 -274
  464. flock_core-0.5.0b28.dist-info/RECORD +0 -561
  465. flock_core-0.5.0b28.dist-info/entry_points.txt +0 -2
  466. /flock/{core/logging → logging}/formatters/themes.py +0 -0
  467. /flock/{core/logging → logging}/span_middleware/baggage_span_processor.py +0 -0
  468. /flock/{core/mcp → mcp}/util/__init__.py +0 -0
  469. {flock_core-0.5.0b28.dist-info → flock_core-0.5.0b51.dist-info}/WHEEL +0 -0
File without changes
@@ -1,196 +0,0 @@
1
- """Defines Temporal activities for running a chain of agents with logging and tracing."""
2
-
3
- from datetime import datetime
4
-
5
- from opentelemetry import trace
6
- from temporalio import activity
7
-
8
- from flock.core.context.context import FlockContext
9
- from flock.core.context.context_vars import FLOCK_CURRENT_AGENT, FLOCK_MODEL
10
-
11
- # HandOffRequest removed - using agent.next_agent directly
12
- from flock.core.logging.logging import get_logger
13
- from flock.core.registry import get_registry
14
- from flock.core.util.input_resolver import resolve_inputs
15
-
16
- logger = get_logger("activities")
17
- tracer = trace.get_tracer(__name__)
18
-
19
- def apply_handoff_strategy(previous_agent_output:str, next_agent_input:str, previous_agent_handoff_strategy:str, previous_agent_handoff_map:dict[str, str]) -> str:
20
- if previous_agent_handoff_strategy == "append":
21
- return next_agent_input + previous_agent_output
22
- elif previous_agent_handoff_strategy == "override":
23
- return previous_agent_output
24
- elif previous_agent_handoff_strategy == "static":
25
- return next_agent_input
26
- elif previous_agent_handoff_strategy == "map":
27
- for key, value in previous_agent_handoff_map.items():
28
- next_agent_input = next_agent_input.replace(key, value)
29
- return next_agent_input
30
- raise NotImplementedError
31
-
32
-
33
- @activity.defn
34
- async def run_agent(context: FlockContext) -> dict:
35
- """Runs a chain of agents using the provided context.
36
-
37
- The context contains state, history, and agent definitions.
38
- After each agent run, its output is merged into the context.
39
- """
40
- # Start a top-level span for the entire run_agent activity.
41
- with tracer.start_as_current_span("run_agent") as span:
42
- registry = get_registry()
43
-
44
- previous_agent_name = ""
45
- previous_agent_output = ""
46
- previous_agent_handoff_strategy = ""
47
- previous_agent_handoff_map = {}
48
- if isinstance(context, dict):
49
- context = FlockContext.from_dict(context)
50
- current_agent_name = context.get_variable(FLOCK_CURRENT_AGENT)
51
- span.set_attribute("initial.agent", current_agent_name)
52
- logger.info("Starting agent chain", initial_agent=current_agent_name)
53
-
54
- agent = registry.get_agent(current_agent_name)
55
- if agent.model is None or agent.evaluator.config.model is None:
56
- agent.set_model(context.get_variable(FLOCK_MODEL))
57
-
58
- if not agent:
59
- logger.error("Agent not found", agent=current_agent_name)
60
- span.record_exception(
61
- Exception(f"Agent '{current_agent_name}' not found")
62
- )
63
- return {"error": f"Agent '{current_agent_name}' not found."}
64
-
65
- # Loop over agents in the chain.
66
- while agent:
67
- # Create a nested span for this iteration.
68
- with tracer.start_as_current_span("agent_iteration") as iter_span:
69
- iter_span.set_attribute("agent.name", agent.name)
70
- agent.context = context
71
- # Resolve inputs for the agent.
72
- # Gets values from context, previous agent output, and handoff strategy.
73
- agent_inputs = resolve_inputs(
74
- agent.input,
75
- context,
76
- previous_agent_name,
77
- previous_agent_output,
78
- previous_agent_handoff_strategy,
79
- previous_agent_handoff_map
80
- )
81
- iter_span.add_event(
82
- "resolved inputs", attributes={"inputs": str(agent_inputs)}
83
- )
84
-
85
- # Execute the agent with its own span.
86
- with tracer.start_as_current_span("execute_agent") as exec_span:
87
- logger.info("Executing agent", agent=agent.name)
88
- try:
89
- result = await agent.run_async(agent_inputs)
90
- exec_span.set_attribute("result", str(result))
91
- logger.debug(
92
- "Agent execution completed", agent=agent.name
93
- )
94
- context.record(
95
- agent.name,
96
- result,
97
- timestamp=datetime.now().isoformat(),
98
- hand_off=None,
99
- called_from=previous_agent_name,
100
- )
101
- except Exception as e:
102
- logger.error(
103
- "Agent execution failed",
104
- agent=agent.name,
105
- error=str(e),
106
- )
107
- exec_span.record_exception(e)
108
- raise
109
-
110
- # Determine the next agent using routing component if available
111
- next_agent_name = None
112
-
113
- if agent.router:
114
- logger.info(
115
- f"Using router: {agent.router.__class__.__name__}",
116
- agent=agent.name,
117
- )
118
- try:
119
- # Route to the next agent using new routing component
120
- next_agent_name = await agent.router.determine_next_step(
121
- agent, result, context
122
- )
123
-
124
- # Set next_agent on the agent instance
125
- agent.next_agent = next_agent_name
126
-
127
- except Exception as e:
128
- logger.error(
129
- f"Router error: {e}",
130
- agent=agent.name,
131
- error=str(e),
132
- )
133
- iter_span.record_exception(e)
134
- return {"error": f"Router error: {e}"}
135
- else:
136
- # Check if next_agent was set directly by user
137
- next_agent_name = agent.next_agent
138
- if callable(next_agent_name):
139
- try:
140
- next_agent_name = next_agent_name(context, result)
141
- except Exception as e:
142
- logger.error(f"next_agent callable error: {e}")
143
- return {"error": f"next_agent callable error: {e}"}
144
-
145
- if not next_agent_name:
146
- logger.info(
147
- "No next agent found, completing chain",
148
- agent=agent.name,
149
- )
150
- iter_span.add_event("chain completed")
151
- return result
152
-
153
- # Record the agent run in the context.
154
- context.record(
155
- agent.name,
156
- result,
157
- timestamp=datetime.now().isoformat(),
158
- hand_off={"next_agent": next_agent_name} if next_agent_name else None,
159
- called_from=previous_agent_name,
160
- )
161
- # Remember the current agent's details for the next iteration.
162
- previous_agent_name = agent.name
163
- previous_agent_output = agent.output
164
- previous_agent_handoff_strategy = agent.config.handoff_strategy
165
- previous_agent_handoff_map = agent.config.handoff_map
166
-
167
- # Activate the next agent.
168
- try:
169
- agent = registry.get_agent(next_agent_name)
170
- if not agent:
171
- logger.error(
172
- "Next agent not found",
173
- agent=next_agent_name,
174
- )
175
- iter_span.record_exception(
176
- Exception(
177
- f"Next agent '{next_agent_name}' not found"
178
- )
179
- )
180
- return {
181
- "error": f"Next agent '{next_agent_name}' not found."
182
- }
183
-
184
-
185
-
186
- context.set_variable(FLOCK_CURRENT_AGENT, agent.name)
187
-
188
- logger.info("Handing off to next agent", next=agent.name)
189
- iter_span.set_attribute("next.agent", agent.name)
190
- except Exception as e:
191
- logger.error("Error during handoff", error=str(e))
192
- iter_span.record_exception(e)
193
- return {"error": f"Error during handoff: {e}"}
194
-
195
- # If the loop exits unexpectedly, return the initial input.
196
- return context.get_variable("init_input")
@@ -1,24 +0,0 @@
1
- from temporalio import activity
2
-
3
- from flock.core.context.context import FlockContext
4
- from flock.core.flock_agent import FlockAgent
5
-
6
-
7
- @activity.defn
8
- async def run_declarative_agent_activity(params: dict) -> dict:
9
- """Temporal activity to run a declarative (or batch) agent.
10
-
11
- Expects a dictionary with:
12
- - "agent_data": a dict representation of the agent (as produced by .dict()),
13
- - "context_data": a dict containing the FlockContext state and optionally other fields.
14
-
15
- The activity reconstructs the agent and a FlockContext, then calls the agent’s _evaluate() method.
16
- """
17
- agent_data = params.get("agent_data")
18
- context_data = params.get("context_data", {})
19
- # Reconstruct the agent from its serialized representation.
20
- agent = FlockAgent.from_dict(agent_data)
21
- # Reconstruct the FlockContext from the state.
22
- context = FlockContext.from_dict(context_data)
23
- result = await agent.evaluate(context)
24
- return result
@@ -1,202 +0,0 @@
1
- """Defines granular Temporal activities for executing a single agent
2
- and determining the next agent in a Flock workflow.
3
- """
4
-
5
- from collections.abc import Callable
6
-
7
- from opentelemetry import trace
8
- from temporalio import activity
9
-
10
- # Third-party imports only within activity functions if needed, or pass context
11
- # For core flock types, import directly
12
- from flock.core.context.context import FlockContext
13
- from flock.core.context.context_vars import FLOCK_MODEL
14
- from flock.core.flock_agent import FlockAgent # Import concrete class if needed
15
- from flock.core.registry import get_registry
16
- # HandOffRequest removed - using agent.next_agent directly
17
- from flock.core.logging.logging import get_logger
18
- from flock.core.util.input_resolver import resolve_inputs
19
-
20
- logger = get_logger("agent_activity") # Using a distinct logger category
21
- tracer = trace.get_tracer(__name__)
22
- registry = get_registry() # Get registry instance once
23
-
24
-
25
- @activity.defn
26
- async def execute_single_agent(agent_name: str, context_dict: dict) -> dict:
27
- """Executes a single specified agent and returns its result.
28
-
29
- Args:
30
- agent_name: The name of the agent to execute.
31
- context: The current FlockContext (passed from the workflow).
32
-
33
- Returns:
34
- The raw result dictionary from the agent's execution.
35
-
36
- Raises:
37
- ValueError: If the agent is not found in the registry.
38
- Exception: Propagates exceptions from agent execution for Temporal retries.
39
- """
40
- with tracer.start_as_current_span("execute_single_agent") as span:
41
- span.set_attribute("agent.name", agent_name)
42
- logger.info("Executing single agent", agent=agent_name)
43
-
44
- agent = registry.get_agent(agent_name)
45
- if not agent:
46
- logger.error("Agent not found in registry", agent=agent_name)
47
- # Raise error for Temporal to potentially retry/fail the activity
48
- raise ValueError(f"Agent '{agent_name}' not found in registry.")
49
-
50
- # Rehydrate context from dict and set on agent (transient for this execution)
51
- context = FlockContext.from_dict(context_dict)
52
- agent.context = context
53
-
54
- # Ensure model is set (using context value if needed)
55
- # Consider if this should be done once when agent is added or workflow starts
56
- if agent.model is None:
57
- agent_model = context.get_variable(FLOCK_MODEL)
58
- if agent_model:
59
- agent.set_model(agent_model)
60
- logger.debug(
61
- f"Set model for agent '{agent_name}' from context: {agent_model}"
62
- )
63
-
64
- # Resolve agent-specific callables if necessary
65
- # This might be better handled in the workflow before the loop starts
66
- # or when agents are initially loaded. Assuming it's handled elsewhere for now.
67
- # agent.resolve_callables(context=context)
68
-
69
- # Resolve inputs for this specific agent run
70
- previous_agent_name = context.get_last_agent_name() # May be None on first agent
71
- prev_def = (
72
- context.get_agent_definition(previous_agent_name)
73
- if previous_agent_name
74
- else None
75
- )
76
- prev_out_spec = (
77
- (prev_def.agent_data.get("output_spec") if isinstance(prev_def, type(prev_def)) else None)
78
- if prev_def and isinstance(prev_def.agent_data, dict)
79
- else None
80
- )
81
- prev_cfg = (
82
- prev_def.agent_data.get("config")
83
- if prev_def and isinstance(prev_def.agent_data, dict)
84
- else {}
85
- )
86
- prev_strategy = (
87
- prev_cfg.get("handoff_strategy") if isinstance(prev_cfg, dict) else None
88
- ) or "static"
89
- prev_map = (
90
- prev_cfg.get("handoff_map") if isinstance(prev_cfg, dict) else None
91
- ) or {}
92
- logger.debug(
93
- f"Resolving inputs for {agent_name} with previous agent {previous_agent_name}"
94
- )
95
- agent_inputs = resolve_inputs(
96
- agent.input,
97
- context,
98
- previous_agent_name or "",
99
- prev_out_spec or "",
100
- prev_strategy,
101
- prev_map,
102
- )
103
- span.add_event(
104
- "resolved inputs", attributes={"inputs": str(agent_inputs)}
105
- )
106
-
107
- try:
108
- # Execute just this agent
109
- result = await agent.run_async(agent_inputs)
110
- # Avoid logging potentially large results directly to span attributes
111
- result_str = str(result)
112
- span.set_attribute("result.type", type(result).__name__)
113
- span.set_attribute(
114
- "result.preview",
115
- result_str[:500] + ("..." if len(result_str) > 500 else ""),
116
- )
117
- logger.info("Single agent execution completed", agent=agent_name)
118
- return result
119
- except Exception as e:
120
- logger.error(
121
- "Single agent execution failed",
122
- agent=agent_name,
123
- error=str(e),
124
- exc_info=True,
125
- )
126
- # Debug aid: ensure exception prints even if logger is muted in environment
127
- print(f"[agent_activity] Single agent execution failed for {agent_name}: {e!r}")
128
- span.record_exception(e)
129
- # Re-raise the exception for Temporal to handle based on retry policy
130
- raise
131
-
132
-
133
- @activity.defn
134
- async def determine_next_agent(
135
- current_agent_name: str, result: dict, context_dict: dict
136
- ) -> str | None:
137
- """Determine the next agent using the agent's routing component.
138
-
139
- Returns the next agent's name or None if the workflow should terminate.
140
- """
141
- with tracer.start_as_current_span("determine_next_agent") as span:
142
- span.set_attribute("agent.name", current_agent_name)
143
- logger.info("Determining next agent after", agent=current_agent_name)
144
-
145
- agent = registry.get_agent(current_agent_name)
146
- if not agent:
147
- logger.error(
148
- "Agent not found for routing", agent=current_agent_name
149
- )
150
- raise ValueError(
151
- f"Agent '{current_agent_name}' not found for routing."
152
- )
153
-
154
- if not agent.router:
155
- logger.info(
156
- "No router defined for agent", agent=current_agent_name
157
- )
158
- span.add_event("no_router")
159
- return None # Indicate no handoff
160
-
161
- logger.debug(
162
- f"Using router {agent.router.__class__.__name__}",
163
- agent=agent.name,
164
- )
165
- try:
166
- # Execute routing logic on the router component (unified architecture)
167
- context = FlockContext.from_dict(context_dict)
168
- next_val = await agent.router.determine_next_step(agent, result, context)
169
-
170
- # Convert to a simple agent name if needed
171
- if isinstance(next_val, FlockAgent):
172
- next_name = next_val.name
173
- elif isinstance(next_val, str):
174
- next_name = next_val
175
- else:
176
- next_name = None
177
-
178
- if not next_name:
179
- logger.info("Router determined no next agent", agent=agent.name)
180
- span.add_event("no_next_agent_from_router")
181
- return None
182
-
183
- logger.info(
184
- "Next agent determined",
185
- next_agent=next_name,
186
- agent=agent.name,
187
- )
188
- span.set_attribute("next_agent", next_name)
189
- return next_name
190
-
191
- except Exception as e:
192
- # Catch potential errors during routing execution
193
- logger.error(
194
- "Router execution failed",
195
- agent=agent.name,
196
- error=str(e),
197
- exc_info=True,
198
- )
199
- print(f"[agent_activity] Router execution failed for {agent.name}: {e!r}")
200
- span.record_exception(e)
201
- # Let Temporal handle the activity failure based on retry policy
202
- raise
@@ -1,214 +0,0 @@
1
- from datetime import timedelta
2
- from typing import Any
3
-
4
- from temporalio import workflow
5
-
6
- # Import activities from the new file
7
- with workflow.unsafe.imports_passed_through():
8
- from flock.core.context.context import AgentDefinition, FlockContext
9
- from flock.core.context.context_vars import FLOCK_CURRENT_AGENT
10
- # HandOffRequest removed - using agent.next_agent directly
11
- from flock.core.logging.logging import get_logger
12
- from flock.workflow.agent_execution_activity import (
13
- determine_next_agent,
14
- execute_single_agent,
15
- )
16
- from flock.workflow.temporal_config import (
17
- TemporalActivityConfig,
18
- TemporalRetryPolicyConfig,
19
- )
20
-
21
-
22
- logger = get_logger("workflow")
23
-
24
-
25
- @workflow.defn
26
- class FlockWorkflow:
27
- # No need for __init__ storing context anymore if passed to run
28
-
29
- @workflow.run
30
- async def run(self, workflow_args: dict[str, Any]) -> dict:
31
- # --- Workflow Initialization ---
32
- # Arguments are packed into a single dictionary
33
- context_dict = workflow_args["context_dict"]
34
- default_retry_config_dict = workflow_args["default_retry_config_dict"]
35
-
36
- # Deserialize context and default retry config
37
- context = FlockContext.from_dict(context_dict)
38
- default_retry_config = TemporalRetryPolicyConfig.model_validate(
39
- default_retry_config_dict
40
- )
41
-
42
- context.workflow_id = workflow.info().workflow_id
43
- context.workflow_timestamp = workflow.info().start_time.strftime(
44
- "%Y-%m-%d %H:%M:%S"
45
- )
46
-
47
- current_agent_name = context.get_variable(FLOCK_CURRENT_AGENT)
48
- final_result = None
49
- previous_agent_name = (
50
- None # Keep track of the agent that called the current one
51
- )
52
-
53
- logger.info(
54
- "Starting workflow execution",
55
- workflow_id=context.workflow_id,
56
- start_time=context.workflow_timestamp,
57
- initial_agent=current_agent_name,
58
- )
59
-
60
- try:
61
- while current_agent_name:
62
- logger.info(
63
- "Executing agent activity", agent=current_agent_name
64
- )
65
-
66
- # --- Determine Activity Settings ---
67
- agent_def: AgentDefinition | None = (
68
- context.get_agent_definition(current_agent_name)
69
- )
70
- agent_activity_config: TemporalActivityConfig | None = None
71
- final_retry_config = (
72
- default_retry_config # Start with the workflow default
73
- )
74
-
75
- if agent_def and agent_def.agent_data.get(
76
- "temporal_activity_config"
77
- ):
78
- try:
79
- agent_activity_config = (
80
- TemporalActivityConfig.model_validate(
81
- agent_def.agent_data["temporal_activity_config"]
82
- )
83
- )
84
- logger.debug(
85
- f"Loaded agent-specific temporal config for {current_agent_name}"
86
- )
87
- except Exception as e:
88
- logger.warn(
89
- f"Failed to validate agent temporal config for {current_agent_name}: {e}. Using defaults."
90
- )
91
-
92
- # Layering logic: Agent config overrides workflow default config
93
- activity_task_queue = (
94
- workflow.info().task_queue
95
- ) # Default to workflow task queue
96
- activity_timeout = timedelta(
97
- minutes=5
98
- ) # Fallback default timeout
99
-
100
- if agent_activity_config:
101
- activity_task_queue = (
102
- agent_activity_config.task_queue or activity_task_queue
103
- )
104
- activity_timeout = (
105
- agent_activity_config.start_to_close_timeout
106
- or activity_timeout
107
- )
108
- if agent_activity_config.retry_policy:
109
- final_retry_config = agent_activity_config.retry_policy
110
-
111
- # Convert config to actual Temporal object
112
- final_retry_policy = final_retry_config.to_temporalio_policy()
113
-
114
- logger.debug(
115
- f"Final activity settings for {current_agent_name}: "
116
- f"queue='{activity_task_queue}', timeout={activity_timeout}, "
117
- f"retries={final_retry_policy.maximum_attempts}"
118
- )
119
-
120
- # --- Execute the current agent activity ---
121
- try:
122
- agent_result = await workflow.execute_activity(
123
- execute_single_agent,
124
- args=[current_agent_name, context.model_dump(mode="json")],
125
- task_queue=activity_task_queue,
126
- start_to_close_timeout=activity_timeout,
127
- retry_policy=final_retry_policy,
128
- )
129
- except Exception as e:
130
- logger.error("execute_single_agent activity failed", agent=current_agent_name, error=str(e))
131
- raise
132
-
133
- # Record the execution in the context history
134
- # Note: The 'called_from' is the agent *before* this one
135
- context.record(
136
- agent_name=current_agent_name,
137
- data=agent_result,
138
- timestamp=workflow.now().isoformat(), # Use deterministic workflow time
139
- hand_off=None, # Will be updated if handoff occurs
140
- called_from=previous_agent_name, # Pass the correct previous agent
141
- )
142
-
143
- final_result = agent_result # Store the result of the last successful agent
144
-
145
- logger.info(
146
- "Determining next agent activity",
147
- current_agent=current_agent_name,
148
- )
149
- # --- Determine the next agent (using routing component) ---
150
- try:
151
- next_agent_name = await workflow.execute_activity(
152
- determine_next_agent,
153
- args=[current_agent_name, agent_result, context.model_dump(mode="json")],
154
- start_to_close_timeout=timedelta(minutes=1),
155
- retry_policy=default_retry_config.to_temporalio_policy(),
156
- )
157
- except Exception as e:
158
- logger.error("determine_next_agent activity failed", agent=current_agent_name, error=str(e))
159
- raise
160
-
161
- # Update previous agent name for the next loop iteration
162
- previous_agent_name = current_agent_name
163
-
164
- if next_agent_name:
165
- logger.debug("Next agent received", data=next_agent_name)
166
- # Update the last record's handoff information for observability
167
- if context.history:
168
- context.history[-1].hand_off = {"next_agent": next_agent_name}
169
-
170
- # Set the next agent
171
- current_agent_name = next_agent_name
172
- context.set_variable(FLOCK_CURRENT_AGENT, current_agent_name)
173
- logger.info("Next agent set", agent=current_agent_name)
174
- else:
175
- logger.info("No next agent, workflow terminating.")
176
- current_agent_name = None
177
-
178
- # --- Workflow Completion ---
179
- logger.success(
180
- "Workflow completed successfully",
181
- final_agent=previous_agent_name,
182
- )
183
- context.set_variable(
184
- "flock.result",
185
- {
186
- "result": final_result, # Return the last agent's result
187
- "success": True,
188
- },
189
- )
190
- return final_result # Return the actual result of the last agent
191
-
192
- except Exception as e:
193
- # Catch exceptions from activities (e.g., after retries fail) or workflow logic errors
194
- logger.exception(f"Workflow execution failed: {e!r}")
195
- try:
196
- from temporalio.exceptions import ActivityError
197
- if isinstance(e, ActivityError) and getattr(e, "cause", None) is not None:
198
- logger.error(f"ActivityError cause: {e.cause!r}")
199
- except Exception:
200
- pass
201
- context.set_variable(
202
- "flock.result",
203
- {
204
- "result": f"Workflow failed: {e}",
205
- "success": False,
206
- },
207
- )
208
- # It's often better to let Temporal record the failure status
209
- # by re-raising the exception rather than returning a custom error dict.
210
- # However, returning the context might be useful for debugging.
211
- # Consider re-raising: raise
212
- return context.model_dump(
213
- mode="json"
214
- ) # Return context state on failure