flock-core 0.4.542__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.542.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.542.dist-info/METADATA +0 -676
  496. flock_core-0.4.542.dist-info/RECORD +0 -572
  497. flock_core-0.4.542.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.542.dist-info → flock_core-0.5.0.dist-info}/WHEEL +0 -0
@@ -0,0 +1,647 @@
1
+ /**
2
+ * Phase 5: Integration Tests for Backend Snapshot Consumption
3
+ *
4
+ * Tests the complete flow from backend API to UI rendering, including:
5
+ * - Initial graph loading from backend
6
+ * - WebSocket-triggered debounced refreshes
7
+ * - Position persistence
8
+ * - Filter application
9
+ * - Error handling
10
+ * - Empty states
11
+ */
12
+
13
+ import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
14
+ import { act } from '@testing-library/react';
15
+ import { useGraphStore } from '../../store/graphStore';
16
+ import { useFilterStore } from '../../store/filterStore';
17
+ import { GraphSnapshot } from '../../types/graph';
18
+
19
+ // Mock the graph service
20
+ vi.mock('../../services/graphService', () => ({
21
+ fetchGraphSnapshot: vi.fn(),
22
+ mergeNodePositions: vi.fn((backendNodes) => backendNodes),
23
+ overlayWebSocketState: vi.fn((nodes) => nodes),
24
+ }));
25
+
26
+ // Mock IndexedDB for persistence tests
27
+ const mockIndexedDB = {
28
+ open: vi.fn(),
29
+ databases: vi.fn().mockResolvedValue([]),
30
+ };
31
+ globalThis.indexedDB = mockIndexedDB as any;
32
+
33
+ import * as graphService from '../../services/graphService';
34
+
35
+ describe('Graph Snapshot Integration', () => {
36
+ // Sample backend snapshot fixture
37
+ const createMockSnapshot = (overrides?: Partial<GraphSnapshot>): GraphSnapshot => ({
38
+ generatedAt: '2025-10-11T00:00:00Z',
39
+ viewMode: 'agent',
40
+ filters: {
41
+ correlation_id: null,
42
+ time_range: { preset: 'last10min' },
43
+ artifactTypes: [],
44
+ producers: [],
45
+ tags: [],
46
+ visibility: [],
47
+ },
48
+ nodes: [
49
+ {
50
+ id: 'pizza_master',
51
+ type: 'agent',
52
+ data: { name: 'pizza_master', status: 'idle' },
53
+ position: { x: 100, y: 100 },
54
+ hidden: false,
55
+ },
56
+ {
57
+ id: 'topping_picker',
58
+ type: 'agent',
59
+ data: { name: 'topping_picker', status: 'idle' },
60
+ position: { x: 300, y: 200 },
61
+ hidden: false,
62
+ },
63
+ ],
64
+ edges: [
65
+ {
66
+ id: 'edge-1',
67
+ source: 'pizza_master',
68
+ target: 'topping_picker',
69
+ type: 'message_flow',
70
+ label: 'Pizza',
71
+ data: {},
72
+ hidden: false,
73
+ },
74
+ ],
75
+ statistics: {
76
+ producedByAgent: {
77
+ pizza_master: { total: 5, byType: { Pizza: 5 } },
78
+ },
79
+ consumedByAgent: {
80
+ topping_picker: { total: 5, byType: { Pizza: 5 } },
81
+ },
82
+ artifactSummary: {
83
+ total: 10,
84
+ by_type: { Pizza: 10 },
85
+ by_producer: { pizza_master: 10 },
86
+ by_visibility: { public: 10 },
87
+ tag_counts: {},
88
+ earliest_created_at: '2025-10-11T00:00:00Z',
89
+ latest_created_at: '2025-10-11T00:05:00Z',
90
+ },
91
+ },
92
+ totalArtifacts: 10,
93
+ truncated: false,
94
+ ...overrides,
95
+ });
96
+
97
+ beforeEach(() => {
98
+ // Reset all stores before each test
99
+ useGraphStore.setState({
100
+ agentStatus: new Map(),
101
+ streamingTokens: new Map(),
102
+ nodes: [],
103
+ edges: [],
104
+ statistics: null,
105
+ events: [],
106
+ viewMode: 'agent',
107
+ isLoading: false,
108
+ error: null,
109
+ });
110
+
111
+ useFilterStore.setState({
112
+ correlationId: null,
113
+ timeRange: { preset: 'last10min' },
114
+ selectedArtifactTypes: [],
115
+ selectedProducers: [],
116
+ selectedTags: [],
117
+ selectedVisibility: [],
118
+ availableArtifactTypes: [],
119
+ availableProducers: [],
120
+ availableTags: [],
121
+ availableVisibility: [],
122
+ });
123
+
124
+ vi.clearAllMocks();
125
+ });
126
+
127
+ afterEach(() => {
128
+ vi.restoreAllMocks();
129
+ });
130
+
131
+ describe('Test 1: Initial Graph Loading', () => {
132
+ it('should fetch agent view graph from backend on mount', async () => {
133
+ const mockSnapshot = createMockSnapshot();
134
+ vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
135
+
136
+ // Trigger graph generation
137
+ await act(async () => {
138
+ await useGraphStore.getState().generateAgentViewGraph();
139
+ });
140
+
141
+ // Verify backend was called with correct request
142
+ expect(graphService.fetchGraphSnapshot).toHaveBeenCalledWith({
143
+ viewMode: 'agent',
144
+ filters: {
145
+ correlation_id: null,
146
+ time_range: { preset: 'last10min' },
147
+ artifactTypes: [],
148
+ producers: [],
149
+ tags: [],
150
+ visibility: [],
151
+ },
152
+ options: { include_statistics: true },
153
+ });
154
+
155
+ // Verify store was updated with snapshot data
156
+ const state = useGraphStore.getState();
157
+ expect(state.nodes).toHaveLength(2);
158
+ expect(state.edges).toHaveLength(1);
159
+ expect(state.statistics).toBeTruthy();
160
+ expect(state.viewMode).toBe('agent');
161
+ });
162
+
163
+ it('should fetch blackboard view graph from backend', async () => {
164
+ const mockSnapshot = createMockSnapshot({ viewMode: 'blackboard' });
165
+ vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
166
+
167
+ await act(async () => {
168
+ await useGraphStore.getState().generateBlackboardViewGraph();
169
+ });
170
+
171
+ expect(graphService.fetchGraphSnapshot).toHaveBeenCalledWith({
172
+ viewMode: 'blackboard',
173
+ filters: expect.any(Object),
174
+ options: { include_statistics: true },
175
+ });
176
+
177
+ const state = useGraphStore.getState();
178
+ expect(state.viewMode).toBe('blackboard');
179
+ });
180
+
181
+ it('should call fetchGraphSnapshot only once on initial load', async () => {
182
+ const mockSnapshot = createMockSnapshot();
183
+ vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
184
+
185
+ await act(async () => {
186
+ await useGraphStore.getState().generateAgentViewGraph();
187
+ });
188
+
189
+ expect(graphService.fetchGraphSnapshot).toHaveBeenCalledTimes(1);
190
+ });
191
+ });
192
+
193
+ describe('Test 2: WebSocket Event Handling', () => {
194
+ it('should update agent status immediately without backend fetch', () => {
195
+ // Set up initial nodes
196
+ useGraphStore.setState({
197
+ nodes: [
198
+ {
199
+ id: 'pizza_master',
200
+ type: 'agent',
201
+ data: { name: 'pizza_master', status: 'idle' },
202
+ position: { x: 100, y: 100 },
203
+ },
204
+ ],
205
+ });
206
+
207
+ // Update status (simulating WebSocket event)
208
+ act(() => {
209
+ useGraphStore.getState().updateAgentStatus('pizza_master', 'running');
210
+ });
211
+
212
+ // Verify status updated immediately
213
+ const state = useGraphStore.getState();
214
+ const node = state.nodes.find((n) => n.id === 'pizza_master');
215
+ expect(node?.data.status).toBe('running');
216
+
217
+ // Verify NO backend fetch happened (fast path)
218
+ expect(graphService.fetchGraphSnapshot).not.toHaveBeenCalled();
219
+ });
220
+
221
+ it('should update streaming tokens without backend fetch', () => {
222
+ useGraphStore.setState({
223
+ nodes: [
224
+ {
225
+ id: 'pizza_master',
226
+ type: 'agent',
227
+ data: { name: 'pizza_master', streamingTokens: [] },
228
+ position: { x: 100, y: 100 },
229
+ },
230
+ ],
231
+ });
232
+
233
+ act(() => {
234
+ useGraphStore.getState().updateStreamingTokens('pizza_master', ['token1', 'token2', 'token3']);
235
+ });
236
+
237
+ const state = useGraphStore.getState();
238
+ const node = state.nodes.find((n) => n.id === 'pizza_master');
239
+ expect(node?.data.streamingTokens).toEqual(['token1', 'token2', 'token3']);
240
+
241
+ // No backend fetch for streaming tokens (fast path)
242
+ expect(graphService.fetchGraphSnapshot).not.toHaveBeenCalled();
243
+ });
244
+
245
+ it('should keep only last 6 streaming tokens', () => {
246
+ useGraphStore.setState({
247
+ nodes: [
248
+ {
249
+ id: 'pizza_master',
250
+ type: 'agent',
251
+ data: { streamingTokens: [] },
252
+ position: { x: 100, y: 100 },
253
+ },
254
+ ],
255
+ });
256
+
257
+ const tokens = ['t1', 't2', 't3', 't4', 't5', 't6', 't7', 't8'];
258
+ act(() => {
259
+ useGraphStore.getState().updateStreamingTokens('pizza_master', tokens);
260
+ });
261
+
262
+ const state = useGraphStore.getState();
263
+ const node = state.nodes.find((n) => n.id === 'pizza_master');
264
+ expect(node?.data.streamingTokens).toHaveLength(6);
265
+ expect(node?.data.streamingTokens).toEqual(['t3', 't4', 't5', 't6', 't7', 't8']);
266
+ });
267
+ });
268
+
269
+ describe('Test 3: View Refresh', () => {
270
+ it('should call appropriate view generator based on viewMode', async () => {
271
+ const mockSnapshot = createMockSnapshot();
272
+ vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
273
+
274
+ // Set agent view mode
275
+ useGraphStore.setState({ viewMode: 'agent' });
276
+
277
+ await act(async () => {
278
+ await useGraphStore.getState().refreshCurrentView();
279
+ });
280
+
281
+ // Should call agent view
282
+ expect(graphService.fetchGraphSnapshot).toHaveBeenCalledWith(
283
+ expect.objectContaining({ viewMode: 'agent' })
284
+ );
285
+ });
286
+
287
+ it('should refresh blackboard view when viewMode is blackboard', async () => {
288
+ const mockSnapshot = createMockSnapshot({ viewMode: 'blackboard' });
289
+ vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
290
+
291
+ // Set blackboard view mode
292
+ useGraphStore.setState({ viewMode: 'blackboard' });
293
+
294
+ await act(async () => {
295
+ await useGraphStore.getState().refreshCurrentView();
296
+ });
297
+
298
+ expect(graphService.fetchGraphSnapshot).toHaveBeenCalledWith(
299
+ expect.objectContaining({ viewMode: 'blackboard' })
300
+ );
301
+ });
302
+
303
+ it('should accumulate events from multiple addEvent calls', () => {
304
+ // Simulate multiple message published events
305
+ act(() => {
306
+ useGraphStore.getState().addEvent({
307
+ id: 'msg1',
308
+ type: 'Pizza',
309
+ producedBy: 'pizza_master',
310
+ timestamp: Date.now(),
311
+ } as any);
312
+
313
+ useGraphStore.getState().addEvent({
314
+ id: 'msg2',
315
+ type: 'Pizza',
316
+ producedBy: 'pizza_master',
317
+ timestamp: Date.now(),
318
+ } as any);
319
+
320
+ useGraphStore.getState().addEvent({
321
+ id: 'msg3',
322
+ type: 'Pizza',
323
+ producedBy: 'pizza_master',
324
+ timestamp: Date.now(),
325
+ } as any);
326
+ });
327
+
328
+ // Verify all 3 events in store
329
+ const state = useGraphStore.getState();
330
+ expect(state.events).toHaveLength(3);
331
+ });
332
+ });
333
+
334
+ describe('Test 4: Position Persistence', () => {
335
+ it('should merge saved positions with backend nodes', async () => {
336
+ // Mock persistence to return saved positions
337
+ vi.mocked(graphService.mergeNodePositions).mockImplementation(
338
+ (backendNodes, saved) => {
339
+ return backendNodes.map((node) => ({
340
+ ...node,
341
+ position: saved.get(node.id) || node.position,
342
+ }));
343
+ }
344
+ );
345
+
346
+ const mockSnapshot = createMockSnapshot();
347
+ vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
348
+
349
+ await act(async () => {
350
+ await useGraphStore.getState().generateAgentViewGraph();
351
+ });
352
+
353
+ // Verify mergeNodePositions was called with saved positions
354
+ expect(graphService.mergeNodePositions).toHaveBeenCalled();
355
+ });
356
+
357
+ it('should update node position when user drags node', () => {
358
+ useGraphStore.setState({
359
+ nodes: [
360
+ {
361
+ id: 'pizza_master',
362
+ type: 'agent',
363
+ data: {},
364
+ position: { x: 100, y: 100 },
365
+ },
366
+ ],
367
+ });
368
+
369
+ // Simulate user dragging node to new position
370
+ act(() => {
371
+ useGraphStore.getState().updateNodePosition('pizza_master', { x: 250, y: 350 });
372
+ });
373
+
374
+ // Verify position updated in store
375
+ const state = useGraphStore.getState();
376
+ const node = state.nodes.find((n) => n.id === 'pizza_master');
377
+ expect(node?.position).toEqual({ x: 250, y: 350 });
378
+ });
379
+ });
380
+
381
+ describe('Test 5: Filter Application', () => {
382
+ it('should trigger backend fetch when filters applied', async () => {
383
+ const mockSnapshot = createMockSnapshot();
384
+ vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
385
+
386
+ // Update filters using correct method names
387
+ act(() => {
388
+ useFilterStore.getState().setArtifactTypes(['Pizza']);
389
+ useFilterStore.getState().setProducers(['pizza_master']);
390
+ });
391
+
392
+ // Apply filters
393
+ await act(async () => {
394
+ await useFilterStore.getState().applyFilters();
395
+ });
396
+
397
+ // Verify backend called with updated filters
398
+ expect(graphService.fetchGraphSnapshot).toHaveBeenCalledWith({
399
+ viewMode: 'agent',
400
+ filters: {
401
+ correlation_id: null,
402
+ time_range: expect.objectContaining({ preset: 'last10min' }),
403
+ artifactTypes: ['Pizza'],
404
+ producers: ['pizza_master'],
405
+ tags: [],
406
+ visibility: [],
407
+ },
408
+ options: { include_statistics: true },
409
+ });
410
+ });
411
+
412
+ it('should update available facets from backend statistics', async () => {
413
+ const mockSnapshot = createMockSnapshot();
414
+ vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
415
+
416
+ await act(async () => {
417
+ await useGraphStore.getState().generateAgentViewGraph();
418
+ });
419
+
420
+ // Verify filter facets updated from backend
421
+ const filterState = useFilterStore.getState();
422
+ expect(filterState.availableArtifactTypes).toContain('Pizza');
423
+ expect(filterState.availableProducers).toContain('pizza_master');
424
+ expect(filterState.availableVisibility).toContain('public');
425
+ });
426
+ });
427
+
428
+ describe('Test 6: Error Handling', () => {
429
+ it('should handle backend API errors gracefully', async () => {
430
+ vi.mocked(graphService.fetchGraphSnapshot).mockRejectedValue(
431
+ new Error('Backend API unavailable')
432
+ );
433
+
434
+ // Verify error is thrown
435
+ await expect(async () => {
436
+ await act(async () => {
437
+ await useGraphStore.getState().generateAgentViewGraph();
438
+ });
439
+ }).rejects.toThrow('Backend API unavailable');
440
+
441
+ // Verify error state was set
442
+ const state = useGraphStore.getState();
443
+ expect(state.error).toBe('Backend API unavailable');
444
+ expect(state.isLoading).toBe(false);
445
+ });
446
+
447
+ it('should not crash when fetchGraphSnapshot returns invalid data', async () => {
448
+ // Mock returns empty snapshot
449
+ vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue({
450
+ generatedAt: '2025-10-11T00:00:00Z',
451
+ viewMode: 'agent',
452
+ filters: {
453
+ correlation_id: null,
454
+ time_range: { preset: 'last10min' },
455
+ artifactTypes: [],
456
+ producers: [],
457
+ tags: [],
458
+ visibility: [],
459
+ },
460
+ nodes: [],
461
+ edges: [],
462
+ statistics: null,
463
+ totalArtifacts: 0,
464
+ truncated: false,
465
+ });
466
+
467
+ await act(async () => {
468
+ await useGraphStore.getState().generateAgentViewGraph();
469
+ });
470
+
471
+ // Should handle empty data gracefully
472
+ const state = useGraphStore.getState();
473
+ expect(state.nodes).toEqual([]);
474
+ expect(state.edges).toEqual([]);
475
+ });
476
+ });
477
+
478
+ describe('Test 7: Empty State', () => {
479
+ it('should handle empty graph from backend', async () => {
480
+ const emptySnapshot = createMockSnapshot({
481
+ nodes: [],
482
+ edges: [],
483
+ statistics: null,
484
+ totalArtifacts: 0,
485
+ });
486
+
487
+ vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(emptySnapshot);
488
+
489
+ await act(async () => {
490
+ await useGraphStore.getState().generateAgentViewGraph();
491
+ });
492
+
493
+ const state = useGraphStore.getState();
494
+ expect(state.nodes).toEqual([]);
495
+ expect(state.edges).toEqual([]);
496
+ expect(state.statistics).toBeNull();
497
+ });
498
+
499
+ it('should clear events when limit exceeded', () => {
500
+ useGraphStore.setState({ events: [] });
501
+
502
+ // Add 101 events (limit is 100)
503
+ act(() => {
504
+ for (let i = 0; i < 101; i++) {
505
+ useGraphStore.getState().addEvent({
506
+ id: `msg${i}`,
507
+ type: 'Pizza',
508
+ producedBy: 'pizza_master',
509
+ timestamp: Date.now() + i,
510
+ } as any);
511
+ }
512
+ });
513
+
514
+ // Should keep only last 100
515
+ const state = useGraphStore.getState();
516
+ expect(state.events).toHaveLength(100);
517
+ expect(state.events[0]?.id).toBe('msg100'); // Most recent first
518
+ });
519
+ });
520
+
521
+ describe('Test 8: View Mode Switching', () => {
522
+ it('should switch between agent and blackboard views', async () => {
523
+ const agentSnapshot = createMockSnapshot({ viewMode: 'agent' });
524
+ const blackboardSnapshot = createMockSnapshot({ viewMode: 'blackboard' });
525
+
526
+ vi.mocked(graphService.fetchGraphSnapshot)
527
+ .mockResolvedValueOnce(agentSnapshot)
528
+ .mockResolvedValueOnce(blackboardSnapshot);
529
+
530
+ // Load agent view
531
+ await act(async () => {
532
+ await useGraphStore.getState().generateAgentViewGraph();
533
+ });
534
+
535
+ expect(useGraphStore.getState().viewMode).toBe('agent');
536
+
537
+ // Switch to blackboard view
538
+ await act(async () => {
539
+ await useGraphStore.getState().generateBlackboardViewGraph();
540
+ });
541
+
542
+ expect(useGraphStore.getState().viewMode).toBe('blackboard');
543
+ expect(graphService.fetchGraphSnapshot).toHaveBeenCalledTimes(2);
544
+ });
545
+
546
+ it('should update viewMode state', () => {
547
+ act(() => {
548
+ useGraphStore.getState().setViewMode('blackboard');
549
+ });
550
+
551
+ expect(useGraphStore.getState().viewMode).toBe('blackboard');
552
+
553
+ act(() => {
554
+ useGraphStore.getState().setViewMode('agent');
555
+ });
556
+
557
+ expect(useGraphStore.getState().viewMode).toBe('agent');
558
+ });
559
+ });
560
+
561
+ describe('Test 9: Debounced Refresh (Critical Optimization)', () => {
562
+ beforeEach(() => {
563
+ vi.useFakeTimers();
564
+ });
565
+
566
+ afterEach(() => {
567
+ vi.useRealTimers();
568
+ });
569
+
570
+ it('should batch multiple rapid events into single backend fetch after 100ms', async () => {
571
+ const mockSnapshot = createMockSnapshot();
572
+ vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
573
+
574
+ // Load initial graph
575
+ await act(async () => {
576
+ await useGraphStore.getState().generateAgentViewGraph();
577
+ });
578
+
579
+ vi.clearAllMocks();
580
+
581
+ // Simulate 5 rapid scheduleRefresh() calls (like rapid WebSocket events)
582
+ act(() => {
583
+ for (let i = 0; i < 5; i++) {
584
+ useGraphStore.getState().scheduleRefresh();
585
+ }
586
+ });
587
+
588
+ // No fetch yet (debounce delay)
589
+ expect(graphService.fetchGraphSnapshot).not.toHaveBeenCalled();
590
+
591
+ // Advance timers by 100ms (debounce threshold)
592
+ await act(async () => {
593
+ vi.advanceTimersByTime(100);
594
+ // Wait for any pending promises
595
+ await Promise.resolve();
596
+ });
597
+
598
+ // Should have triggered exactly ONE backend fetch (batching worked!)
599
+ expect(graphService.fetchGraphSnapshot).toHaveBeenCalledTimes(1);
600
+ });
601
+
602
+ it('should reset debounce timer if new event arrives within 100ms', async () => {
603
+ const mockSnapshot = createMockSnapshot();
604
+ vi.mocked(graphService.fetchGraphSnapshot).mockResolvedValue(mockSnapshot);
605
+
606
+ await act(async () => {
607
+ await useGraphStore.getState().generateAgentViewGraph();
608
+ });
609
+
610
+ vi.clearAllMocks();
611
+
612
+ // First scheduleRefresh call
613
+ act(() => {
614
+ useGraphStore.getState().scheduleRefresh();
615
+ });
616
+
617
+ // Advance 50ms (not enough to trigger)
618
+ act(() => {
619
+ vi.advanceTimersByTime(50);
620
+ });
621
+
622
+ expect(graphService.fetchGraphSnapshot).not.toHaveBeenCalled();
623
+
624
+ // Second scheduleRefresh call (resets timer)
625
+ act(() => {
626
+ useGraphStore.getState().scheduleRefresh();
627
+ });
628
+
629
+ // Advance another 50ms (100ms total, but timer was reset at 50ms)
630
+ act(() => {
631
+ vi.advanceTimersByTime(50);
632
+ });
633
+
634
+ // Still no fetch (timer was reset)
635
+ expect(graphService.fetchGraphSnapshot).not.toHaveBeenCalled();
636
+
637
+ // Advance final 50ms (100ms since last scheduleRefresh)
638
+ await act(async () => {
639
+ vi.advanceTimersByTime(50);
640
+ await Promise.resolve();
641
+ });
642
+
643
+ // Now it should fetch (100ms of quiet time)
644
+ expect(graphService.fetchGraphSnapshot).toHaveBeenCalledTimes(1);
645
+ });
646
+ });
647
+ });