flock-core 0.5.0b28__py3-none-any.whl → 0.5.0b51__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of flock-core might be problematic. Click here for more details.

Files changed (469) hide show
  1. flock/__init__.py +12 -217
  2. flock/agent.py +678 -0
  3. flock/api/themes.py +71 -0
  4. flock/artifacts.py +79 -0
  5. flock/cli.py +75 -0
  6. flock/components.py +173 -0
  7. flock/dashboard/__init__.py +28 -0
  8. flock/dashboard/collector.py +283 -0
  9. flock/dashboard/events.py +182 -0
  10. flock/dashboard/launcher.py +230 -0
  11. flock/dashboard/service.py +537 -0
  12. flock/dashboard/websocket.py +235 -0
  13. flock/engines/__init__.py +6 -0
  14. flock/engines/dspy_engine.py +856 -0
  15. flock/examples.py +128 -0
  16. flock/frontend/README.md +678 -0
  17. flock/frontend/docs/DESIGN_SYSTEM.md +1980 -0
  18. flock/frontend/index.html +12 -0
  19. flock/frontend/package-lock.json +4347 -0
  20. flock/frontend/package.json +48 -0
  21. flock/frontend/src/App.tsx +79 -0
  22. flock/frontend/src/__tests__/e2e/critical-scenarios.test.tsx +587 -0
  23. flock/frontend/src/__tests__/integration/filtering-e2e.test.tsx +387 -0
  24. flock/frontend/src/__tests__/integration/graph-rendering.test.tsx +640 -0
  25. flock/frontend/src/__tests__/integration/indexeddb-persistence.test.tsx +699 -0
  26. flock/frontend/src/components/common/BuildInfo.tsx +39 -0
  27. flock/frontend/src/components/common/EmptyState.module.css +115 -0
  28. flock/frontend/src/components/common/EmptyState.tsx +128 -0
  29. flock/frontend/src/components/common/ErrorBoundary.module.css +169 -0
  30. flock/frontend/src/components/common/ErrorBoundary.tsx +118 -0
  31. flock/frontend/src/components/common/KeyboardShortcutsDialog.css +251 -0
  32. flock/frontend/src/components/common/KeyboardShortcutsDialog.tsx +151 -0
  33. flock/frontend/src/components/common/LoadingSpinner.module.css +97 -0
  34. flock/frontend/src/components/common/LoadingSpinner.tsx +29 -0
  35. flock/frontend/src/components/controls/PublishControl.css +547 -0
  36. flock/frontend/src/components/controls/PublishControl.test.tsx +543 -0
  37. flock/frontend/src/components/controls/PublishControl.tsx +432 -0
  38. flock/frontend/src/components/details/DetailWindowContainer.tsx +62 -0
  39. flock/frontend/src/components/details/LiveOutputTab.test.tsx +792 -0
  40. flock/frontend/src/components/details/LiveOutputTab.tsx +220 -0
  41. flock/frontend/src/components/details/MessageHistoryTab.tsx +299 -0
  42. flock/frontend/src/components/details/NodeDetailWindow.test.tsx +501 -0
  43. flock/frontend/src/components/details/NodeDetailWindow.tsx +218 -0
  44. flock/frontend/src/components/details/RunStatusTab.tsx +307 -0
  45. flock/frontend/src/components/details/tabs.test.tsx +1015 -0
  46. flock/frontend/src/components/filters/CorrelationIDFilter.module.css +102 -0
  47. flock/frontend/src/components/filters/CorrelationIDFilter.test.tsx +197 -0
  48. flock/frontend/src/components/filters/CorrelationIDFilter.tsx +121 -0
  49. flock/frontend/src/components/filters/FilterBar.module.css +29 -0
  50. flock/frontend/src/components/filters/FilterBar.test.tsx +133 -0
  51. flock/frontend/src/components/filters/FilterBar.tsx +33 -0
  52. flock/frontend/src/components/filters/FilterPills.module.css +79 -0
  53. flock/frontend/src/components/filters/FilterPills.test.tsx +173 -0
  54. flock/frontend/src/components/filters/FilterPills.tsx +67 -0
  55. flock/frontend/src/components/filters/TimeRangeFilter.module.css +91 -0
  56. flock/frontend/src/components/filters/TimeRangeFilter.test.tsx +154 -0
  57. flock/frontend/src/components/filters/TimeRangeFilter.tsx +105 -0
  58. flock/frontend/src/components/graph/AgentNode.test.tsx +75 -0
  59. flock/frontend/src/components/graph/AgentNode.tsx +322 -0
  60. flock/frontend/src/components/graph/GraphCanvas.tsx +406 -0
  61. flock/frontend/src/components/graph/MessageFlowEdge.tsx +128 -0
  62. flock/frontend/src/components/graph/MessageNode.test.tsx +62 -0
  63. flock/frontend/src/components/graph/MessageNode.tsx +116 -0
  64. flock/frontend/src/components/graph/MiniMap.tsx +47 -0
  65. flock/frontend/src/components/graph/TransformEdge.tsx +123 -0
  66. flock/frontend/src/components/layout/DashboardLayout.css +407 -0
  67. flock/frontend/src/components/layout/DashboardLayout.tsx +300 -0
  68. flock/frontend/src/components/layout/Header.module.css +88 -0
  69. flock/frontend/src/components/layout/Header.tsx +52 -0
  70. flock/frontend/src/components/modules/EventLogModule.test.tsx +401 -0
  71. flock/frontend/src/components/modules/EventLogModule.tsx +396 -0
  72. flock/frontend/src/components/modules/EventLogModuleWrapper.tsx +17 -0
  73. flock/frontend/src/components/modules/ModuleRegistry.test.ts +333 -0
  74. flock/frontend/src/components/modules/ModuleRegistry.ts +85 -0
  75. flock/frontend/src/components/modules/ModuleWindow.tsx +155 -0
  76. flock/frontend/src/components/modules/registerModules.ts +20 -0
  77. flock/frontend/src/components/settings/AdvancedSettings.tsx +175 -0
  78. flock/frontend/src/components/settings/AppearanceSettings.tsx +185 -0
  79. flock/frontend/src/components/settings/GraphSettings.tsx +110 -0
  80. flock/frontend/src/components/settings/SettingsPanel.css +327 -0
  81. flock/frontend/src/components/settings/SettingsPanel.tsx +131 -0
  82. flock/frontend/src/components/settings/ThemeSelector.tsx +298 -0
  83. flock/frontend/src/hooks/useKeyboardShortcuts.ts +148 -0
  84. flock/frontend/src/hooks/useModulePersistence.test.ts +442 -0
  85. flock/frontend/src/hooks/useModulePersistence.ts +154 -0
  86. flock/frontend/src/hooks/useModules.ts +139 -0
  87. flock/frontend/src/hooks/usePersistence.ts +139 -0
  88. flock/frontend/src/main.tsx +13 -0
  89. flock/frontend/src/services/api.ts +213 -0
  90. flock/frontend/src/services/indexeddb.test.ts +793 -0
  91. flock/frontend/src/services/indexeddb.ts +794 -0
  92. flock/frontend/src/services/layout.test.ts +437 -0
  93. flock/frontend/src/services/layout.ts +146 -0
  94. flock/frontend/src/services/themeApplicator.ts +140 -0
  95. flock/frontend/src/services/themeService.ts +77 -0
  96. flock/frontend/src/services/websocket.test.ts +595 -0
  97. flock/frontend/src/services/websocket.ts +685 -0
  98. flock/frontend/src/store/filterStore.test.ts +242 -0
  99. flock/frontend/src/store/filterStore.ts +103 -0
  100. flock/frontend/src/store/graphStore.test.ts +186 -0
  101. flock/frontend/src/store/graphStore.ts +414 -0
  102. flock/frontend/src/store/moduleStore.test.ts +253 -0
  103. flock/frontend/src/store/moduleStore.ts +57 -0
  104. flock/frontend/src/store/settingsStore.ts +188 -0
  105. flock/frontend/src/store/streamStore.ts +68 -0
  106. flock/frontend/src/store/uiStore.test.ts +54 -0
  107. flock/frontend/src/store/uiStore.ts +110 -0
  108. flock/frontend/src/store/wsStore.ts +34 -0
  109. flock/frontend/src/styles/index.css +15 -0
  110. flock/frontend/src/styles/scrollbar.css +47 -0
  111. flock/frontend/src/styles/variables.css +488 -0
  112. flock/frontend/src/test/setup.ts +1 -0
  113. flock/frontend/src/types/filters.ts +14 -0
  114. flock/frontend/src/types/graph.ts +55 -0
  115. flock/frontend/src/types/modules.ts +7 -0
  116. flock/frontend/src/types/theme.ts +55 -0
  117. flock/frontend/src/utils/mockData.ts +85 -0
  118. flock/frontend/src/utils/performance.ts +16 -0
  119. flock/frontend/src/utils/transforms.test.ts +860 -0
  120. flock/frontend/src/utils/transforms.ts +323 -0
  121. flock/frontend/src/vite-env.d.ts +17 -0
  122. flock/frontend/tsconfig.json +27 -0
  123. flock/frontend/tsconfig.node.json +11 -0
  124. flock/frontend/vite.config.ts +25 -0
  125. flock/frontend/vitest.config.ts +11 -0
  126. flock/{core/util → helper}/cli_helper.py +4 -3
  127. flock/{core/logging → logging}/__init__.py +2 -3
  128. flock/{core/logging → logging}/formatters/enum_builder.py +3 -4
  129. flock/{core/logging → logging}/formatters/theme_builder.py +19 -44
  130. flock/{core/logging → logging}/formatters/themed_formatter.py +69 -115
  131. flock/{core/logging → logging}/logging.py +77 -61
  132. flock/{core/logging → logging}/telemetry.py +20 -26
  133. flock/{core/logging → logging}/telemetry_exporter/base_exporter.py +2 -2
  134. flock/{core/logging → logging}/telemetry_exporter/file_exporter.py +6 -9
  135. flock/{core/logging → logging}/telemetry_exporter/sqlite_exporter.py +2 -3
  136. flock/{core/logging → logging}/trace_and_logged.py +20 -24
  137. flock/mcp/__init__.py +91 -0
  138. flock/{core/mcp/mcp_client.py → mcp/client.py} +103 -154
  139. flock/{core/mcp/mcp_config.py → mcp/config.py} +62 -117
  140. flock/mcp/manager.py +255 -0
  141. flock/mcp/servers/sse/__init__.py +1 -1
  142. flock/mcp/servers/sse/flock_sse_server.py +11 -53
  143. flock/mcp/servers/stdio/__init__.py +1 -1
  144. flock/mcp/servers/stdio/flock_stdio_server.py +8 -48
  145. flock/mcp/servers/streamable_http/flock_streamable_http_server.py +17 -62
  146. flock/mcp/servers/websockets/flock_websocket_server.py +7 -40
  147. flock/{core/mcp/flock_mcp_tool.py → mcp/tool.py} +16 -26
  148. flock/mcp/types/__init__.py +42 -0
  149. flock/{core/mcp → mcp}/types/callbacks.py +9 -15
  150. flock/{core/mcp → mcp}/types/factories.py +7 -6
  151. flock/{core/mcp → mcp}/types/handlers.py +13 -18
  152. flock/{core/mcp → mcp}/types/types.py +70 -74
  153. flock/{core/mcp → mcp}/util/helpers.py +1 -1
  154. flock/orchestrator.py +645 -0
  155. flock/registry.py +148 -0
  156. flock/runtime.py +262 -0
  157. flock/service.py +140 -0
  158. flock/store.py +69 -0
  159. flock/subscription.py +111 -0
  160. flock/themes/andromeda.toml +1 -1
  161. flock/themes/apple-system-colors.toml +1 -1
  162. flock/themes/arcoiris.toml +1 -1
  163. flock/themes/atomonelight.toml +1 -1
  164. flock/themes/ayu copy.toml +1 -1
  165. flock/themes/ayu-light.toml +1 -1
  166. flock/themes/belafonte-day.toml +1 -1
  167. flock/themes/belafonte-night.toml +1 -1
  168. flock/themes/blulocodark.toml +1 -1
  169. flock/themes/breeze.toml +1 -1
  170. flock/themes/broadcast.toml +1 -1
  171. flock/themes/brogrammer.toml +1 -1
  172. flock/themes/builtin-dark.toml +1 -1
  173. flock/themes/builtin-pastel-dark.toml +1 -1
  174. flock/themes/catppuccin-latte.toml +1 -1
  175. flock/themes/catppuccin-macchiato.toml +1 -1
  176. flock/themes/catppuccin-mocha.toml +1 -1
  177. flock/themes/cga.toml +1 -1
  178. flock/themes/chalk.toml +1 -1
  179. flock/themes/ciapre.toml +1 -1
  180. flock/themes/coffee-theme.toml +1 -1
  181. flock/themes/cyberpunkscarletprotocol.toml +1 -1
  182. flock/themes/dark+.toml +1 -1
  183. flock/themes/darkermatrix.toml +1 -1
  184. flock/themes/darkside.toml +1 -1
  185. flock/themes/desert.toml +1 -1
  186. flock/themes/django.toml +1 -1
  187. flock/themes/djangosmooth.toml +1 -1
  188. flock/themes/doomone.toml +1 -1
  189. flock/themes/dotgov.toml +1 -1
  190. flock/themes/dracula+.toml +1 -1
  191. flock/themes/duckbones.toml +1 -1
  192. flock/themes/encom.toml +1 -1
  193. flock/themes/espresso.toml +1 -1
  194. flock/themes/everblush.toml +1 -1
  195. flock/themes/fairyfloss.toml +1 -1
  196. flock/themes/fideloper.toml +1 -1
  197. flock/themes/fishtank.toml +1 -1
  198. flock/themes/flexoki-light.toml +1 -1
  199. flock/themes/floraverse.toml +1 -1
  200. flock/themes/framer.toml +1 -1
  201. flock/themes/galizur.toml +1 -1
  202. flock/themes/github.toml +1 -1
  203. flock/themes/grass.toml +1 -1
  204. flock/themes/grey-green.toml +1 -1
  205. flock/themes/gruvboxlight.toml +1 -1
  206. flock/themes/guezwhoz.toml +1 -1
  207. flock/themes/harper.toml +1 -1
  208. flock/themes/hax0r-blue.toml +1 -1
  209. flock/themes/hopscotch.256.toml +1 -1
  210. flock/themes/ic-green-ppl.toml +1 -1
  211. flock/themes/iceberg-dark.toml +1 -1
  212. flock/themes/japanesque.toml +1 -1
  213. flock/themes/jubi.toml +1 -1
  214. flock/themes/kibble.toml +1 -1
  215. flock/themes/kolorit.toml +1 -1
  216. flock/themes/kurokula.toml +1 -1
  217. flock/themes/materialdesigncolors.toml +1 -1
  218. flock/themes/matrix.toml +1 -1
  219. flock/themes/mellifluous.toml +1 -1
  220. flock/themes/midnight-in-mojave.toml +1 -1
  221. flock/themes/monokai-remastered.toml +1 -1
  222. flock/themes/monokai-soda.toml +1 -1
  223. flock/themes/neon.toml +1 -1
  224. flock/themes/neopolitan.toml +1 -1
  225. flock/themes/nord-light.toml +1 -1
  226. flock/themes/ocean.toml +1 -1
  227. flock/themes/onehalfdark.toml +1 -1
  228. flock/themes/onehalflight.toml +1 -1
  229. flock/themes/palenighthc.toml +1 -1
  230. flock/themes/paulmillr.toml +1 -1
  231. flock/themes/pencildark.toml +1 -1
  232. flock/themes/pnevma.toml +1 -1
  233. flock/themes/purple-rain.toml +1 -1
  234. flock/themes/purplepeter.toml +1 -1
  235. flock/themes/raycast-dark.toml +1 -1
  236. flock/themes/red-sands.toml +1 -1
  237. flock/themes/relaxed.toml +1 -1
  238. flock/themes/retro.toml +1 -1
  239. flock/themes/rose-pine.toml +1 -1
  240. flock/themes/royal.toml +1 -1
  241. flock/themes/ryuuko.toml +1 -1
  242. flock/themes/sakura.toml +1 -1
  243. flock/themes/scarlet-protocol.toml +1 -1
  244. flock/themes/seoulbones-dark.toml +1 -1
  245. flock/themes/shades-of-purple.toml +1 -1
  246. flock/themes/smyck.toml +1 -1
  247. flock/themes/softserver.toml +1 -1
  248. flock/themes/solarized-darcula.toml +1 -1
  249. flock/themes/square.toml +1 -1
  250. flock/themes/sugarplum.toml +1 -1
  251. flock/themes/thayer-bright.toml +1 -1
  252. flock/themes/tokyonight.toml +1 -1
  253. flock/themes/tomorrow.toml +1 -1
  254. flock/themes/ubuntu.toml +1 -1
  255. flock/themes/ultradark.toml +1 -1
  256. flock/themes/ultraviolent.toml +1 -1
  257. flock/themes/unikitty.toml +1 -1
  258. flock/themes/urple.toml +1 -1
  259. flock/themes/vesper.toml +1 -1
  260. flock/themes/vimbones.toml +1 -1
  261. flock/themes/wildcherry.toml +1 -1
  262. flock/themes/wilmersdorf.toml +1 -1
  263. flock/themes/wryan.toml +1 -1
  264. flock/themes/xcodedarkhc.toml +1 -1
  265. flock/themes/xcodelight.toml +1 -1
  266. flock/themes/zenbones-light.toml +1 -1
  267. flock/themes/zenwritten-dark.toml +1 -1
  268. flock/utilities.py +301 -0
  269. flock/{components/utility → utility}/output_utility_component.py +68 -53
  270. flock/visibility.py +107 -0
  271. flock_core-0.5.0b51.dist-info/METADATA +747 -0
  272. flock_core-0.5.0b51.dist-info/RECORD +508 -0
  273. flock_core-0.5.0b51.dist-info/entry_points.txt +2 -0
  274. {flock_core-0.5.0b28.dist-info → flock_core-0.5.0b51.dist-info}/licenses/LICENSE +1 -1
  275. flock/adapter/__init__.py +0 -14
  276. flock/adapter/azure_adapter.py +0 -68
  277. flock/adapter/chroma_adapter.py +0 -73
  278. flock/adapter/faiss_adapter.py +0 -97
  279. flock/adapter/pinecone_adapter.py +0 -51
  280. flock/adapter/vector_base.py +0 -47
  281. flock/cli/assets/release_notes.md +0 -140
  282. flock/cli/config.py +0 -8
  283. flock/cli/constants.py +0 -36
  284. flock/cli/create_agent.py +0 -1
  285. flock/cli/create_flock.py +0 -280
  286. flock/cli/execute_flock.py +0 -620
  287. flock/cli/load_agent.py +0 -1
  288. flock/cli/load_examples.py +0 -1
  289. flock/cli/load_flock.py +0 -192
  290. flock/cli/load_release_notes.py +0 -20
  291. flock/cli/loaded_flock_cli.py +0 -254
  292. flock/cli/manage_agents.py +0 -459
  293. flock/cli/registry_management.py +0 -889
  294. flock/cli/runner.py +0 -41
  295. flock/cli/settings.py +0 -857
  296. flock/cli/utils.py +0 -135
  297. flock/cli/view_results.py +0 -29
  298. flock/cli/yaml_editor.py +0 -396
  299. flock/components/__init__.py +0 -30
  300. flock/components/evaluation/__init__.py +0 -9
  301. flock/components/evaluation/declarative_evaluation_component.py +0 -606
  302. flock/components/routing/__init__.py +0 -15
  303. flock/components/routing/conditional_routing_component.py +0 -494
  304. flock/components/routing/default_routing_component.py +0 -103
  305. flock/components/routing/llm_routing_component.py +0 -206
  306. flock/components/utility/__init__.py +0 -22
  307. flock/components/utility/example_utility_component.py +0 -250
  308. flock/components/utility/feedback_utility_component.py +0 -206
  309. flock/components/utility/memory_utility_component.py +0 -550
  310. flock/components/utility/metrics_utility_component.py +0 -700
  311. flock/config.py +0 -61
  312. flock/core/__init__.py +0 -110
  313. flock/core/agent/__init__.py +0 -16
  314. flock/core/agent/default_agent.py +0 -216
  315. flock/core/agent/flock_agent_components.py +0 -104
  316. flock/core/agent/flock_agent_execution.py +0 -101
  317. flock/core/agent/flock_agent_integration.py +0 -260
  318. flock/core/agent/flock_agent_lifecycle.py +0 -186
  319. flock/core/agent/flock_agent_serialization.py +0 -381
  320. flock/core/api/__init__.py +0 -10
  321. flock/core/api/custom_endpoint.py +0 -45
  322. flock/core/api/endpoints.py +0 -254
  323. flock/core/api/main.py +0 -162
  324. flock/core/api/models.py +0 -97
  325. flock/core/api/run_store.py +0 -224
  326. flock/core/api/runner.py +0 -44
  327. flock/core/api/service.py +0 -214
  328. flock/core/component/__init__.py +0 -15
  329. flock/core/component/agent_component_base.py +0 -309
  330. flock/core/component/evaluation_component.py +0 -62
  331. flock/core/component/routing_component.py +0 -74
  332. flock/core/component/utility_component.py +0 -69
  333. flock/core/config/flock_agent_config.py +0 -58
  334. flock/core/config/scheduled_agent_config.py +0 -40
  335. flock/core/context/context.py +0 -213
  336. flock/core/context/context_manager.py +0 -37
  337. flock/core/context/context_vars.py +0 -10
  338. flock/core/evaluation/utils.py +0 -396
  339. flock/core/execution/batch_executor.py +0 -369
  340. flock/core/execution/evaluation_executor.py +0 -438
  341. flock/core/execution/local_executor.py +0 -31
  342. flock/core/execution/opik_executor.py +0 -103
  343. flock/core/execution/temporal_executor.py +0 -164
  344. flock/core/flock.py +0 -634
  345. flock/core/flock_agent.py +0 -336
  346. flock/core/flock_factory.py +0 -613
  347. flock/core/flock_scheduler.py +0 -166
  348. flock/core/flock_server_manager.py +0 -136
  349. flock/core/interpreter/python_interpreter.py +0 -689
  350. flock/core/mcp/__init__.py +0 -1
  351. flock/core/mcp/flock_mcp_server.py +0 -680
  352. flock/core/mcp/mcp_client_manager.py +0 -201
  353. flock/core/mcp/types/__init__.py +0 -1
  354. flock/core/mixin/dspy_integration.py +0 -403
  355. flock/core/mixin/prompt_parser.py +0 -125
  356. flock/core/orchestration/__init__.py +0 -15
  357. flock/core/orchestration/flock_batch_processor.py +0 -94
  358. flock/core/orchestration/flock_evaluator.py +0 -113
  359. flock/core/orchestration/flock_execution.py +0 -295
  360. flock/core/orchestration/flock_initialization.py +0 -149
  361. flock/core/orchestration/flock_server_manager.py +0 -67
  362. flock/core/orchestration/flock_web_server.py +0 -117
  363. flock/core/registry/__init__.py +0 -45
  364. flock/core/registry/agent_registry.py +0 -69
  365. flock/core/registry/callable_registry.py +0 -139
  366. flock/core/registry/component_discovery.py +0 -142
  367. flock/core/registry/component_registry.py +0 -64
  368. flock/core/registry/config_mapping.py +0 -64
  369. flock/core/registry/decorators.py +0 -137
  370. flock/core/registry/registry_hub.py +0 -205
  371. flock/core/registry/server_registry.py +0 -57
  372. flock/core/registry/type_registry.py +0 -86
  373. flock/core/serialization/__init__.py +0 -13
  374. flock/core/serialization/callable_registry.py +0 -52
  375. flock/core/serialization/flock_serializer.py +0 -832
  376. flock/core/serialization/json_encoder.py +0 -41
  377. flock/core/serialization/secure_serializer.py +0 -175
  378. flock/core/serialization/serializable.py +0 -342
  379. flock/core/serialization/serialization_utils.py +0 -412
  380. flock/core/util/file_path_utils.py +0 -223
  381. flock/core/util/hydrator.py +0 -309
  382. flock/core/util/input_resolver.py +0 -164
  383. flock/core/util/loader.py +0 -59
  384. flock/core/util/splitter.py +0 -219
  385. flock/di.py +0 -27
  386. flock/platform/docker_tools.py +0 -49
  387. flock/platform/jaeger_install.py +0 -86
  388. flock/webapp/__init__.py +0 -1
  389. flock/webapp/app/__init__.py +0 -0
  390. flock/webapp/app/api/__init__.py +0 -0
  391. flock/webapp/app/api/agent_management.py +0 -241
  392. flock/webapp/app/api/execution.py +0 -709
  393. flock/webapp/app/api/flock_management.py +0 -129
  394. flock/webapp/app/api/registry_viewer.py +0 -30
  395. flock/webapp/app/chat.py +0 -665
  396. flock/webapp/app/config.py +0 -104
  397. flock/webapp/app/dependencies.py +0 -117
  398. flock/webapp/app/main.py +0 -1070
  399. flock/webapp/app/middleware.py +0 -113
  400. flock/webapp/app/models_ui.py +0 -7
  401. flock/webapp/app/services/__init__.py +0 -0
  402. flock/webapp/app/services/feedback_file_service.py +0 -363
  403. flock/webapp/app/services/flock_service.py +0 -337
  404. flock/webapp/app/services/sharing_models.py +0 -81
  405. flock/webapp/app/services/sharing_store.py +0 -762
  406. flock/webapp/app/templates/theme_mapper.html +0 -326
  407. flock/webapp/app/theme_mapper.py +0 -812
  408. flock/webapp/app/utils.py +0 -85
  409. flock/webapp/run.py +0 -215
  410. flock/webapp/static/css/chat.css +0 -301
  411. flock/webapp/static/css/components.css +0 -167
  412. flock/webapp/static/css/header.css +0 -39
  413. flock/webapp/static/css/layout.css +0 -46
  414. flock/webapp/static/css/sidebar.css +0 -127
  415. flock/webapp/static/css/two-pane.css +0 -48
  416. flock/webapp/templates/base.html +0 -200
  417. flock/webapp/templates/chat.html +0 -152
  418. flock/webapp/templates/chat_settings.html +0 -19
  419. flock/webapp/templates/flock_editor.html +0 -16
  420. flock/webapp/templates/index.html +0 -12
  421. flock/webapp/templates/partials/_agent_detail_form.html +0 -93
  422. flock/webapp/templates/partials/_agent_list.html +0 -18
  423. flock/webapp/templates/partials/_agent_manager_view.html +0 -51
  424. flock/webapp/templates/partials/_agent_tools_checklist.html +0 -14
  425. flock/webapp/templates/partials/_chat_container.html +0 -15
  426. flock/webapp/templates/partials/_chat_messages.html +0 -57
  427. flock/webapp/templates/partials/_chat_settings_form.html +0 -85
  428. flock/webapp/templates/partials/_create_flock_form.html +0 -50
  429. flock/webapp/templates/partials/_dashboard_flock_detail.html +0 -17
  430. flock/webapp/templates/partials/_dashboard_flock_file_list.html +0 -16
  431. flock/webapp/templates/partials/_dashboard_flock_properties_preview.html +0 -28
  432. flock/webapp/templates/partials/_dashboard_upload_flock_form.html +0 -16
  433. flock/webapp/templates/partials/_dynamic_input_form_content.html +0 -22
  434. flock/webapp/templates/partials/_env_vars_table.html +0 -23
  435. flock/webapp/templates/partials/_execution_form.html +0 -118
  436. flock/webapp/templates/partials/_execution_view_container.html +0 -28
  437. flock/webapp/templates/partials/_flock_file_list.html +0 -23
  438. flock/webapp/templates/partials/_flock_properties_form.html +0 -52
  439. flock/webapp/templates/partials/_flock_upload_form.html +0 -16
  440. flock/webapp/templates/partials/_header_flock_status.html +0 -5
  441. flock/webapp/templates/partials/_load_manager_view.html +0 -49
  442. flock/webapp/templates/partials/_registry_table.html +0 -25
  443. flock/webapp/templates/partials/_registry_viewer_content.html +0 -70
  444. flock/webapp/templates/partials/_results_display.html +0 -78
  445. flock/webapp/templates/partials/_settings_env_content.html +0 -9
  446. flock/webapp/templates/partials/_settings_theme_content.html +0 -14
  447. flock/webapp/templates/partials/_settings_view.html +0 -36
  448. flock/webapp/templates/partials/_share_chat_link_snippet.html +0 -11
  449. flock/webapp/templates/partials/_share_link_snippet.html +0 -35
  450. flock/webapp/templates/partials/_sidebar.html +0 -74
  451. flock/webapp/templates/partials/_streaming_results_container.html +0 -195
  452. flock/webapp/templates/partials/_structured_data_view.html +0 -40
  453. flock/webapp/templates/partials/_theme_preview.html +0 -36
  454. flock/webapp/templates/registry_viewer.html +0 -84
  455. flock/webapp/templates/shared_run_page.html +0 -140
  456. flock/workflow/__init__.py +0 -0
  457. flock/workflow/activities.py +0 -196
  458. flock/workflow/agent_activities.py +0 -24
  459. flock/workflow/agent_execution_activity.py +0 -202
  460. flock/workflow/flock_workflow.py +0 -214
  461. flock/workflow/temporal_config.py +0 -96
  462. flock/workflow/temporal_setup.py +0 -68
  463. flock_core-0.5.0b28.dist-info/METADATA +0 -274
  464. flock_core-0.5.0b28.dist-info/RECORD +0 -561
  465. flock_core-0.5.0b28.dist-info/entry_points.txt +0 -2
  466. /flock/{core/logging → logging}/formatters/themes.py +0 -0
  467. /flock/{core/logging → logging}/span_middleware/baggage_span_processor.py +0 -0
  468. /flock/{core/mcp → mcp}/util/__init__.py +0 -0
  469. {flock_core-0.5.0b28.dist-info → flock_core-0.5.0b51.dist-info}/WHEEL +0 -0
@@ -0,0 +1,640 @@
1
+ import { describe, it, expect, beforeEach } from 'vitest';
2
+ import { render, screen, waitFor } from '@testing-library/react';
3
+ import { ReactFlowProvider } from '@xyflow/react';
4
+ import { useGraphStore } from '../../store/graphStore';
5
+ import { useUIStore } from '../../store/uiStore';
6
+ import { Agent, Message } from '../../types/graph';
7
+
8
+ /**
9
+ * Phase 4: Graph Visualization & Dual Views - Integration Tests
10
+ *
11
+ * Tests full graph rendering for both Agent View and Blackboard View modes.
12
+ * Validates mode toggling performance and WebSocket event integration.
13
+ *
14
+ * SPECIFICATION: docs/specs/003-real-time-dashboard/PLAN.md Phase 4
15
+ * REQUIREMENTS:
16
+ * - Agent View renders correctly (agents as nodes, message flow edges)
17
+ * - Blackboard View renders correctly (artifacts as nodes, transformation edges)
18
+ * - Mode toggle switches between views within 100ms
19
+ * - Graph updates on new WebSocket events
20
+ */
21
+
22
+ // Mock GraphCanvas component (to be implemented)
23
+ const MockGraphCanvas = () => {
24
+ const nodes = useGraphStore((state) => state.nodes);
25
+ const edges = useGraphStore((state) => state.edges);
26
+ const mode = useUIStore((state) => state.mode);
27
+
28
+ return (
29
+ <div data-testid="graph-canvas">
30
+ <div data-testid="mode-indicator">{mode}</div>
31
+ <div data-testid="node-count">{nodes.length}</div>
32
+ <div data-testid="edge-count">{edges.length}</div>
33
+ {nodes.map((node) => (
34
+ <div key={node.id} data-testid={`node-${node.id}`} data-type={node.type}>
35
+ {node.type === 'agent' && <span>{(node.data as any).name}</span>}
36
+ {node.type === 'message' && <span>{(node.data as any).artifactType}</span>}
37
+ </div>
38
+ ))}
39
+ {edges.map((edge) => (
40
+ <div key={edge.id} data-testid={`edge-${edge.id}`}>
41
+ {edge.source} → {edge.target}
42
+ </div>
43
+ ))}
44
+ </div>
45
+ );
46
+ };
47
+
48
+ describe('Graph Rendering Integration', () => {
49
+ beforeEach(() => {
50
+ // Reset stores before each test
51
+ useGraphStore.setState({
52
+ agents: new Map(),
53
+ messages: new Map(),
54
+ events: [],
55
+ nodes: [],
56
+ edges: [],
57
+ runs: new Map(), // Phase 11: Reset runs map
58
+ consumptions: new Map(), // Phase 11: Reset consumptions map
59
+ });
60
+
61
+ useUIStore.setState({
62
+ mode: 'agent',
63
+ });
64
+ });
65
+
66
+ describe('Agent View Rendering', () => {
67
+ it('should render agents as nodes in Agent View', async () => {
68
+ const agents: Agent[] = [
69
+ {
70
+ id: 'movie-agent',
71
+ name: 'movie-agent',
72
+ status: 'idle',
73
+ subscriptions: [],
74
+ lastActive: Date.now(),
75
+ sentCount: 2,
76
+ recvCount: 0,
77
+ position: { x: 100, y: 100 },
78
+ },
79
+ {
80
+ id: 'tagline-agent',
81
+ name: 'tagline-agent',
82
+ status: 'running',
83
+ subscriptions: ['Movie'],
84
+ lastActive: Date.now(),
85
+ sentCount: 1,
86
+ recvCount: 2,
87
+ position: { x: 300, y: 100 },
88
+ },
89
+ {
90
+ id: 'summary-agent',
91
+ name: 'summary-agent',
92
+ status: 'idle',
93
+ subscriptions: ['Movie'],
94
+ lastActive: Date.now(),
95
+ sentCount: 0,
96
+ recvCount: 2,
97
+ position: { x: 500, y: 100 },
98
+ },
99
+ ];
100
+
101
+ useGraphStore.getState().batchUpdate({ agents });
102
+ useGraphStore.getState().generateAgentViewGraph();
103
+
104
+ render(
105
+ <ReactFlowProvider>
106
+ <MockGraphCanvas />
107
+ </ReactFlowProvider>
108
+ );
109
+
110
+ await waitFor(() => {
111
+ expect(screen.getByTestId('mode-indicator')).toHaveTextContent('agent');
112
+ expect(screen.getByTestId('node-count')).toHaveTextContent('3');
113
+
114
+ // Verify all agent nodes are rendered
115
+ expect(screen.getByTestId('node-movie-agent')).toBeInTheDocument();
116
+ expect(screen.getByTestId('node-tagline-agent')).toBeInTheDocument();
117
+ expect(screen.getByTestId('node-summary-agent')).toBeInTheDocument();
118
+
119
+ // Verify node types
120
+ expect(screen.getByTestId('node-movie-agent')).toHaveAttribute('data-type', 'agent');
121
+ expect(screen.getByTestId('node-tagline-agent')).toHaveAttribute('data-type', 'agent');
122
+ });
123
+ });
124
+
125
+ it('should render message flow edges in Agent View', async () => {
126
+ const agents: Agent[] = [
127
+ {
128
+ id: 'movie-agent',
129
+ name: 'movie-agent',
130
+ status: 'idle',
131
+ subscriptions: [],
132
+ lastActive: Date.now(),
133
+ sentCount: 2,
134
+ recvCount: 0,
135
+ },
136
+ {
137
+ id: 'tagline-agent',
138
+ name: 'tagline-agent',
139
+ status: 'idle',
140
+ subscriptions: ['Movie'],
141
+ lastActive: Date.now(),
142
+ sentCount: 0,
143
+ recvCount: 2,
144
+ },
145
+ ];
146
+
147
+ const messages: Message[] = [
148
+ {
149
+ id: 'msg-1',
150
+ type: 'Movie',
151
+ payload: { title: 'Inception' },
152
+ timestamp: Date.now(),
153
+ correlationId: 'corr-1',
154
+ producedBy: 'movie-agent',
155
+ },
156
+ {
157
+ id: 'msg-2',
158
+ type: 'Movie',
159
+ payload: { title: 'Interstellar' },
160
+ timestamp: Date.now(),
161
+ correlationId: 'corr-2',
162
+ producedBy: 'movie-agent',
163
+ },
164
+ ];
165
+
166
+ useGraphStore.getState().batchUpdate({ agents, messages });
167
+ // Phase 11 fix: Record consumption to populate consumed_by field
168
+ useGraphStore.getState().recordConsumption(['msg-1', 'msg-2'], 'tagline-agent');
169
+ useGraphStore.getState().generateAgentViewGraph();
170
+
171
+ render(
172
+ <ReactFlowProvider>
173
+ <MockGraphCanvas />
174
+ </ReactFlowProvider>
175
+ );
176
+
177
+ await waitFor(() => {
178
+ expect(screen.getByTestId('node-count')).toHaveTextContent('2');
179
+ expect(screen.getByTestId('edge-count')).not.toHaveTextContent('0');
180
+
181
+ // Verify edge exists between movie-agent and tagline-agent
182
+ const edges = useGraphStore.getState().edges;
183
+ const edge = edges.find(
184
+ (e) => e.source === 'movie-agent' && e.target === 'tagline-agent'
185
+ );
186
+ expect(edge).toBeDefined();
187
+ expect(edge?.label).toContain('Movie');
188
+ expect(edge?.label).toContain('2'); // Count of messages
189
+ });
190
+ });
191
+
192
+ it('should display correct message counts on edges', async () => {
193
+ const agents: Agent[] = [
194
+ {
195
+ id: 'producer',
196
+ name: 'producer',
197
+ status: 'idle',
198
+ subscriptions: [],
199
+ lastActive: Date.now(),
200
+ sentCount: 5,
201
+ recvCount: 0,
202
+ },
203
+ {
204
+ id: 'consumer',
205
+ name: 'consumer',
206
+ status: 'idle',
207
+ subscriptions: ['DataType'],
208
+ lastActive: Date.now(),
209
+ sentCount: 0,
210
+ recvCount: 5,
211
+ },
212
+ ];
213
+
214
+ const messages: Message[] = Array.from({ length: 5 }, (_, i) => ({
215
+ id: `msg-${i}`,
216
+ type: 'DataType',
217
+ payload: { index: i },
218
+ timestamp: Date.now() + i,
219
+ correlationId: `corr-${i}`,
220
+ producedBy: 'producer',
221
+ }));
222
+
223
+ useGraphStore.getState().batchUpdate({ agents, messages });
224
+ // Phase 11 fix: Record consumption to populate consumed_by field
225
+ useGraphStore.getState().recordConsumption(['msg-0', 'msg-1', 'msg-2', 'msg-3', 'msg-4'], 'consumer');
226
+ useGraphStore.getState().generateAgentViewGraph();
227
+
228
+ render(
229
+ <ReactFlowProvider>
230
+ <MockGraphCanvas />
231
+ </ReactFlowProvider>
232
+ );
233
+
234
+ await waitFor(() => {
235
+ const edges = useGraphStore.getState().edges;
236
+ expect(edges).toHaveLength(1);
237
+
238
+ const edge = edges[0];
239
+ expect(edge).toBeDefined();
240
+ expect(edge?.label).toBe('DataType (5)');
241
+ expect(edge?.data?.messageCount).toBe(5);
242
+ expect(edge?.data?.artifactIds).toHaveLength(5);
243
+ });
244
+ });
245
+ });
246
+
247
+ describe('Blackboard View Rendering', () => {
248
+ it('should render artifacts as nodes in Blackboard View', async () => {
249
+ const messages: Message[] = [
250
+ {
251
+ id: 'msg-1',
252
+ type: 'Movie',
253
+ payload: { title: 'Inception' },
254
+ timestamp: Date.now(),
255
+ correlationId: 'corr-1',
256
+ producedBy: 'movie-agent',
257
+ },
258
+ {
259
+ id: 'msg-2',
260
+ type: 'Tagline',
261
+ payload: { text: 'Dream within a dream' },
262
+ timestamp: Date.now(),
263
+ correlationId: 'corr-1',
264
+ producedBy: 'tagline-agent',
265
+ },
266
+ {
267
+ id: 'msg-3',
268
+ type: 'Summary',
269
+ payload: { text: 'A sci-fi thriller' },
270
+ timestamp: Date.now(),
271
+ correlationId: 'corr-1',
272
+ producedBy: 'summary-agent',
273
+ },
274
+ ];
275
+
276
+ useGraphStore.getState().batchUpdate({ messages });
277
+ useGraphStore.getState().generateBlackboardViewGraph();
278
+ useUIStore.setState({ mode: 'blackboard' });
279
+
280
+ render(
281
+ <ReactFlowProvider>
282
+ <MockGraphCanvas />
283
+ </ReactFlowProvider>
284
+ );
285
+
286
+ await waitFor(() => {
287
+ expect(screen.getByTestId('mode-indicator')).toHaveTextContent('blackboard');
288
+ expect(screen.getByTestId('node-count')).toHaveTextContent('3');
289
+
290
+ // Verify all message nodes are rendered
291
+ expect(screen.getByTestId('node-msg-1')).toBeInTheDocument();
292
+ expect(screen.getByTestId('node-msg-2')).toBeInTheDocument();
293
+ expect(screen.getByTestId('node-msg-3')).toBeInTheDocument();
294
+
295
+ // Verify node types
296
+ expect(screen.getByTestId('node-msg-1')).toHaveAttribute('data-type', 'message');
297
+ });
298
+ });
299
+
300
+ it('should render transformation edges in Blackboard View', async () => {
301
+ // Note: This test will need actual transformation edge logic
302
+ // For now, we test the basic structure
303
+
304
+ const messages: Message[] = [
305
+ {
306
+ id: 'msg-1',
307
+ type: 'Movie',
308
+ payload: { title: 'Inception' },
309
+ timestamp: Date.now(),
310
+ correlationId: 'corr-1',
311
+ producedBy: 'movie-agent',
312
+ },
313
+ {
314
+ id: 'msg-2',
315
+ type: 'Tagline',
316
+ payload: { text: 'Dream within a dream' },
317
+ timestamp: Date.now(),
318
+ correlationId: 'corr-1',
319
+ producedBy: 'tagline-agent',
320
+ },
321
+ ];
322
+
323
+ useGraphStore.getState().batchUpdate({ messages });
324
+ useGraphStore.getState().generateBlackboardViewGraph();
325
+ useUIStore.setState({ mode: 'blackboard' });
326
+
327
+ render(
328
+ <ReactFlowProvider>
329
+ <MockGraphCanvas />
330
+ </ReactFlowProvider>
331
+ );
332
+
333
+ await waitFor(() => {
334
+ expect(screen.getByTestId('mode-indicator')).toHaveTextContent('blackboard');
335
+ expect(screen.getByTestId('node-count')).toHaveTextContent('2');
336
+
337
+ // Currently generateBlackboardViewGraph doesn't create edges
338
+ // This will be implemented with transforms.ts
339
+ const edges = useGraphStore.getState().edges;
340
+ expect(edges).toHaveLength(0); // Will change after implementation
341
+ });
342
+ });
343
+
344
+ it('should display artifact types correctly', async () => {
345
+ const messages: Message[] = [
346
+ {
347
+ id: 'msg-1',
348
+ type: 'Movie',
349
+ payload: { title: 'Inception', year: 2010 },
350
+ timestamp: Date.now(),
351
+ correlationId: 'corr-1',
352
+ producedBy: 'movie-agent',
353
+ },
354
+ ];
355
+
356
+ useGraphStore.getState().batchUpdate({ messages });
357
+ useGraphStore.getState().generateBlackboardViewGraph();
358
+ useUIStore.setState({ mode: 'blackboard' });
359
+
360
+ render(
361
+ <ReactFlowProvider>
362
+ <MockGraphCanvas />
363
+ </ReactFlowProvider>
364
+ );
365
+
366
+ await waitFor(() => {
367
+ const nodes = useGraphStore.getState().nodes;
368
+ expect(nodes).toHaveLength(1);
369
+ expect(nodes[0]).toBeDefined();
370
+ expect((nodes[0]?.data as any).artifactType).toBe('Movie');
371
+ expect(screen.getByText('Movie')).toBeInTheDocument();
372
+ });
373
+ });
374
+ });
375
+
376
+ describe('Mode Toggle Performance', () => {
377
+ it('should switch between views within 100ms (REQUIREMENT)', async () => {
378
+ // Setup both agent and message data
379
+ const agents: Agent[] = [
380
+ {
381
+ id: 'agent-1',
382
+ name: 'agent-1',
383
+ status: 'idle',
384
+ subscriptions: [],
385
+ lastActive: Date.now(),
386
+ sentCount: 1,
387
+ recvCount: 0,
388
+ },
389
+ ];
390
+
391
+ const messages: Message[] = [
392
+ {
393
+ id: 'msg-1',
394
+ type: 'TestType',
395
+ payload: {},
396
+ timestamp: Date.now(),
397
+ correlationId: 'corr-1',
398
+ producedBy: 'agent-1',
399
+ },
400
+ ];
401
+
402
+ useGraphStore.getState().batchUpdate({ agents, messages });
403
+
404
+ // Measure Agent View generation
405
+ const agentStartTime = performance.now();
406
+ useGraphStore.getState().generateAgentViewGraph();
407
+ const agentEndTime = performance.now();
408
+ const agentDuration = agentEndTime - agentStartTime;
409
+
410
+ expect(agentDuration).toBeLessThan(100); // REQUIREMENT
411
+
412
+ // Measure Blackboard View generation
413
+ const blackboardStartTime = performance.now();
414
+ useGraphStore.getState().generateBlackboardViewGraph();
415
+ const blackboardEndTime = performance.now();
416
+ const blackboardDuration = blackboardEndTime - blackboardStartTime;
417
+
418
+ expect(blackboardDuration).toBeLessThan(100); // REQUIREMENT
419
+
420
+ // Verify both views generated correctly
421
+ const agentNodes = useGraphStore.getState().nodes;
422
+ expect(agentNodes.length).toBeGreaterThan(0);
423
+ });
424
+
425
+ it('should toggle mode quickly with UI store update', async () => {
426
+ const startMode = useUIStore.getState().mode;
427
+ expect(startMode).toBe('agent');
428
+
429
+ const startTime = performance.now();
430
+ useUIStore.getState().setMode('blackboard');
431
+ const endTime = performance.now();
432
+ const duration = endTime - startTime;
433
+
434
+ expect(duration).toBeLessThan(10); // Mode toggle should be instant
435
+ expect(useUIStore.getState().mode).toBe('blackboard');
436
+
437
+ // Toggle back
438
+ useUIStore.getState().setMode('agent');
439
+ expect(useUIStore.getState().mode).toBe('agent');
440
+ });
441
+
442
+ it('should handle rapid mode toggling', async () => {
443
+ const agents: Agent[] = [
444
+ {
445
+ id: 'agent-1',
446
+ name: 'agent-1',
447
+ status: 'idle',
448
+ subscriptions: [],
449
+ lastActive: Date.now(),
450
+ sentCount: 0,
451
+ recvCount: 0,
452
+ },
453
+ ];
454
+
455
+ useGraphStore.getState().batchUpdate({ agents });
456
+
457
+ // Rapidly toggle modes
458
+ const iterations = 10;
459
+ const startTime = performance.now();
460
+
461
+ for (let i = 0; i < iterations; i++) {
462
+ const newMode = i % 2 === 0 ? 'blackboard' : 'agent';
463
+ useUIStore.getState().setMode(newMode);
464
+
465
+ if (useUIStore.getState().mode === 'agent') {
466
+ useGraphStore.getState().generateAgentViewGraph();
467
+ } else {
468
+ useGraphStore.getState().generateBlackboardViewGraph();
469
+ }
470
+ }
471
+
472
+ const endTime = performance.now();
473
+ const totalDuration = endTime - startTime;
474
+ const avgDuration = totalDuration / iterations;
475
+
476
+ expect(avgDuration).toBeLessThan(100); // Average should be under 100ms
477
+ });
478
+ });
479
+
480
+ describe('Graph Updates on WebSocket Events', () => {
481
+ it('should update graph when new agent is added', async () => {
482
+ useGraphStore.getState().generateAgentViewGraph();
483
+
484
+ render(
485
+ <ReactFlowProvider>
486
+ <MockGraphCanvas />
487
+ </ReactFlowProvider>
488
+ );
489
+
490
+ expect(screen.getByTestId('node-count')).toHaveTextContent('0');
491
+
492
+ // Simulate WebSocket event adding an agent
493
+ const newAgent: Agent = {
494
+ id: 'new-agent',
495
+ name: 'new-agent',
496
+ status: 'running',
497
+ subscriptions: ['TestType'],
498
+ lastActive: Date.now(),
499
+ sentCount: 0,
500
+ recvCount: 0,
501
+ };
502
+
503
+ useGraphStore.getState().addAgent(newAgent);
504
+ useGraphStore.getState().generateAgentViewGraph();
505
+
506
+ await waitFor(() => {
507
+ expect(screen.getByTestId('node-count')).toHaveTextContent('1');
508
+ expect(screen.getByTestId('node-new-agent')).toBeInTheDocument();
509
+ });
510
+ });
511
+
512
+ it('should update graph when new message is published', async () => {
513
+ const agents: Agent[] = [
514
+ {
515
+ id: 'producer',
516
+ name: 'producer',
517
+ status: 'idle',
518
+ subscriptions: [],
519
+ lastActive: Date.now(),
520
+ sentCount: 0,
521
+ recvCount: 0,
522
+ },
523
+ {
524
+ id: 'consumer',
525
+ name: 'consumer',
526
+ status: 'idle',
527
+ subscriptions: ['TestType'],
528
+ lastActive: Date.now(),
529
+ sentCount: 0,
530
+ recvCount: 0,
531
+ },
532
+ ];
533
+
534
+ useGraphStore.getState().batchUpdate({ agents });
535
+ useGraphStore.getState().generateAgentViewGraph();
536
+
537
+ render(
538
+ <ReactFlowProvider>
539
+ <MockGraphCanvas />
540
+ </ReactFlowProvider>
541
+ );
542
+
543
+ const initialEdgeCount = useGraphStore.getState().edges.length;
544
+
545
+ // Simulate WebSocket event publishing a message
546
+ const newMessage: Message = {
547
+ id: 'new-msg',
548
+ type: 'TestType',
549
+ payload: { data: 'test' },
550
+ timestamp: Date.now(),
551
+ correlationId: 'corr-1',
552
+ producedBy: 'producer',
553
+ };
554
+
555
+ useGraphStore.getState().addMessage(newMessage);
556
+ // Phase 11 fix: Record consumption to create edge
557
+ useGraphStore.getState().recordConsumption(['new-msg'], 'consumer');
558
+ useGraphStore.getState().generateAgentViewGraph();
559
+
560
+ await waitFor(() => {
561
+ const newEdgeCount = useGraphStore.getState().edges.length;
562
+ expect(newEdgeCount).toBeGreaterThan(initialEdgeCount);
563
+ });
564
+ });
565
+
566
+ it('should handle incremental updates efficiently', async () => {
567
+ const initialAgents: Agent[] = Array.from({ length: 5 }, (_, i) => ({
568
+ id: `agent-${i}`,
569
+ name: `agent-${i}`,
570
+ status: 'idle' as const,
571
+ subscriptions: [],
572
+ lastActive: Date.now(),
573
+ sentCount: 0,
574
+ recvCount: 0,
575
+ }));
576
+
577
+ useGraphStore.getState().batchUpdate({ agents: initialAgents });
578
+ useGraphStore.getState().generateAgentViewGraph();
579
+
580
+ // Measure incremental update performance
581
+ const startTime = performance.now();
582
+
583
+ const newAgent: Agent = {
584
+ id: 'agent-new',
585
+ name: 'agent-new',
586
+ status: 'running',
587
+ subscriptions: [],
588
+ lastActive: Date.now(),
589
+ sentCount: 0,
590
+ recvCount: 0,
591
+ };
592
+
593
+ useGraphStore.getState().addAgent(newAgent);
594
+ useGraphStore.getState().generateAgentViewGraph();
595
+
596
+ const endTime = performance.now();
597
+ const duration = endTime - startTime;
598
+
599
+ // Incremental update should be very fast (<50ms)
600
+ expect(duration).toBeLessThan(50);
601
+
602
+ const nodes = useGraphStore.getState().nodes;
603
+ expect(nodes).toHaveLength(6);
604
+ });
605
+ });
606
+
607
+ describe('Empty States', () => {
608
+ it('should render empty Agent View gracefully', async () => {
609
+ useGraphStore.getState().generateAgentViewGraph();
610
+
611
+ render(
612
+ <ReactFlowProvider>
613
+ <MockGraphCanvas />
614
+ </ReactFlowProvider>
615
+ );
616
+
617
+ await waitFor(() => {
618
+ expect(screen.getByTestId('node-count')).toHaveTextContent('0');
619
+ expect(screen.getByTestId('edge-count')).toHaveTextContent('0');
620
+ });
621
+ });
622
+
623
+ it('should render empty Blackboard View gracefully', async () => {
624
+ useGraphStore.getState().generateBlackboardViewGraph();
625
+ useUIStore.setState({ mode: 'blackboard' });
626
+
627
+ render(
628
+ <ReactFlowProvider>
629
+ <MockGraphCanvas />
630
+ </ReactFlowProvider>
631
+ );
632
+
633
+ await waitFor(() => {
634
+ expect(screen.getByTestId('mode-indicator')).toHaveTextContent('blackboard');
635
+ expect(screen.getByTestId('node-count')).toHaveTextContent('0');
636
+ expect(screen.getByTestId('edge-count')).toHaveTextContent('0');
637
+ });
638
+ });
639
+ });
640
+ });