flock-core 0.4.543__py3-none-any.whl → 0.5.0__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 (501) hide show
  1. flock/__init__.py +12 -217
  2. flock/agent.py +1079 -0
  3. flock/api/themes.py +71 -0
  4. flock/artifacts.py +86 -0
  5. flock/cli.py +147 -0
  6. flock/components.py +189 -0
  7. flock/dashboard/__init__.py +30 -0
  8. flock/dashboard/collector.py +559 -0
  9. flock/dashboard/events.py +188 -0
  10. flock/dashboard/graph_builder.py +563 -0
  11. flock/dashboard/launcher.py +235 -0
  12. flock/dashboard/models/graph.py +156 -0
  13. flock/dashboard/service.py +991 -0
  14. flock/dashboard/static_v2/assets/index-DFRnI_mt.js +111 -0
  15. flock/dashboard/static_v2/assets/index-fPLNdmp1.css +1 -0
  16. flock/dashboard/static_v2/index.html +13 -0
  17. flock/dashboard/websocket.py +246 -0
  18. flock/engines/__init__.py +6 -0
  19. flock/engines/dspy_engine.py +932 -0
  20. flock/examples.py +131 -0
  21. flock/frontend/README.md +778 -0
  22. flock/frontend/docs/DESIGN_SYSTEM.md +1980 -0
  23. flock/frontend/index.html +12 -0
  24. flock/frontend/package-lock.json +4337 -0
  25. flock/frontend/package.json +48 -0
  26. flock/frontend/src/App.tsx +139 -0
  27. flock/frontend/src/__tests__/integration/graph-snapshot.test.tsx +647 -0
  28. flock/frontend/src/__tests__/integration/indexeddb-persistence.test.tsx +699 -0
  29. flock/frontend/src/components/common/BuildInfo.tsx +39 -0
  30. flock/frontend/src/components/common/EmptyState.module.css +115 -0
  31. flock/frontend/src/components/common/EmptyState.tsx +128 -0
  32. flock/frontend/src/components/common/ErrorBoundary.module.css +169 -0
  33. flock/frontend/src/components/common/ErrorBoundary.tsx +118 -0
  34. flock/frontend/src/components/common/KeyboardShortcutsDialog.css +251 -0
  35. flock/frontend/src/components/common/KeyboardShortcutsDialog.tsx +151 -0
  36. flock/frontend/src/components/common/LoadingSpinner.module.css +97 -0
  37. flock/frontend/src/components/common/LoadingSpinner.tsx +29 -0
  38. flock/frontend/src/components/controls/PublishControl.css +547 -0
  39. flock/frontend/src/components/controls/PublishControl.test.tsx +543 -0
  40. flock/frontend/src/components/controls/PublishControl.tsx +432 -0
  41. flock/frontend/src/components/details/DetailWindowContainer.tsx +58 -0
  42. flock/frontend/src/components/details/LiveOutputTab.test.tsx +792 -0
  43. flock/frontend/src/components/details/LiveOutputTab.tsx +220 -0
  44. flock/frontend/src/components/details/MessageDetailWindow.tsx +439 -0
  45. flock/frontend/src/components/details/MessageHistoryTab.tsx +374 -0
  46. flock/frontend/src/components/details/NodeDetailWindow.test.tsx +501 -0
  47. flock/frontend/src/components/details/NodeDetailWindow.tsx +218 -0
  48. flock/frontend/src/components/details/RunStatusTab.tsx +348 -0
  49. flock/frontend/src/components/details/tabs.test.tsx +1015 -0
  50. flock/frontend/src/components/filters/ArtifactTypeFilter.tsx +21 -0
  51. flock/frontend/src/components/filters/CorrelationIDFilter.module.css +102 -0
  52. flock/frontend/src/components/filters/CorrelationIDFilter.test.tsx +197 -0
  53. flock/frontend/src/components/filters/CorrelationIDFilter.tsx +121 -0
  54. flock/frontend/src/components/filters/FilterFlyout.module.css +104 -0
  55. flock/frontend/src/components/filters/FilterFlyout.tsx +80 -0
  56. flock/frontend/src/components/filters/FilterPills.module.css +220 -0
  57. flock/frontend/src/components/filters/FilterPills.test.tsx +189 -0
  58. flock/frontend/src/components/filters/FilterPills.tsx +143 -0
  59. flock/frontend/src/components/filters/ProducerFilter.tsx +21 -0
  60. flock/frontend/src/components/filters/SavedFiltersControl.module.css +60 -0
  61. flock/frontend/src/components/filters/SavedFiltersControl.test.tsx +158 -0
  62. flock/frontend/src/components/filters/SavedFiltersControl.tsx +159 -0
  63. flock/frontend/src/components/filters/TagFilter.tsx +21 -0
  64. flock/frontend/src/components/filters/TimeRangeFilter.module.css +115 -0
  65. flock/frontend/src/components/filters/TimeRangeFilter.test.tsx +154 -0
  66. flock/frontend/src/components/filters/TimeRangeFilter.tsx +110 -0
  67. flock/frontend/src/components/filters/VisibilityFilter.tsx +21 -0
  68. flock/frontend/src/components/graph/AgentNode.test.tsx +77 -0
  69. flock/frontend/src/components/graph/AgentNode.tsx +324 -0
  70. flock/frontend/src/components/graph/GraphCanvas.tsx +613 -0
  71. flock/frontend/src/components/graph/MessageFlowEdge.tsx +128 -0
  72. flock/frontend/src/components/graph/MessageNode.test.tsx +64 -0
  73. flock/frontend/src/components/graph/MessageNode.tsx +129 -0
  74. flock/frontend/src/components/graph/MiniMap.tsx +47 -0
  75. flock/frontend/src/components/graph/TransformEdge.tsx +123 -0
  76. flock/frontend/src/components/layout/DashboardLayout.css +420 -0
  77. flock/frontend/src/components/layout/DashboardLayout.tsx +287 -0
  78. flock/frontend/src/components/layout/Header.module.css +88 -0
  79. flock/frontend/src/components/layout/Header.tsx +52 -0
  80. flock/frontend/src/components/modules/HistoricalArtifactsModule.module.css +288 -0
  81. flock/frontend/src/components/modules/HistoricalArtifactsModule.tsx +450 -0
  82. flock/frontend/src/components/modules/HistoricalArtifactsModuleWrapper.tsx +13 -0
  83. flock/frontend/src/components/modules/JsonAttributeRenderer.tsx +140 -0
  84. flock/frontend/src/components/modules/ModuleRegistry.test.ts +333 -0
  85. flock/frontend/src/components/modules/ModuleRegistry.ts +93 -0
  86. flock/frontend/src/components/modules/ModuleWindow.tsx +223 -0
  87. flock/frontend/src/components/modules/TraceModuleJaeger.tsx +1971 -0
  88. flock/frontend/src/components/modules/TraceModuleJaegerWrapper.tsx +13 -0
  89. flock/frontend/src/components/modules/registerModules.ts +29 -0
  90. flock/frontend/src/components/settings/AdvancedSettings.tsx +175 -0
  91. flock/frontend/src/components/settings/AppearanceSettings.tsx +185 -0
  92. flock/frontend/src/components/settings/GraphSettings.tsx +110 -0
  93. flock/frontend/src/components/settings/MultiSelect.tsx +235 -0
  94. flock/frontend/src/components/settings/SettingsPanel.css +327 -0
  95. flock/frontend/src/components/settings/SettingsPanel.tsx +131 -0
  96. flock/frontend/src/components/settings/ThemeSelector.tsx +298 -0
  97. flock/frontend/src/components/settings/TracingSettings.tsx +404 -0
  98. flock/frontend/src/hooks/useKeyboardShortcuts.ts +148 -0
  99. flock/frontend/src/hooks/useModulePersistence.test.ts +442 -0
  100. flock/frontend/src/hooks/useModulePersistence.ts +154 -0
  101. flock/frontend/src/hooks/useModules.ts +157 -0
  102. flock/frontend/src/hooks/usePersistence.ts +141 -0
  103. flock/frontend/src/main.tsx +13 -0
  104. flock/frontend/src/services/api.ts +337 -0
  105. flock/frontend/src/services/graphService.test.ts +330 -0
  106. flock/frontend/src/services/graphService.ts +75 -0
  107. flock/frontend/src/services/indexeddb.test.ts +793 -0
  108. flock/frontend/src/services/indexeddb.ts +848 -0
  109. flock/frontend/src/services/layout.test.ts +437 -0
  110. flock/frontend/src/services/layout.ts +357 -0
  111. flock/frontend/src/services/themeApplicator.ts +140 -0
  112. flock/frontend/src/services/themeService.ts +77 -0
  113. flock/frontend/src/services/websocket.ts +650 -0
  114. flock/frontend/src/store/filterStore.test.ts +250 -0
  115. flock/frontend/src/store/filterStore.ts +272 -0
  116. flock/frontend/src/store/graphStore.test.ts +570 -0
  117. flock/frontend/src/store/graphStore.ts +462 -0
  118. flock/frontend/src/store/moduleStore.test.ts +253 -0
  119. flock/frontend/src/store/moduleStore.ts +75 -0
  120. flock/frontend/src/store/settingsStore.ts +188 -0
  121. flock/frontend/src/store/streamStore.ts +68 -0
  122. flock/frontend/src/store/uiStore.test.ts +54 -0
  123. flock/frontend/src/store/uiStore.ts +122 -0
  124. flock/frontend/src/store/wsStore.ts +34 -0
  125. flock/frontend/src/styles/index.css +15 -0
  126. flock/frontend/src/styles/scrollbar.css +47 -0
  127. flock/frontend/src/styles/variables.css +488 -0
  128. flock/frontend/src/test/setup.ts +1 -0
  129. flock/frontend/src/types/filters.ts +47 -0
  130. flock/frontend/src/types/graph.ts +95 -0
  131. flock/frontend/src/types/modules.ts +10 -0
  132. flock/frontend/src/types/theme.ts +55 -0
  133. flock/frontend/src/utils/artifacts.ts +24 -0
  134. flock/frontend/src/utils/mockData.ts +98 -0
  135. flock/frontend/src/utils/performance.ts +16 -0
  136. flock/frontend/src/vite-env.d.ts +17 -0
  137. flock/frontend/tsconfig.json +27 -0
  138. flock/frontend/tsconfig.node.json +11 -0
  139. flock/frontend/vite.config.ts +25 -0
  140. flock/frontend/vitest.config.ts +11 -0
  141. flock/{core/util → helper}/cli_helper.py +9 -5
  142. flock/{core/logging → logging}/__init__.py +2 -3
  143. flock/logging/auto_trace.py +159 -0
  144. flock/{core/logging → logging}/formatters/enum_builder.py +3 -4
  145. flock/{core/logging → logging}/formatters/theme_builder.py +19 -44
  146. flock/{core/logging → logging}/formatters/themed_formatter.py +69 -107
  147. flock/{core/logging → logging}/logging.py +78 -61
  148. flock/{core/logging → logging}/telemetry.py +66 -26
  149. flock/{core/logging → logging}/telemetry_exporter/base_exporter.py +2 -2
  150. flock/logging/telemetry_exporter/duckdb_exporter.py +216 -0
  151. flock/{core/logging → logging}/telemetry_exporter/file_exporter.py +13 -10
  152. flock/{core/logging → logging}/telemetry_exporter/sqlite_exporter.py +2 -3
  153. flock/logging/trace_and_logged.py +304 -0
  154. flock/mcp/__init__.py +91 -0
  155. flock/{core/mcp/mcp_client.py → mcp/client.py} +131 -158
  156. flock/{core/mcp/mcp_config.py → mcp/config.py} +86 -132
  157. flock/mcp/manager.py +286 -0
  158. flock/mcp/servers/sse/__init__.py +1 -1
  159. flock/mcp/servers/sse/flock_sse_server.py +16 -58
  160. flock/mcp/servers/stdio/__init__.py +1 -1
  161. flock/mcp/servers/stdio/flock_stdio_server.py +13 -53
  162. flock/mcp/servers/streamable_http/flock_streamable_http_server.py +22 -67
  163. flock/mcp/servers/websockets/flock_websocket_server.py +12 -45
  164. flock/{core/mcp/flock_mcp_tool_base.py → mcp/tool.py} +24 -78
  165. flock/mcp/types/__init__.py +42 -0
  166. flock/{core/mcp → mcp}/types/callbacks.py +12 -15
  167. flock/{core/mcp → mcp}/types/factories.py +7 -6
  168. flock/{core/mcp → mcp}/types/handlers.py +13 -18
  169. flock/{core/mcp → mcp}/types/types.py +70 -74
  170. flock/{core/mcp → mcp}/util/helpers.py +3 -3
  171. flock/orchestrator.py +970 -0
  172. flock/registry.py +148 -0
  173. flock/runtime.py +262 -0
  174. flock/service.py +277 -0
  175. flock/store.py +1214 -0
  176. flock/subscription.py +111 -0
  177. flock/themes/andromeda.toml +1 -1
  178. flock/themes/apple-system-colors.toml +1 -1
  179. flock/themes/arcoiris.toml +1 -1
  180. flock/themes/atomonelight.toml +1 -1
  181. flock/themes/ayu copy.toml +1 -1
  182. flock/themes/ayu-light.toml +1 -1
  183. flock/themes/belafonte-day.toml +1 -1
  184. flock/themes/belafonte-night.toml +1 -1
  185. flock/themes/blulocodark.toml +1 -1
  186. flock/themes/breeze.toml +1 -1
  187. flock/themes/broadcast.toml +1 -1
  188. flock/themes/brogrammer.toml +1 -1
  189. flock/themes/builtin-dark.toml +1 -1
  190. flock/themes/builtin-pastel-dark.toml +1 -1
  191. flock/themes/catppuccin-latte.toml +1 -1
  192. flock/themes/catppuccin-macchiato.toml +1 -1
  193. flock/themes/catppuccin-mocha.toml +1 -1
  194. flock/themes/cga.toml +1 -1
  195. flock/themes/chalk.toml +1 -1
  196. flock/themes/ciapre.toml +1 -1
  197. flock/themes/coffee-theme.toml +1 -1
  198. flock/themes/cyberpunkscarletprotocol.toml +1 -1
  199. flock/themes/dark+.toml +1 -1
  200. flock/themes/darkermatrix.toml +1 -1
  201. flock/themes/darkmatrix.toml +2 -2
  202. flock/themes/darkside.toml +1 -1
  203. flock/themes/deep.toml +2 -2
  204. flock/themes/desert.toml +1 -1
  205. flock/themes/django.toml +1 -1
  206. flock/themes/djangosmooth.toml +1 -1
  207. flock/themes/doomone.toml +1 -1
  208. flock/themes/dotgov.toml +1 -1
  209. flock/themes/dracula+.toml +1 -1
  210. flock/themes/duckbones.toml +1 -1
  211. flock/themes/encom.toml +1 -1
  212. flock/themes/espresso.toml +1 -1
  213. flock/themes/everblush.toml +1 -1
  214. flock/themes/fairyfloss.toml +1 -1
  215. flock/themes/fideloper.toml +1 -1
  216. flock/themes/fishtank.toml +1 -1
  217. flock/themes/flexoki-light.toml +1 -1
  218. flock/themes/floraverse.toml +1 -1
  219. flock/themes/framer.toml +1 -1
  220. flock/themes/galizur.toml +1 -1
  221. flock/themes/github.toml +1 -1
  222. flock/themes/grass.toml +1 -1
  223. flock/themes/grey-green.toml +1 -1
  224. flock/themes/gruvboxlight.toml +1 -1
  225. flock/themes/guezwhoz.toml +1 -1
  226. flock/themes/harper.toml +1 -1
  227. flock/themes/hax0r-blue.toml +1 -1
  228. flock/themes/hopscotch.256.toml +1 -1
  229. flock/themes/ic-green-ppl.toml +1 -1
  230. flock/themes/iceberg-dark.toml +1 -1
  231. flock/themes/japanesque.toml +1 -1
  232. flock/themes/jubi.toml +1 -1
  233. flock/themes/kibble.toml +1 -1
  234. flock/themes/kolorit.toml +1 -1
  235. flock/themes/kurokula.toml +1 -1
  236. flock/themes/materialdesigncolors.toml +1 -1
  237. flock/themes/matrix.toml +1 -1
  238. flock/themes/mellifluous.toml +1 -1
  239. flock/themes/midnight-in-mojave.toml +1 -1
  240. flock/themes/monokai-remastered.toml +1 -1
  241. flock/themes/monokai-soda.toml +1 -1
  242. flock/themes/neon.toml +1 -1
  243. flock/themes/neopolitan.toml +5 -5
  244. flock/themes/nord-light.toml +1 -1
  245. flock/themes/ocean.toml +1 -1
  246. flock/themes/onehalfdark.toml +1 -1
  247. flock/themes/onehalflight.toml +1 -1
  248. flock/themes/palenighthc.toml +1 -1
  249. flock/themes/paulmillr.toml +1 -1
  250. flock/themes/pencildark.toml +1 -1
  251. flock/themes/pnevma.toml +1 -1
  252. flock/themes/purple-rain.toml +1 -1
  253. flock/themes/purplepeter.toml +1 -1
  254. flock/themes/raycast-dark.toml +1 -1
  255. flock/themes/red-sands.toml +1 -1
  256. flock/themes/relaxed.toml +1 -1
  257. flock/themes/retro.toml +1 -1
  258. flock/themes/rose-pine.toml +1 -1
  259. flock/themes/royal.toml +1 -1
  260. flock/themes/ryuuko.toml +1 -1
  261. flock/themes/sakura.toml +1 -1
  262. flock/themes/scarlet-protocol.toml +1 -1
  263. flock/themes/seoulbones-dark.toml +1 -1
  264. flock/themes/shades-of-purple.toml +1 -1
  265. flock/themes/smyck.toml +1 -1
  266. flock/themes/softserver.toml +1 -1
  267. flock/themes/solarized-darcula.toml +1 -1
  268. flock/themes/square.toml +1 -1
  269. flock/themes/sugarplum.toml +1 -1
  270. flock/themes/thayer-bright.toml +1 -1
  271. flock/themes/tokyonight.toml +1 -1
  272. flock/themes/tomorrow.toml +1 -1
  273. flock/themes/ubuntu.toml +1 -1
  274. flock/themes/ultradark.toml +1 -1
  275. flock/themes/ultraviolent.toml +1 -1
  276. flock/themes/unikitty.toml +1 -1
  277. flock/themes/urple.toml +1 -1
  278. flock/themes/vesper.toml +1 -1
  279. flock/themes/vimbones.toml +1 -1
  280. flock/themes/wildcherry.toml +1 -1
  281. flock/themes/wilmersdorf.toml +1 -1
  282. flock/themes/wryan.toml +1 -1
  283. flock/themes/xcodedarkhc.toml +1 -1
  284. flock/themes/xcodelight.toml +1 -1
  285. flock/themes/zenbones-light.toml +1 -1
  286. flock/themes/zenwritten-dark.toml +1 -1
  287. flock/utilities.py +301 -0
  288. flock/utility/output_utility_component.py +226 -0
  289. flock/visibility.py +107 -0
  290. flock_core-0.5.0.dist-info/METADATA +964 -0
  291. flock_core-0.5.0.dist-info/RECORD +525 -0
  292. flock_core-0.5.0.dist-info/entry_points.txt +2 -0
  293. {flock_core-0.4.543.dist-info → flock_core-0.5.0.dist-info}/licenses/LICENSE +1 -1
  294. flock/adapter/__init__.py +0 -14
  295. flock/adapter/azure_adapter.py +0 -68
  296. flock/adapter/chroma_adapter.py +0 -73
  297. flock/adapter/faiss_adapter.py +0 -97
  298. flock/adapter/pinecone_adapter.py +0 -51
  299. flock/adapter/vector_base.py +0 -47
  300. flock/cli/assets/release_notes.md +0 -140
  301. flock/cli/config.py +0 -8
  302. flock/cli/constants.py +0 -36
  303. flock/cli/create_agent.py +0 -1
  304. flock/cli/create_flock.py +0 -280
  305. flock/cli/execute_flock.py +0 -620
  306. flock/cli/load_agent.py +0 -1
  307. flock/cli/load_examples.py +0 -1
  308. flock/cli/load_flock.py +0 -192
  309. flock/cli/load_release_notes.py +0 -20
  310. flock/cli/loaded_flock_cli.py +0 -254
  311. flock/cli/manage_agents.py +0 -459
  312. flock/cli/registry_management.py +0 -889
  313. flock/cli/runner.py +0 -41
  314. flock/cli/settings.py +0 -857
  315. flock/cli/utils.py +0 -135
  316. flock/cli/view_results.py +0 -29
  317. flock/cli/yaml_editor.py +0 -396
  318. flock/config.py +0 -56
  319. flock/core/__init__.py +0 -44
  320. flock/core/api/__init__.py +0 -10
  321. flock/core/api/custom_endpoint.py +0 -45
  322. flock/core/api/endpoints.py +0 -262
  323. flock/core/api/main.py +0 -162
  324. flock/core/api/models.py +0 -101
  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/config/flock_agent_config.py +0 -11
  329. flock/core/config/scheduled_agent_config.py +0 -40
  330. flock/core/context/context.py +0 -214
  331. flock/core/context/context_manager.py +0 -40
  332. flock/core/context/context_vars.py +0 -11
  333. flock/core/evaluation/utils.py +0 -395
  334. flock/core/execution/batch_executor.py +0 -369
  335. flock/core/execution/evaluation_executor.py +0 -438
  336. flock/core/execution/local_executor.py +0 -31
  337. flock/core/execution/opik_executor.py +0 -103
  338. flock/core/execution/temporal_executor.py +0 -166
  339. flock/core/flock.py +0 -1003
  340. flock/core/flock_agent.py +0 -1258
  341. flock/core/flock_evaluator.py +0 -60
  342. flock/core/flock_factory.py +0 -513
  343. flock/core/flock_module.py +0 -207
  344. flock/core/flock_registry.py +0 -702
  345. flock/core/flock_router.py +0 -83
  346. flock/core/flock_scheduler.py +0 -166
  347. flock/core/flock_server_manager.py +0 -136
  348. flock/core/interpreter/python_interpreter.py +0 -689
  349. flock/core/logging/live_capture.py +0 -137
  350. flock/core/logging/trace_and_logged.py +0 -59
  351. flock/core/mcp/__init__.py +0 -1
  352. flock/core/mcp/flock_mcp_server.py +0 -640
  353. flock/core/mcp/mcp_client_manager.py +0 -201
  354. flock/core/mcp/types/__init__.py +0 -1
  355. flock/core/mixin/dspy_integration.py +0 -445
  356. flock/core/mixin/prompt_parser.py +0 -125
  357. flock/core/serialization/__init__.py +0 -13
  358. flock/core/serialization/callable_registry.py +0 -52
  359. flock/core/serialization/flock_serializer.py +0 -854
  360. flock/core/serialization/json_encoder.py +0 -41
  361. flock/core/serialization/secure_serializer.py +0 -175
  362. flock/core/serialization/serializable.py +0 -342
  363. flock/core/serialization/serialization_utils.py +0 -409
  364. flock/core/util/file_path_utils.py +0 -223
  365. flock/core/util/hydrator.py +0 -309
  366. flock/core/util/input_resolver.py +0 -141
  367. flock/core/util/loader.py +0 -59
  368. flock/core/util/splitter.py +0 -219
  369. flock/di.py +0 -41
  370. flock/evaluators/__init__.py +0 -1
  371. flock/evaluators/declarative/__init__.py +0 -1
  372. flock/evaluators/declarative/declarative_evaluator.py +0 -217
  373. flock/evaluators/memory/memory_evaluator.py +0 -90
  374. flock/evaluators/test/test_case_evaluator.py +0 -38
  375. flock/evaluators/zep/zep_evaluator.py +0 -59
  376. flock/modules/__init__.py +0 -1
  377. flock/modules/assertion/__init__.py +0 -1
  378. flock/modules/assertion/assertion_module.py +0 -286
  379. flock/modules/callback/__init__.py +0 -1
  380. flock/modules/callback/callback_module.py +0 -91
  381. flock/modules/enterprise_memory/README.md +0 -99
  382. flock/modules/enterprise_memory/enterprise_memory_module.py +0 -526
  383. flock/modules/mem0/__init__.py +0 -1
  384. flock/modules/mem0/mem0_module.py +0 -126
  385. flock/modules/mem0_async/__init__.py +0 -1
  386. flock/modules/mem0_async/async_mem0_module.py +0 -126
  387. flock/modules/memory/__init__.py +0 -1
  388. flock/modules/memory/memory_module.py +0 -429
  389. flock/modules/memory/memory_parser.py +0 -125
  390. flock/modules/memory/memory_storage.py +0 -736
  391. flock/modules/output/__init__.py +0 -1
  392. flock/modules/output/output_module.py +0 -196
  393. flock/modules/performance/__init__.py +0 -1
  394. flock/modules/performance/metrics_module.py +0 -678
  395. flock/modules/zep/__init__.py +0 -1
  396. flock/modules/zep/zep_module.py +0 -192
  397. flock/platform/docker_tools.py +0 -49
  398. flock/platform/jaeger_install.py +0 -86
  399. flock/routers/__init__.py +0 -1
  400. flock/routers/agent/__init__.py +0 -1
  401. flock/routers/agent/agent_router.py +0 -236
  402. flock/routers/agent/handoff_agent.py +0 -58
  403. flock/routers/conditional/conditional_router.py +0 -486
  404. flock/routers/default/__init__.py +0 -1
  405. flock/routers/default/default_router.py +0 -80
  406. flock/routers/feedback/feedback_router.py +0 -114
  407. flock/routers/list_generator/list_generator_router.py +0 -166
  408. flock/routers/llm/__init__.py +0 -1
  409. flock/routers/llm/llm_router.py +0 -365
  410. flock/tools/__init__.py +0 -0
  411. flock/tools/azure_tools.py +0 -781
  412. flock/tools/code_tools.py +0 -167
  413. flock/tools/file_tools.py +0 -149
  414. flock/tools/github_tools.py +0 -157
  415. flock/tools/markdown_tools.py +0 -205
  416. flock/tools/system_tools.py +0 -9
  417. flock/tools/text_tools.py +0 -810
  418. flock/tools/web_tools.py +0 -92
  419. flock/tools/zendesk_tools.py +0 -501
  420. flock/webapp/__init__.py +0 -1
  421. flock/webapp/app/__init__.py +0 -0
  422. flock/webapp/app/api/__init__.py +0 -0
  423. flock/webapp/app/api/agent_management.py +0 -237
  424. flock/webapp/app/api/execution.py +0 -503
  425. flock/webapp/app/api/flock_management.py +0 -125
  426. flock/webapp/app/api/registry_viewer.py +0 -29
  427. flock/webapp/app/chat.py +0 -662
  428. flock/webapp/app/config.py +0 -104
  429. flock/webapp/app/dependencies.py +0 -117
  430. flock/webapp/app/main.py +0 -1086
  431. flock/webapp/app/middleware.py +0 -113
  432. flock/webapp/app/models_ui.py +0 -7
  433. flock/webapp/app/services/__init__.py +0 -0
  434. flock/webapp/app/services/feedback_file_service.py +0 -363
  435. flock/webapp/app/services/flock_service.py +0 -345
  436. flock/webapp/app/services/sharing_models.py +0 -81
  437. flock/webapp/app/services/sharing_store.py +0 -597
  438. flock/webapp/app/templates/theme_mapper.html +0 -326
  439. flock/webapp/app/theme_mapper.py +0 -811
  440. flock/webapp/app/utils.py +0 -85
  441. flock/webapp/run.py +0 -219
  442. flock/webapp/static/css/chat.css +0 -301
  443. flock/webapp/static/css/components.css +0 -167
  444. flock/webapp/static/css/header.css +0 -39
  445. flock/webapp/static/css/layout.css +0 -281
  446. flock/webapp/static/css/sidebar.css +0 -127
  447. flock/webapp/static/css/two-pane.css +0 -48
  448. flock/webapp/templates/base.html +0 -389
  449. flock/webapp/templates/chat.html +0 -152
  450. flock/webapp/templates/chat_settings.html +0 -19
  451. flock/webapp/templates/flock_editor.html +0 -16
  452. flock/webapp/templates/index.html +0 -12
  453. flock/webapp/templates/partials/_agent_detail_form.html +0 -93
  454. flock/webapp/templates/partials/_agent_list.html +0 -18
  455. flock/webapp/templates/partials/_agent_manager_view.html +0 -51
  456. flock/webapp/templates/partials/_agent_tools_checklist.html +0 -14
  457. flock/webapp/templates/partials/_chat_container.html +0 -15
  458. flock/webapp/templates/partials/_chat_messages.html +0 -57
  459. flock/webapp/templates/partials/_chat_settings_form.html +0 -85
  460. flock/webapp/templates/partials/_create_flock_form.html +0 -50
  461. flock/webapp/templates/partials/_dashboard_flock_detail.html +0 -17
  462. flock/webapp/templates/partials/_dashboard_flock_file_list.html +0 -16
  463. flock/webapp/templates/partials/_dashboard_flock_properties_preview.html +0 -28
  464. flock/webapp/templates/partials/_dashboard_upload_flock_form.html +0 -16
  465. flock/webapp/templates/partials/_dynamic_input_form_content.html +0 -22
  466. flock/webapp/templates/partials/_env_vars_table.html +0 -23
  467. flock/webapp/templates/partials/_execution_form.html +0 -127
  468. flock/webapp/templates/partials/_execution_view_container.html +0 -28
  469. flock/webapp/templates/partials/_flock_file_list.html +0 -23
  470. flock/webapp/templates/partials/_flock_properties_form.html +0 -52
  471. flock/webapp/templates/partials/_flock_upload_form.html +0 -16
  472. flock/webapp/templates/partials/_header_flock_status.html +0 -5
  473. flock/webapp/templates/partials/_live_logs.html +0 -13
  474. flock/webapp/templates/partials/_load_manager_view.html +0 -49
  475. flock/webapp/templates/partials/_registry_table.html +0 -25
  476. flock/webapp/templates/partials/_registry_viewer_content.html +0 -70
  477. flock/webapp/templates/partials/_results_display.html +0 -78
  478. flock/webapp/templates/partials/_settings_env_content.html +0 -9
  479. flock/webapp/templates/partials/_settings_theme_content.html +0 -14
  480. flock/webapp/templates/partials/_settings_view.html +0 -36
  481. flock/webapp/templates/partials/_share_chat_link_snippet.html +0 -11
  482. flock/webapp/templates/partials/_share_link_snippet.html +0 -35
  483. flock/webapp/templates/partials/_sidebar.html +0 -74
  484. flock/webapp/templates/partials/_structured_data_view.html +0 -40
  485. flock/webapp/templates/partials/_theme_preview.html +0 -36
  486. flock/webapp/templates/registry_viewer.html +0 -84
  487. flock/webapp/templates/shared_run_page.html +0 -140
  488. flock/workflow/__init__.py +0 -0
  489. flock/workflow/activities.py +0 -237
  490. flock/workflow/agent_activities.py +0 -24
  491. flock/workflow/agent_execution_activity.py +0 -240
  492. flock/workflow/flock_workflow.py +0 -225
  493. flock/workflow/temporal_config.py +0 -96
  494. flock/workflow/temporal_setup.py +0 -60
  495. flock_core-0.4.543.dist-info/METADATA +0 -676
  496. flock_core-0.4.543.dist-info/RECORD +0 -572
  497. flock_core-0.4.543.dist-info/entry_points.txt +0 -2
  498. /flock/{core/logging → logging}/formatters/themes.py +0 -0
  499. /flock/{core/logging → logging}/span_middleware/baggage_span_processor.py +0 -0
  500. /flock/{core/mcp → mcp}/util/__init__.py +0 -0
  501. {flock_core-0.4.543.dist-info → flock_core-0.5.0.dist-info}/WHEEL +0 -0
flock/registry.py ADDED
@@ -0,0 +1,148 @@
1
+ from __future__ import annotations
2
+
3
+
4
+ """Runtime registries for blackboard artifact and function declarations."""
5
+
6
+ from typing import TYPE_CHECKING, Any
7
+
8
+ from pydantic import BaseModel
9
+
10
+
11
+ if TYPE_CHECKING:
12
+ from collections.abc import Callable
13
+
14
+
15
+ class RegistryError(RuntimeError):
16
+ """Raised when a registry operation fails."""
17
+
18
+
19
+ class TypeRegistry:
20
+ """In-memory registry for blackboard artifact types."""
21
+
22
+ def __init__(self) -> None:
23
+ self._by_name: dict[str, type[BaseModel]] = {}
24
+ self._by_cls: dict[type[BaseModel], str] = {}
25
+
26
+ def register(self, model: type[BaseModel], name: str | None = None) -> str:
27
+ if not issubclass(model, BaseModel):
28
+ raise RegistryError("Only Pydantic models can be registered as artifact types.")
29
+ type_name = (
30
+ name or getattr(model, "__flock_type__", None) or f"{model.__module__}.{model.__name__}"
31
+ )
32
+ existing_model = self._by_name.get(type_name)
33
+ if existing_model is not None and existing_model is not model:
34
+ self._by_cls.pop(existing_model, None)
35
+ existing_name = self._by_cls.get(model)
36
+ if existing_name and existing_name != type_name:
37
+ self._by_name.pop(existing_name, None)
38
+
39
+ self._by_name[type_name] = model
40
+ self._by_cls[model] = type_name
41
+ model.__flock_type__ = type_name
42
+ return type_name
43
+
44
+ def resolve(self, type_name: str) -> type[BaseModel]:
45
+ try:
46
+ return self._by_name[type_name]
47
+ except KeyError as exc: # pragma: no cover - defensive
48
+ raise RegistryError(f"Unknown artifact type '{type_name}'.") from exc
49
+
50
+ def resolve_name(self, type_name: str) -> str:
51
+ """
52
+ Resolve a type name (simple or qualified) to its canonical form.
53
+
54
+ Args:
55
+ type_name: Simple name ("Document") or qualified ("__main__.Document")
56
+
57
+ Returns:
58
+ Canonical type name from registry
59
+
60
+ Raises:
61
+ RegistryError: Type not found or ambiguous
62
+ """
63
+ # If already canonical, return as-is (O(1) lookup)
64
+ if type_name in self._by_name:
65
+ return type_name
66
+
67
+ # Search for models with matching simple name (O(n) scan)
68
+ matches = []
69
+ for canonical_name, model_cls in self._by_name.items():
70
+ if model_cls.__name__ == type_name:
71
+ matches.append(canonical_name)
72
+
73
+ if len(matches) == 0:
74
+ raise RegistryError(f"Unknown artifact type '{type_name}'.")
75
+ if len(matches) == 1:
76
+ return matches[0]
77
+ raise RegistryError(
78
+ f"Ambiguous type name '{type_name}'. Matches: {', '.join(matches)}. Use qualified name."
79
+ )
80
+
81
+ def name_for(self, model: type[BaseModel]) -> str:
82
+ try:
83
+ return self._by_cls[model]
84
+ except KeyError as exc:
85
+ raise RegistryError(
86
+ f"Model '{model.__name__}' is not registered as an artifact type."
87
+ ) from exc
88
+
89
+
90
+ class FunctionRegistry:
91
+ """Registry for deterministic callable helpers (flock_tool)."""
92
+
93
+ def __init__(self) -> None:
94
+ self._callables: dict[str, Callable[..., Any]] = {}
95
+
96
+ def register(self, func: Callable[..., Any], *, name: str | None = None) -> str:
97
+ func_name = name or getattr(func, "__flock_tool__", None) or func.__name__
98
+ existing = self._callables.get(func_name)
99
+ if existing is func:
100
+ return func_name
101
+ self._callables[func_name] = func
102
+ func.__flock_tool__ = func_name
103
+ return func_name
104
+
105
+ def resolve(self, func_name: str) -> Callable[..., Any]:
106
+ try:
107
+ return self._callables[func_name]
108
+ except KeyError as exc:
109
+ raise RegistryError(
110
+ f"Function '{func_name}' is not registered with flock_tool."
111
+ ) from exc
112
+
113
+
114
+ type_registry = TypeRegistry()
115
+ function_registry = FunctionRegistry()
116
+
117
+
118
+ def flock_type(model: type[BaseModel] | None = None, *, name: str | None = None) -> Any:
119
+ """Decorator to register a Pydantic model as a blackboard artifact type."""
120
+
121
+ def _wrap(cls: type[BaseModel]) -> type[BaseModel]:
122
+ type_registry.register(cls, name=name)
123
+ return cls
124
+
125
+ if model is None:
126
+ return _wrap
127
+ return _wrap(model)
128
+
129
+
130
+ def flock_tool(func: Callable[..., Any] | None = None, *, name: str | None = None) -> Any:
131
+ """Decorator to register a deterministic helper function for agents."""
132
+
133
+ def _wrap(callable_: Callable[..., Any]) -> Callable[..., Any]:
134
+ function_registry.register(callable_, name=name)
135
+ return callable_
136
+
137
+ if func is None:
138
+ return _wrap
139
+ return _wrap(func)
140
+
141
+
142
+ __all__ = [
143
+ "RegistryError",
144
+ "flock_tool",
145
+ "flock_type",
146
+ "function_registry",
147
+ "type_registry",
148
+ ]
flock/runtime.py ADDED
@@ -0,0 +1,262 @@
1
+ """Runtime envelopes exchanged between orchestrator and components."""
2
+
3
+ from __future__ import annotations
4
+
5
+ from typing import Any
6
+ from uuid import UUID
7
+
8
+ from pydantic import BaseModel, Field
9
+
10
+ from flock.artifacts import Artifact
11
+
12
+
13
+ class EvalInputs(BaseModel):
14
+ artifacts: list[Artifact] = Field(default_factory=list)
15
+ state: dict[str, Any] = Field(default_factory=dict)
16
+
17
+ def first_as(self, model_cls: type[BaseModel]) -> BaseModel | None:
18
+ """Extract first artifact as model instance.
19
+
20
+ Convenience method to deserialize the first artifact's payload
21
+ into a typed Pydantic model.
22
+
23
+ Args:
24
+ model_cls: Pydantic model class to deserialize to
25
+
26
+ Returns:
27
+ Model instance or None if no artifacts
28
+
29
+ Example:
30
+ >>> class TaskProcessor(EngineComponent):
31
+ ... async def evaluate(self, agent, ctx, inputs: EvalInputs) -> EvalResult:
32
+ ... task = inputs.first_as(Task)
33
+ ... if not task:
34
+ ... return EvalResult.empty()
35
+ ... # Process task...
36
+ """
37
+ if not self.artifacts:
38
+ return None
39
+ return model_cls(**self.artifacts[0].payload)
40
+
41
+ def all_as(self, model_cls: type[BaseModel]) -> list[BaseModel]:
42
+ """Extract all artifacts as model instances.
43
+
44
+ Args:
45
+ model_cls: Pydantic model class to deserialize to
46
+
47
+ Returns:
48
+ List of model instances (empty if no artifacts)
49
+
50
+ Example:
51
+ >>> tasks = inputs.all_as(Task)
52
+ >>> for task in tasks:
53
+ ... # Process each task
54
+ """
55
+ return [model_cls(**artifact.payload) for artifact in self.artifacts]
56
+
57
+
58
+ class EvalResult(BaseModel):
59
+ artifacts: list[Artifact] = Field(default_factory=list)
60
+ state: dict[str, Any] = Field(default_factory=dict)
61
+ metrics: dict[str, float] = Field(default_factory=dict)
62
+ logs: list[str] = Field(default_factory=list)
63
+
64
+ @classmethod
65
+ def from_object(
66
+ cls,
67
+ obj: BaseModel,
68
+ *,
69
+ agent: Any,
70
+ state: dict | None = None,
71
+ metrics: dict | None = None,
72
+ errors: list[str] | None = None,
73
+ ) -> EvalResult:
74
+ """Create EvalResult from a single model instance.
75
+
76
+ Automatically constructs an Artifact from the model instance,
77
+ handling type registry lookup and payload serialization.
78
+
79
+ Args:
80
+ obj: Pydantic model instance to publish as artifact
81
+ agent: Agent producing the artifact (for produced_by field)
82
+ state: Optional state dict to include in result
83
+ metrics: Optional metrics dict (e.g., {"confidence": 0.95})
84
+ errors: Optional list of error messages
85
+
86
+ Returns:
87
+ EvalResult with single artifact
88
+
89
+ Example:
90
+ >>> class TaskProcessor(EngineComponent):
91
+ ... async def evaluate(self, agent, ctx, inputs: EvalInputs) -> EvalResult:
92
+ ... task = inputs.first_as(Task)
93
+ ... processed = Task(name=f"Done: {task.name}", priority=task.priority)
94
+ ... return EvalResult.from_object(processed, agent=agent)
95
+ """
96
+ from flock.artifacts import Artifact
97
+ from flock.registry import type_registry
98
+
99
+ type_name = type_registry.name_for(type(obj))
100
+ artifact = Artifact(
101
+ type=type_name,
102
+ payload=obj.model_dump(),
103
+ produced_by=agent.name,
104
+ )
105
+
106
+ return cls(
107
+ artifacts=[artifact],
108
+ state=state or {},
109
+ metrics=metrics or {},
110
+ errors=errors or [],
111
+ )
112
+
113
+ @classmethod
114
+ def from_objects(
115
+ cls,
116
+ *objs: BaseModel,
117
+ agent: Any,
118
+ state: dict | None = None,
119
+ metrics: dict | None = None,
120
+ errors: list[str] | None = None,
121
+ ) -> EvalResult:
122
+ """Create EvalResult from multiple model instances.
123
+
124
+ Automatically constructs Artifacts from all model instances.
125
+ Useful when an agent produces multiple outputs in one evaluation.
126
+
127
+ Args:
128
+ *objs: Pydantic model instances to publish as artifacts
129
+ agent: Agent producing the artifacts
130
+ state: Optional state dict
131
+ metrics: Optional metrics dict
132
+ errors: Optional list of error messages
133
+
134
+ Returns:
135
+ EvalResult with multiple artifacts
136
+
137
+ Example:
138
+ >>> class MovieEngine(EngineComponent):
139
+ ... async def evaluate(self, agent, ctx, inputs: EvalInputs) -> EvalResult:
140
+ ... idea = inputs.first_as(Idea)
141
+ ... movie = Movie(title=idea.topic.upper(), runtime=240, synopsis="...")
142
+ ... tagline = Tagline(line="Don't miss it!")
143
+ ... return EvalResult.from_objects(
144
+ ... movie, tagline,
145
+ ... agent=agent,
146
+ ... metrics={"confidence": 0.9}
147
+ ... )
148
+ """
149
+ from flock.artifacts import Artifact
150
+ from flock.registry import type_registry
151
+
152
+ artifacts = []
153
+ for obj in objs:
154
+ type_name = type_registry.name_for(type(obj))
155
+ artifact = Artifact(
156
+ type=type_name,
157
+ payload=obj.model_dump(),
158
+ produced_by=agent.name,
159
+ )
160
+ artifacts.append(artifact)
161
+
162
+ return cls(
163
+ artifacts=artifacts,
164
+ state=state or {},
165
+ metrics=metrics or {},
166
+ errors=errors or [],
167
+ )
168
+
169
+ @classmethod
170
+ def empty(
171
+ cls,
172
+ state: dict | None = None,
173
+ metrics: dict | None = None,
174
+ errors: list[str] | None = None,
175
+ ) -> EvalResult:
176
+ """Return empty result with no artifacts.
177
+
178
+ Useful when:
179
+ - Conditions aren't met for processing
180
+ - Agent only updates state without producing output
181
+ - Processing failed (use errors parameter)
182
+
183
+ Args:
184
+ state: Optional state dict
185
+ metrics: Optional metrics dict
186
+ errors: Optional list of error messages
187
+
188
+ Returns:
189
+ EvalResult with empty artifacts list
190
+
191
+ Example:
192
+ >>> class ConditionalProcessor(EngineComponent):
193
+ ... async def evaluate(self, agent, ctx, inputs: EvalInputs) -> EvalResult:
194
+ ... task = inputs.first_as(Task)
195
+ ... if task.priority < 3:
196
+ ... return EvalResult.empty() # Skip low priority
197
+ ... # Process high priority tasks...
198
+
199
+ >>> # With error reporting
200
+ >>> return EvalResult.empty(errors=["Validation failed: missing field"])
201
+ """
202
+ return cls(
203
+ artifacts=[],
204
+ state=state or {},
205
+ metrics=metrics or {},
206
+ errors=errors or [],
207
+ )
208
+
209
+ @classmethod
210
+ def with_state(
211
+ cls,
212
+ state: dict,
213
+ *,
214
+ metrics: dict | None = None,
215
+ errors: list[str] | None = None,
216
+ ) -> EvalResult:
217
+ """Return result with only state updates (no artifacts).
218
+
219
+ Useful for agents that only update context without producing outputs,
220
+ such as validation or enrichment agents.
221
+
222
+ Args:
223
+ state: State dict to pass to downstream agents
224
+ metrics: Optional metrics dict
225
+ errors: Optional list of error messages
226
+
227
+ Returns:
228
+ EvalResult with state but no artifacts
229
+
230
+ Example:
231
+ >>> class ValidationAgent(EngineComponent):
232
+ ... async def evaluate(self, agent, ctx, inputs: EvalInputs) -> EvalResult:
233
+ ... task = inputs.first_as(Task)
234
+ ... is_valid = task.priority >= 1
235
+ ... return EvalResult.with_state(
236
+ ... {"validation_passed": is_valid, "validator": "priority_check"}
237
+ ... )
238
+ """
239
+ return cls(
240
+ artifacts=[],
241
+ state=state,
242
+ metrics=metrics or {},
243
+ errors=errors or [],
244
+ )
245
+
246
+
247
+ class Context(BaseModel):
248
+ board: Any
249
+ orchestrator: Any
250
+ correlation_id: UUID | None = None # NEW!
251
+ task_id: str
252
+ state: dict[str, Any] = Field(default_factory=dict)
253
+
254
+ def get_variable(self, key: str, default: Any = None) -> Any:
255
+ return self.state.get(key, default)
256
+
257
+
258
+ __all__ = [
259
+ "Context",
260
+ "EvalInputs",
261
+ "EvalResult",
262
+ ]
flock/service.py ADDED
@@ -0,0 +1,277 @@
1
+ from __future__ import annotations
2
+
3
+
4
+ """HTTP control plane for the blackboard orchestrator."""
5
+
6
+ from datetime import datetime
7
+ from typing import TYPE_CHECKING, Any
8
+ from uuid import UUID
9
+
10
+ from fastapi import FastAPI, HTTPException, Query
11
+ from fastapi.responses import PlainTextResponse
12
+
13
+ from flock.registry import type_registry
14
+ from flock.store import ArtifactEnvelope, ConsumptionRecord, FilterConfig
15
+
16
+
17
+ if TYPE_CHECKING:
18
+ from flock.orchestrator import Flock
19
+
20
+
21
+ class BlackboardHTTPService:
22
+ def __init__(self, orchestrator: Flock) -> None:
23
+ self.orchestrator = orchestrator
24
+ self.app = FastAPI(title="Blackboard Agents Service", version="1.0.0")
25
+ self._register_routes()
26
+
27
+ def _register_routes(self) -> None:
28
+ app = self.app
29
+ orchestrator = self.orchestrator
30
+
31
+ def _serialize_artifact(
32
+ artifact,
33
+ consumptions: list[ConsumptionRecord] | None = None,
34
+ ) -> dict[str, Any]:
35
+ data = {
36
+ "id": str(artifact.id),
37
+ "type": artifact.type,
38
+ "payload": artifact.payload,
39
+ "produced_by": artifact.produced_by,
40
+ "visibility": artifact.visibility.model_dump(mode="json"),
41
+ "visibility_kind": getattr(artifact.visibility, "kind", "Unknown"),
42
+ "created_at": artifact.created_at.isoformat(),
43
+ "correlation_id": str(artifact.correlation_id) if artifact.correlation_id else None,
44
+ "partition_key": artifact.partition_key,
45
+ "tags": sorted(artifact.tags),
46
+ "version": artifact.version,
47
+ }
48
+ if consumptions is not None:
49
+ data["consumptions"] = [
50
+ {
51
+ "artifact_id": str(record.artifact_id),
52
+ "consumer": record.consumer,
53
+ "run_id": record.run_id,
54
+ "correlation_id": record.correlation_id,
55
+ "consumed_at": record.consumed_at.isoformat(),
56
+ }
57
+ for record in consumptions
58
+ ]
59
+ data["consumed_by"] = sorted({record.consumer for record in consumptions})
60
+ return data
61
+
62
+ def _parse_datetime(value: str | None, label: str) -> datetime | None:
63
+ if value is None:
64
+ return None
65
+ try:
66
+ return datetime.fromisoformat(value)
67
+ except ValueError as exc: # pragma: no cover - FastAPI converts
68
+ raise HTTPException(status_code=400, detail=f"Invalid {label}: {value}") from exc
69
+
70
+ def _make_filter_config(
71
+ type_names: list[str] | None,
72
+ produced_by: list[str] | None,
73
+ correlation_id: str | None,
74
+ tags: list[str] | None,
75
+ visibility: list[str] | None,
76
+ start: str | None,
77
+ end: str | None,
78
+ ) -> FilterConfig:
79
+ return FilterConfig(
80
+ type_names=set(type_names) if type_names else None,
81
+ produced_by=set(produced_by) if produced_by else None,
82
+ correlation_id=correlation_id,
83
+ tags=set(tags) if tags else None,
84
+ visibility=set(visibility) if visibility else None,
85
+ start=_parse_datetime(start, "from"),
86
+ end=_parse_datetime(end, "to"),
87
+ )
88
+
89
+ @app.post("/api/v1/artifacts")
90
+ async def publish_artifact(body: dict[str, Any]) -> dict[str, str]:
91
+ type_name = body.get("type")
92
+ payload = body.get("payload") or {}
93
+ if not type_name:
94
+ raise HTTPException(status_code=400, detail="type is required")
95
+ try:
96
+ await orchestrator.publish({"type": type_name, **payload})
97
+ except Exception as exc: # pragma: no cover - FastAPI converts
98
+ raise HTTPException(status_code=400, detail=str(exc)) from exc
99
+ return {"status": "accepted"}
100
+
101
+ @app.get("/api/v1/artifacts")
102
+ async def list_artifacts(
103
+ type_names: list[str] | None = Query(None, alias="type"),
104
+ produced_by: list[str] | None = Query(None),
105
+ correlation_id: str | None = None,
106
+ tag: list[str] | None = Query(None),
107
+ start: str | None = Query(None, alias="from"),
108
+ end: str | None = Query(None, alias="to"),
109
+ visibility: list[str] | None = Query(None),
110
+ limit: int = Query(50, ge=1, le=500),
111
+ offset: int = Query(0, ge=0),
112
+ embed_meta: bool = Query(False, alias="embed_meta"),
113
+ ) -> dict[str, Any]:
114
+ filters = _make_filter_config(
115
+ type_names,
116
+ produced_by,
117
+ correlation_id,
118
+ tag,
119
+ visibility,
120
+ start,
121
+ end,
122
+ )
123
+ artifacts, total = await orchestrator.store.query_artifacts(
124
+ filters,
125
+ limit=limit,
126
+ offset=offset,
127
+ embed_meta=embed_meta,
128
+ )
129
+ items: list[dict[str, Any]] = []
130
+ for artifact in artifacts:
131
+ if isinstance(artifact, ArtifactEnvelope):
132
+ items.append(_serialize_artifact(artifact.artifact, artifact.consumptions))
133
+ else:
134
+ items.append(_serialize_artifact(artifact))
135
+ return {
136
+ "items": items,
137
+ "pagination": {"limit": limit, "offset": offset, "total": total},
138
+ }
139
+
140
+ @app.get("/api/v1/artifacts/summary")
141
+ async def summarize_artifacts(
142
+ type_names: list[str] | None = Query(None, alias="type"),
143
+ produced_by: list[str] | None = Query(None),
144
+ correlation_id: str | None = None,
145
+ tag: list[str] | None = Query(None),
146
+ start: str | None = Query(None, alias="from"),
147
+ end: str | None = Query(None, alias="to"),
148
+ visibility: list[str] | None = Query(None),
149
+ ) -> dict[str, Any]:
150
+ filters = _make_filter_config(
151
+ type_names,
152
+ produced_by,
153
+ correlation_id,
154
+ tag,
155
+ visibility,
156
+ start,
157
+ end,
158
+ )
159
+ summary = await orchestrator.store.summarize_artifacts(filters)
160
+ return {"summary": summary}
161
+
162
+ @app.get("/api/v1/artifacts/{artifact_id}")
163
+ async def get_artifact(artifact_id: UUID) -> dict[str, Any]:
164
+ artifact = await orchestrator.store.get(artifact_id)
165
+ if artifact is None:
166
+ raise HTTPException(status_code=404, detail="artifact not found")
167
+ return _serialize_artifact(artifact)
168
+
169
+ @app.post("/api/v1/agents/{name}/run")
170
+ async def run_agent(name: str, body: dict[str, Any]) -> dict[str, Any]:
171
+ try:
172
+ agent = orchestrator.get_agent(name)
173
+ except KeyError as exc:
174
+ raise HTTPException(status_code=404, detail="agent not found") from exc
175
+
176
+ inputs_data: list[dict[str, Any]] = body.get("inputs") or []
177
+ inputs = []
178
+ for item in inputs_data:
179
+ type_name = item.get("type")
180
+ payload = item.get("payload") or {}
181
+ if not type_name:
182
+ raise HTTPException(status_code=400, detail="Each input requires 'type'.")
183
+ model = type_registry.resolve(type_name)
184
+ instance = model(**payload)
185
+ inputs.append(instance)
186
+
187
+ try:
188
+ outputs = await orchestrator.direct_invoke(agent, inputs)
189
+ except Exception as exc:
190
+ raise HTTPException(
191
+ status_code=500, detail=f"Agent execution failed: {exc}"
192
+ ) from exc
193
+
194
+ return {
195
+ "artifacts": [
196
+ {
197
+ "id": str(artifact.id),
198
+ "type": artifact.type,
199
+ "payload": artifact.payload,
200
+ "produced_by": artifact.produced_by,
201
+ }
202
+ for artifact in outputs
203
+ ]
204
+ }
205
+
206
+ @app.get("/api/v1/agents")
207
+ async def list_agents() -> dict[str, Any]:
208
+ return {
209
+ "agents": [
210
+ {
211
+ "name": agent.name,
212
+ "description": agent.description,
213
+ "subscriptions": [
214
+ {
215
+ "types": list(subscription.type_names),
216
+ "mode": subscription.mode,
217
+ "delivery": subscription.delivery,
218
+ }
219
+ for subscription in agent.subscriptions
220
+ ],
221
+ "outputs": [output.spec.type_name for output in agent.outputs],
222
+ }
223
+ for agent in orchestrator.agents
224
+ ]
225
+ }
226
+
227
+ @app.get("/api/v1/agents/{agent_id}/history-summary")
228
+ async def agent_history(
229
+ agent_id: str,
230
+ type_names: list[str] | None = Query(None, alias="type"),
231
+ produced_by: list[str] | None = Query(None),
232
+ correlation_id: str | None = None,
233
+ tag: list[str] | None = Query(None),
234
+ start: str | None = Query(None, alias="from"),
235
+ end: str | None = Query(None, alias="to"),
236
+ visibility: list[str] | None = Query(None),
237
+ ) -> dict[str, Any]:
238
+ filters = _make_filter_config(
239
+ type_names,
240
+ produced_by,
241
+ correlation_id,
242
+ tag,
243
+ visibility,
244
+ start,
245
+ end,
246
+ )
247
+ summary = await orchestrator.store.agent_history_summary(agent_id, filters)
248
+ return {"agent_id": agent_id, "summary": summary}
249
+
250
+ @app.get("/health")
251
+ async def health() -> dict[str, str]: # pragma: no cover - trivial
252
+ return {"status": "ok"}
253
+
254
+ @app.get("/metrics")
255
+ async def metrics() -> PlainTextResponse:
256
+ lines = [f"blackboard_{key} {value}" for key, value in orchestrator.metrics.items()]
257
+ return PlainTextResponse("\n".join(lines))
258
+
259
+ def run(
260
+ self, host: str = "127.0.0.1", port: int = 8344
261
+ ) -> None: # pragma: no cover - manual execution
262
+ import uvicorn
263
+
264
+ uvicorn.run(self.app, host=host, port=port)
265
+
266
+ async def run_async(
267
+ self, host: str = "127.0.0.1", port: int = 8344
268
+ ) -> None: # pragma: no cover - manual execution
269
+ """Run the service asynchronously (for use within async context)."""
270
+ import uvicorn
271
+
272
+ config = uvicorn.Config(self.app, host=host, port=port)
273
+ server = uvicorn.Server(config)
274
+ await server.serve()
275
+
276
+
277
+ __all__ = ["BlackboardHTTPService"]