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
@@ -0,0 +1,337 @@
1
+ /**
2
+ * REST API client for orchestrator control operations.
3
+ *
4
+ * Provides methods to publish artifacts and invoke agents via HTTP endpoints.
5
+ * Handles error responses and provides typed return values.
6
+ *
7
+ * Base URL defaults to /api for same-origin requests.
8
+ */
9
+
10
+ import type { ArtifactSummary } from '../types/filters';
11
+
12
+ const BASE_URL = import.meta.env.VITE_API_BASE_URL || '/api';
13
+
14
+ export interface ArtifactType {
15
+ name: string;
16
+ schema: {
17
+ type: string;
18
+ properties: Record<string, any>;
19
+ };
20
+ }
21
+
22
+ export interface Agent {
23
+ name: string;
24
+ description: string;
25
+ status: string;
26
+ subscriptions: string[];
27
+ output_types: string[];
28
+ }
29
+
30
+ export interface PublishResponse {
31
+ status: string;
32
+ correlation_id: string;
33
+ message: string;
34
+ }
35
+
36
+ export interface InvokeResponse {
37
+ status: string;
38
+ invocation_id: string;
39
+ correlation_id?: string | null;
40
+ agent: string;
41
+ message: string;
42
+ }
43
+
44
+ export interface ArtifactTypesResponse {
45
+ artifact_types: ArtifactType[];
46
+ }
47
+
48
+ export interface AgentsResponse {
49
+ agents: Agent[];
50
+ }
51
+
52
+ export interface ArtifactListItem {
53
+ id: string;
54
+ type: string;
55
+ payload: Record<string, any>;
56
+ produced_by: string;
57
+ created_at: string;
58
+ correlation_id: string | null;
59
+ partition_key: string | null;
60
+ tags: string[];
61
+ visibility: { kind: string; [key: string]: any };
62
+ visibility_kind?: string;
63
+ version?: number;
64
+ consumptions?: ArtifactConsumption[];
65
+ consumed_by?: string[];
66
+ }
67
+
68
+ export interface ArtifactConsumption {
69
+ artifact_id: string;
70
+ consumer: string;
71
+ run_id: string | null;
72
+ correlation_id: string | null;
73
+ consumed_at: string;
74
+ }
75
+
76
+ export interface ArtifactListResponse {
77
+ items: ArtifactListItem[];
78
+ pagination: {
79
+ limit: number;
80
+ offset: number;
81
+ total: number;
82
+ };
83
+ }
84
+
85
+ export interface ArtifactSummaryResponse {
86
+ summary: ArtifactSummary;
87
+ }
88
+
89
+ export interface ArtifactQueryOptions {
90
+ types?: string[];
91
+ producers?: string[];
92
+ correlationId?: string | null;
93
+ tags?: string[];
94
+ visibility?: string[];
95
+ from?: string;
96
+ to?: string;
97
+ limit?: number;
98
+ offset?: number;
99
+ embedMeta?: boolean;
100
+ }
101
+
102
+ export interface ErrorResponse {
103
+ error: string;
104
+ message: string;
105
+ }
106
+
107
+ class ApiError extends Error {
108
+ constructor(public status: number, public errorResponse: ErrorResponse) {
109
+ super(errorResponse.message || errorResponse.error);
110
+ this.name = 'ApiError';
111
+ }
112
+ }
113
+
114
+ /**
115
+ * Fetch artifact types from orchestrator
116
+ */
117
+ export async function fetchArtifactTypes(): Promise<ArtifactType[]> {
118
+ try {
119
+ const response = await fetch(`${BASE_URL}/artifact-types`, {
120
+ method: 'GET',
121
+ headers: {
122
+ 'Content-Type': 'application/json',
123
+ },
124
+ });
125
+
126
+ if (!response.ok) {
127
+ const errorData = await response.json().catch(() => ({
128
+ error: 'Unknown error',
129
+ message: 'Failed to fetch artifact types',
130
+ }));
131
+ throw new ApiError(response.status, errorData);
132
+ }
133
+
134
+ const data: ArtifactTypesResponse = await response.json();
135
+ return data.artifact_types;
136
+ } catch (error) {
137
+ if (error instanceof ApiError) {
138
+ throw error;
139
+ }
140
+ throw new Error('Failed to connect to API server');
141
+ }
142
+ }
143
+
144
+ /**
145
+ * Fetch available agents from orchestrator
146
+ */
147
+ export async function fetchAgents(): Promise<Agent[]> {
148
+ try {
149
+ const response = await fetch(`${BASE_URL}/agents`, {
150
+ method: 'GET',
151
+ headers: {
152
+ 'Content-Type': 'application/json',
153
+ },
154
+ });
155
+
156
+ if (!response.ok) {
157
+ const errorData = await response.json().catch(() => ({
158
+ error: 'Unknown error',
159
+ message: 'Failed to fetch agents',
160
+ }));
161
+ throw new ApiError(response.status, errorData);
162
+ }
163
+
164
+ const data: AgentsResponse = await response.json();
165
+ return data.agents;
166
+ } catch (error) {
167
+ if (error instanceof ApiError) {
168
+ throw error;
169
+ }
170
+ throw new Error('Failed to connect to API server');
171
+ }
172
+ }
173
+
174
+ // UI Optimization Migration (Phase 4.1 - Spec 002): Removed fetchRegisteredAgents()
175
+ // OLD Phase 1 function that transformed API agents to graph store format
176
+ // Backend now provides agent data directly in GraphSnapshot
177
+
178
+ /**
179
+ * Publish an artifact to the orchestrator
180
+ * @param artifactType - The type of artifact to publish
181
+ * @param content - The artifact content as a parsed JSON object
182
+ * @returns Response with correlation ID
183
+ */
184
+ export async function publishArtifact(
185
+ artifactType: string,
186
+ content: any
187
+ ): Promise<PublishResponse> {
188
+ try {
189
+ const response = await fetch(`${BASE_URL}/control/publish`, {
190
+ method: 'POST',
191
+ headers: {
192
+ 'Content-Type': 'application/json',
193
+ },
194
+ body: JSON.stringify({
195
+ artifact_type: artifactType,
196
+ content: content,
197
+ }),
198
+ });
199
+
200
+ if (!response.ok) {
201
+ const errorData = await response.json().catch(() => ({
202
+ error: 'Unknown error',
203
+ message: 'Failed to publish artifact',
204
+ }));
205
+ throw new ApiError(response.status, errorData);
206
+ }
207
+
208
+ return await response.json();
209
+ } catch (error) {
210
+ if (error instanceof ApiError) {
211
+ throw error;
212
+ }
213
+ throw new Error('Failed to connect to API server');
214
+ }
215
+ }
216
+
217
+ /**
218
+ * Invoke an agent via the orchestrator
219
+ * @param agentName - The name of the agent to invoke
220
+ * @returns Response with invocation ID
221
+ */
222
+ export async function invokeAgent(agentName: string): Promise<InvokeResponse> {
223
+ try {
224
+ const response = await fetch(`${BASE_URL}/control/invoke`, {
225
+ method: 'POST',
226
+ headers: {
227
+ 'Content-Type': 'application/json',
228
+ },
229
+ body: JSON.stringify({
230
+ agent: agentName,
231
+ }),
232
+ });
233
+
234
+ if (!response.ok) {
235
+ const errorData = await response.json().catch(() => ({
236
+ error: 'Unknown error',
237
+ message: 'Failed to invoke agent',
238
+ }));
239
+ throw new ApiError(response.status, errorData);
240
+ }
241
+
242
+ return await response.json();
243
+ } catch (error) {
244
+ if (error instanceof ApiError) {
245
+ throw error;
246
+ }
247
+ throw new Error('Failed to connect to API server');
248
+ }
249
+ }
250
+
251
+ const buildArtifactQuery = (options: ArtifactQueryOptions): string => {
252
+ const params = new URLSearchParams();
253
+
254
+ options.types?.forEach((value) => params.append('type', value));
255
+ options.producers?.forEach((value) => params.append('produced_by', value));
256
+ options.tags?.forEach((value) => params.append('tag', value));
257
+ options.visibility?.forEach((value) => params.append('visibility', value));
258
+
259
+ if (options.correlationId) {
260
+ params.append('correlation_id', options.correlationId);
261
+ }
262
+ if (options.from) {
263
+ params.append('from', options.from);
264
+ }
265
+ if (options.to) {
266
+ params.append('to', options.to);
267
+ }
268
+ if (typeof options.limit === 'number') {
269
+ params.append('limit', String(options.limit));
270
+ }
271
+ if (typeof options.offset === 'number') {
272
+ params.append('offset', String(options.offset));
273
+ }
274
+ if (options.embedMeta) {
275
+ params.append('embed_meta', 'true');
276
+ }
277
+
278
+ return params.toString();
279
+ };
280
+
281
+ export async function fetchArtifacts(options: ArtifactQueryOptions = {}): Promise<ArtifactListResponse> {
282
+ const query = buildArtifactQuery(options);
283
+
284
+ try {
285
+ const response = await fetch(`${BASE_URL}/v1/artifacts${query ? `?${query}` : ''}`, {
286
+ method: 'GET',
287
+ headers: {
288
+ 'Content-Type': 'application/json',
289
+ },
290
+ });
291
+
292
+ if (!response.ok) {
293
+ const errorData = await response.json().catch(() => ({
294
+ error: 'Unknown error',
295
+ message: 'Failed to fetch artifacts',
296
+ }));
297
+ throw new ApiError(response.status, errorData);
298
+ }
299
+
300
+ const data: ArtifactListResponse = await response.json();
301
+ return data;
302
+ } catch (error) {
303
+ if (error instanceof ApiError) {
304
+ throw error;
305
+ }
306
+ throw new Error('Failed to connect to API server');
307
+ }
308
+ }
309
+
310
+ export async function fetchArtifactSummary(options: ArtifactQueryOptions = {}): Promise<ArtifactSummary> {
311
+ const query = buildArtifactQuery(options);
312
+
313
+ try {
314
+ const response = await fetch(`${BASE_URL}/v1/artifacts/summary${query ? `?${query}` : ''}`, {
315
+ method: 'GET',
316
+ headers: {
317
+ 'Content-Type': 'application/json',
318
+ },
319
+ });
320
+
321
+ if (!response.ok) {
322
+ const errorData = await response.json().catch(() => ({
323
+ error: 'Unknown error',
324
+ message: 'Failed to fetch artifact summary',
325
+ }));
326
+ throw new ApiError(response.status, errorData);
327
+ }
328
+
329
+ const data: ArtifactSummaryResponse = await response.json();
330
+ return data.summary;
331
+ } catch (error) {
332
+ if (error instanceof ApiError) {
333
+ throw error;
334
+ }
335
+ throw new Error('Failed to connect to API server');
336
+ }
337
+ }
@@ -0,0 +1,330 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
2
+ import { fetchGraphSnapshot, mergeNodePositions, overlayWebSocketState } from './graphService';
3
+ import { GraphRequest, GraphSnapshot } from '../types/graph';
4
+ import { Node } from '@xyflow/react';
5
+
6
+ /**
7
+ * Graph Service Tests - UI Optimization Migration (Spec 002)
8
+ *
9
+ * Tests the NEW graph service layer that replaces client-side graph construction
10
+ * with backend snapshot consumption.
11
+ *
12
+ * SPECIFICATION: docs/internal/ui-optimization/03-migration-implementation-guide.md
13
+ * FOCUS: Backend integration, position merging, WebSocket state overlay, error handling
14
+ */
15
+
16
+ describe('graphService', () => {
17
+ beforeEach(() => {
18
+ vi.clearAllMocks();
19
+ });
20
+
21
+ afterEach(() => {
22
+ vi.restoreAllMocks();
23
+ });
24
+
25
+ describe('fetchGraphSnapshot', () => {
26
+ it('should fetch agent view graph from backend with correct request format', async () => {
27
+ const mockSnapshot: GraphSnapshot = {
28
+ nodes: [
29
+ { id: 'agent1', type: 'agent', data: { name: 'agent1' }, position: { x: 0, y: 0 }, hidden: false },
30
+ ],
31
+ edges: [
32
+ { id: 'edge1', source: 'agent1', target: 'agent2', type: 'message_flow', hidden: false, data: {} },
33
+ ],
34
+ statistics: null,
35
+ viewMode: 'agent',
36
+ filters: {
37
+ correlation_id: null,
38
+ time_range: { preset: 'last10min' },
39
+ artifactTypes: [],
40
+ producers: [],
41
+ tags: [],
42
+ visibility: [],
43
+ },
44
+ generatedAt: '2025-10-11T00:00:00Z',
45
+ totalArtifacts: 1,
46
+ truncated: false,
47
+ };
48
+
49
+ globalThis.fetch = vi.fn().mockResolvedValue({
50
+ ok: true,
51
+ json: async () => mockSnapshot,
52
+ }) as any;
53
+
54
+ const request: GraphRequest = {
55
+ viewMode: 'agent',
56
+ filters: {
57
+ correlation_id: null,
58
+ time_range: { preset: 'last10min' },
59
+ artifactTypes: ['Pizza'],
60
+ producers: ['pizza_master'],
61
+ tags: [],
62
+ visibility: [],
63
+ },
64
+ options: { include_statistics: true },
65
+ };
66
+
67
+ const result = await fetchGraphSnapshot(request);
68
+
69
+ expect(globalThis.fetch).toHaveBeenCalledWith('/api/dashboard/graph', {
70
+ method: 'POST',
71
+ headers: { 'Content-Type': 'application/json' },
72
+ body: JSON.stringify(request),
73
+ });
74
+
75
+ expect(result).toEqual(mockSnapshot);
76
+ expect(result.nodes).toHaveLength(1);
77
+ expect(result.edges).toHaveLength(1);
78
+ });
79
+
80
+ it('should throw error when API call fails', async () => {
81
+ globalThis.fetch = vi.fn().mockResolvedValue({
82
+ ok: false,
83
+ statusText: 'Internal Server Error',
84
+ });
85
+
86
+ const request: GraphRequest = {
87
+ viewMode: 'agent',
88
+ filters: {
89
+ correlation_id: null,
90
+ time_range: { preset: 'last10min' },
91
+ artifactTypes: [],
92
+ producers: [],
93
+ tags: [],
94
+ visibility: [],
95
+ },
96
+ };
97
+
98
+ await expect(fetchGraphSnapshot(request)).rejects.toThrow('Graph API error: Internal Server Error');
99
+ });
100
+
101
+ it('should handle network errors gracefully', async () => {
102
+ globalThis.fetch = vi.fn().mockRejectedValue(new Error('Network error'));
103
+
104
+ const request: GraphRequest = {
105
+ viewMode: 'blackboard',
106
+ filters: {
107
+ correlation_id: null,
108
+ time_range: { preset: 'last5min' },
109
+ artifactTypes: [],
110
+ producers: [],
111
+ tags: [],
112
+ visibility: [],
113
+ },
114
+ };
115
+
116
+ await expect(fetchGraphSnapshot(request)).rejects.toThrow('Network error');
117
+ });
118
+ });
119
+
120
+ describe('mergeNodePositions - Priority Logic', () => {
121
+ it('should prioritize saved positions over all others', () => {
122
+ const backendNodes = [
123
+ { id: 'agent1', type: 'agent' as const, data: { name: 'agent1' }, position: { x: 100, y: 100 }, hidden: false },
124
+ ];
125
+
126
+ const savedPositions = new Map([['agent1', { x: 500, y: 500 }]]);
127
+
128
+ const currentNodes: Node[] = [
129
+ { id: 'agent1', type: 'agent', data: { name: 'agent1' }, position: { x: 300, y: 300 } },
130
+ ];
131
+
132
+ const result = mergeNodePositions(backendNodes, savedPositions, currentNodes);
133
+
134
+ // Saved position (500, 500) wins
135
+ expect(result[0]?.position).toEqual({ x: 500, y: 500 });
136
+ });
137
+
138
+ it('should prioritize current positions when no saved position exists', () => {
139
+ const backendNodes = [
140
+ { id: 'agent1', type: 'agent' as const, data: { name: 'agent1' }, position: { x: 100, y: 100 }, hidden: false },
141
+ ];
142
+
143
+ const savedPositions = new Map(); // No saved position
144
+
145
+ const currentNodes: Node[] = [
146
+ { id: 'agent1', type: 'agent', data: { name: 'agent1' }, position: { x: 300, y: 300 } },
147
+ ];
148
+
149
+ const result = mergeNodePositions(backendNodes, savedPositions, currentNodes);
150
+
151
+ // Current position (300, 300) wins
152
+ expect(result[0]!.position).toEqual({ x: 300, y: 300 });
153
+ });
154
+
155
+ it('should use backend position when saved and current are unavailable', () => {
156
+ const backendNodes = [
157
+ { id: 'agent1', type: 'agent' as const, data: { name: 'agent1' }, position: { x: 100, y: 100 }, hidden: false },
158
+ ];
159
+
160
+ const savedPositions = new Map(); // No saved position
161
+ const currentNodes: Node[] = []; // No current position
162
+
163
+ const result = mergeNodePositions(backendNodes, savedPositions, currentNodes);
164
+
165
+ // Backend position (100, 100) wins
166
+ expect(result[0]!.position).toEqual({ x: 100, y: 100 });
167
+ });
168
+
169
+ it('should generate random position when backend has zero coordinates', () => {
170
+ const backendNodes = [
171
+ { id: 'agent1', type: 'agent' as const, data: { name: 'agent1' }, position: { x: 0, y: 0 }, hidden: false },
172
+ ];
173
+
174
+ const savedPositions = new Map(); // No saved position
175
+ const currentNodes: Node[] = []; // No current position
176
+
177
+ const result = mergeNodePositions(backendNodes, savedPositions, currentNodes);
178
+
179
+ // Random position should be generated (not 0, 0)
180
+ expect(result[0]!.position.x).toBeGreaterThan(0);
181
+ expect(result[0]!.position.y).toBeGreaterThan(0);
182
+ // Random position range: x in [400, 600], y in [300, 500]
183
+ expect(result[0]!.position.x).toBeGreaterThanOrEqual(400);
184
+ expect(result[0]!.position.x).toBeLessThanOrEqual(600);
185
+ expect(result[0]!.position.y).toBeGreaterThanOrEqual(300);
186
+ expect(result[0]!.position.y).toBeLessThanOrEqual(500);
187
+ });
188
+
189
+ it('should handle multiple nodes with mixed position sources', () => {
190
+ const backendNodes = [
191
+ { id: 'agent1', type: 'agent' as const, data: { name: 'agent1' }, position: { x: 100, y: 100 }, hidden: false },
192
+ { id: 'agent2', type: 'agent' as const, data: { name: 'agent2' }, position: { x: 200, y: 200 }, hidden: false },
193
+ { id: 'agent3', type: 'agent' as const, data: { name: 'agent3' }, position: { x: 0, y: 0 }, hidden: false },
194
+ ];
195
+
196
+ const savedPositions = new Map([['agent1', { x: 500, y: 500 }]]); // Only agent1
197
+
198
+ const currentNodes: Node[] = [
199
+ { id: 'agent2', type: 'agent', data: { name: 'agent2' }, position: { x: 300, y: 300 } },
200
+ ];
201
+
202
+ const result = mergeNodePositions(backendNodes, savedPositions, currentNodes);
203
+
204
+ // agent1: saved position wins
205
+ expect(result[0]!.position).toEqual({ x: 500, y: 500 });
206
+ // agent2: current position wins
207
+ expect(result[1]!.position).toEqual({ x: 300, y: 300 });
208
+ // agent3: random position (backend is 0,0)
209
+ expect(result[2]!.position.x).toBeGreaterThan(0);
210
+ expect(result[2]!.position.y).toBeGreaterThan(0);
211
+ });
212
+ });
213
+
214
+ describe('overlayWebSocketState', () => {
215
+ it('should overlay agent status from WebSocket state', () => {
216
+ const nodes: Node[] = [
217
+ {
218
+ id: 'agent1',
219
+ type: 'agent',
220
+ data: { name: 'agent1', status: 'idle' },
221
+ position: { x: 100, y: 100 },
222
+ },
223
+ ];
224
+
225
+ const agentStatus = new Map([['agent1', 'running']]);
226
+ const streamingTokens = new Map<string, string[]>();
227
+
228
+ const result = overlayWebSocketState(nodes, agentStatus, streamingTokens);
229
+
230
+ expect(result[0]!.data.status).toBe('running');
231
+ });
232
+
233
+ it('should overlay streaming tokens from WebSocket state', () => {
234
+ const nodes: Node[] = [
235
+ {
236
+ id: 'agent1',
237
+ type: 'agent',
238
+ data: { name: 'agent1', status: 'idle' },
239
+ position: { x: 100, y: 100 },
240
+ },
241
+ ];
242
+
243
+ const agentStatus = new Map<string, string>();
244
+ const streamingTokens = new Map([['agent1', ['token1', 'token2', 'token3']]]);
245
+
246
+ const result = overlayWebSocketState(nodes, agentStatus, streamingTokens);
247
+
248
+ expect(result[0]!.data.streamingTokens).toEqual(['token1', 'token2', 'token3']);
249
+ });
250
+
251
+ it('should use backend status when WebSocket state is unavailable', () => {
252
+ const nodes: Node[] = [
253
+ {
254
+ id: 'agent1',
255
+ type: 'agent',
256
+ data: { name: 'agent1', status: 'idle' },
257
+ position: { x: 100, y: 100 },
258
+ },
259
+ ];
260
+
261
+ const agentStatus = new Map<string, string>(); // Empty
262
+ const streamingTokens = new Map<string, string[]>(); // Empty
263
+
264
+ const result = overlayWebSocketState(nodes, agentStatus, streamingTokens);
265
+
266
+ // Backend status is preserved
267
+ expect(result[0]!.data.status).toBe('idle');
268
+ // Empty array for tokens
269
+ expect(result[0]!.data.streamingTokens).toEqual([]);
270
+ });
271
+
272
+ it('should not modify non-agent nodes', () => {
273
+ const nodes: Node[] = [
274
+ {
275
+ id: 'message1',
276
+ type: 'message',
277
+ data: { type: 'Pizza', payload: {} },
278
+ position: { x: 100, y: 100 },
279
+ },
280
+ ];
281
+
282
+ const agentStatus = new Map([['message1', 'running']]);
283
+ const streamingTokens = new Map([['message1', ['token1']]]);
284
+
285
+ const result = overlayWebSocketState(nodes, agentStatus, streamingTokens);
286
+
287
+ // Message node should not be modified
288
+ expect(result[0]!.data).toEqual({ type: 'Pizza', payload: {} });
289
+ expect(result[0]!.data.status).toBeUndefined();
290
+ expect(result[0]!.data.streamingTokens).toBeUndefined();
291
+ });
292
+
293
+ it('should handle multiple agent nodes with mixed WebSocket state', () => {
294
+ const nodes: Node[] = [
295
+ {
296
+ id: 'agent1',
297
+ type: 'agent',
298
+ data: { name: 'agent1', status: 'idle' },
299
+ position: { x: 100, y: 100 },
300
+ },
301
+ {
302
+ id: 'agent2',
303
+ type: 'agent',
304
+ data: { name: 'agent2', status: 'idle' },
305
+ position: { x: 200, y: 200 },
306
+ },
307
+ {
308
+ id: 'message1',
309
+ type: 'message',
310
+ data: { type: 'Pizza' },
311
+ position: { x: 300, y: 300 },
312
+ },
313
+ ];
314
+
315
+ const agentStatus = new Map([['agent1', 'running']]); // Only agent1
316
+ const streamingTokens = new Map([['agent2', ['token1', 'token2']]]); // Only agent2
317
+
318
+ const result = overlayWebSocketState(nodes, agentStatus, streamingTokens);
319
+
320
+ // agent1: status updated, no tokens
321
+ expect(result[0]!.data.status).toBe('running');
322
+ expect(result[0]!.data.streamingTokens).toEqual([]);
323
+ // agent2: status from backend, tokens updated
324
+ expect(result[1]!.data.status).toBe('idle');
325
+ expect(result[1]!.data.streamingTokens).toEqual(['token1', 'token2']);
326
+ // message1: unchanged
327
+ expect(result[2]!.data).toEqual({ type: 'Pizza' });
328
+ });
329
+ });
330
+ });