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.
- flock/__init__.py +12 -217
- flock/agent.py +1079 -0
- flock/api/themes.py +71 -0
- flock/artifacts.py +86 -0
- flock/cli.py +147 -0
- flock/components.py +189 -0
- flock/dashboard/__init__.py +30 -0
- flock/dashboard/collector.py +559 -0
- flock/dashboard/events.py +188 -0
- flock/dashboard/graph_builder.py +563 -0
- flock/dashboard/launcher.py +235 -0
- flock/dashboard/models/graph.py +156 -0
- flock/dashboard/service.py +991 -0
- flock/dashboard/static_v2/assets/index-DFRnI_mt.js +111 -0
- flock/dashboard/static_v2/assets/index-fPLNdmp1.css +1 -0
- flock/dashboard/static_v2/index.html +13 -0
- flock/dashboard/websocket.py +246 -0
- flock/engines/__init__.py +6 -0
- flock/engines/dspy_engine.py +932 -0
- flock/examples.py +131 -0
- flock/frontend/README.md +778 -0
- flock/frontend/docs/DESIGN_SYSTEM.md +1980 -0
- flock/frontend/index.html +12 -0
- flock/frontend/package-lock.json +4337 -0
- flock/frontend/package.json +48 -0
- flock/frontend/src/App.tsx +139 -0
- flock/frontend/src/__tests__/integration/graph-snapshot.test.tsx +647 -0
- flock/frontend/src/__tests__/integration/indexeddb-persistence.test.tsx +699 -0
- flock/frontend/src/components/common/BuildInfo.tsx +39 -0
- flock/frontend/src/components/common/EmptyState.module.css +115 -0
- flock/frontend/src/components/common/EmptyState.tsx +128 -0
- flock/frontend/src/components/common/ErrorBoundary.module.css +169 -0
- flock/frontend/src/components/common/ErrorBoundary.tsx +118 -0
- flock/frontend/src/components/common/KeyboardShortcutsDialog.css +251 -0
- flock/frontend/src/components/common/KeyboardShortcutsDialog.tsx +151 -0
- flock/frontend/src/components/common/LoadingSpinner.module.css +97 -0
- flock/frontend/src/components/common/LoadingSpinner.tsx +29 -0
- flock/frontend/src/components/controls/PublishControl.css +547 -0
- flock/frontend/src/components/controls/PublishControl.test.tsx +543 -0
- flock/frontend/src/components/controls/PublishControl.tsx +432 -0
- flock/frontend/src/components/details/DetailWindowContainer.tsx +58 -0
- flock/frontend/src/components/details/LiveOutputTab.test.tsx +792 -0
- flock/frontend/src/components/details/LiveOutputTab.tsx +220 -0
- flock/frontend/src/components/details/MessageDetailWindow.tsx +439 -0
- flock/frontend/src/components/details/MessageHistoryTab.tsx +374 -0
- flock/frontend/src/components/details/NodeDetailWindow.test.tsx +501 -0
- flock/frontend/src/components/details/NodeDetailWindow.tsx +218 -0
- flock/frontend/src/components/details/RunStatusTab.tsx +348 -0
- flock/frontend/src/components/details/tabs.test.tsx +1015 -0
- flock/frontend/src/components/filters/ArtifactTypeFilter.tsx +21 -0
- flock/frontend/src/components/filters/CorrelationIDFilter.module.css +102 -0
- flock/frontend/src/components/filters/CorrelationIDFilter.test.tsx +197 -0
- flock/frontend/src/components/filters/CorrelationIDFilter.tsx +121 -0
- flock/frontend/src/components/filters/FilterFlyout.module.css +104 -0
- flock/frontend/src/components/filters/FilterFlyout.tsx +80 -0
- flock/frontend/src/components/filters/FilterPills.module.css +220 -0
- flock/frontend/src/components/filters/FilterPills.test.tsx +189 -0
- flock/frontend/src/components/filters/FilterPills.tsx +143 -0
- flock/frontend/src/components/filters/ProducerFilter.tsx +21 -0
- flock/frontend/src/components/filters/SavedFiltersControl.module.css +60 -0
- flock/frontend/src/components/filters/SavedFiltersControl.test.tsx +158 -0
- flock/frontend/src/components/filters/SavedFiltersControl.tsx +159 -0
- flock/frontend/src/components/filters/TagFilter.tsx +21 -0
- flock/frontend/src/components/filters/TimeRangeFilter.module.css +115 -0
- flock/frontend/src/components/filters/TimeRangeFilter.test.tsx +154 -0
- flock/frontend/src/components/filters/TimeRangeFilter.tsx +110 -0
- flock/frontend/src/components/filters/VisibilityFilter.tsx +21 -0
- flock/frontend/src/components/graph/AgentNode.test.tsx +77 -0
- flock/frontend/src/components/graph/AgentNode.tsx +324 -0
- flock/frontend/src/components/graph/GraphCanvas.tsx +613 -0
- flock/frontend/src/components/graph/MessageFlowEdge.tsx +128 -0
- flock/frontend/src/components/graph/MessageNode.test.tsx +64 -0
- flock/frontend/src/components/graph/MessageNode.tsx +129 -0
- flock/frontend/src/components/graph/MiniMap.tsx +47 -0
- flock/frontend/src/components/graph/TransformEdge.tsx +123 -0
- flock/frontend/src/components/layout/DashboardLayout.css +420 -0
- flock/frontend/src/components/layout/DashboardLayout.tsx +287 -0
- flock/frontend/src/components/layout/Header.module.css +88 -0
- flock/frontend/src/components/layout/Header.tsx +52 -0
- flock/frontend/src/components/modules/HistoricalArtifactsModule.module.css +288 -0
- flock/frontend/src/components/modules/HistoricalArtifactsModule.tsx +450 -0
- flock/frontend/src/components/modules/HistoricalArtifactsModuleWrapper.tsx +13 -0
- flock/frontend/src/components/modules/JsonAttributeRenderer.tsx +140 -0
- flock/frontend/src/components/modules/ModuleRegistry.test.ts +333 -0
- flock/frontend/src/components/modules/ModuleRegistry.ts +93 -0
- flock/frontend/src/components/modules/ModuleWindow.tsx +223 -0
- flock/frontend/src/components/modules/TraceModuleJaeger.tsx +1971 -0
- flock/frontend/src/components/modules/TraceModuleJaegerWrapper.tsx +13 -0
- flock/frontend/src/components/modules/registerModules.ts +29 -0
- flock/frontend/src/components/settings/AdvancedSettings.tsx +175 -0
- flock/frontend/src/components/settings/AppearanceSettings.tsx +185 -0
- flock/frontend/src/components/settings/GraphSettings.tsx +110 -0
- flock/frontend/src/components/settings/MultiSelect.tsx +235 -0
- flock/frontend/src/components/settings/SettingsPanel.css +327 -0
- flock/frontend/src/components/settings/SettingsPanel.tsx +131 -0
- flock/frontend/src/components/settings/ThemeSelector.tsx +298 -0
- flock/frontend/src/components/settings/TracingSettings.tsx +404 -0
- flock/frontend/src/hooks/useKeyboardShortcuts.ts +148 -0
- flock/frontend/src/hooks/useModulePersistence.test.ts +442 -0
- flock/frontend/src/hooks/useModulePersistence.ts +154 -0
- flock/frontend/src/hooks/useModules.ts +157 -0
- flock/frontend/src/hooks/usePersistence.ts +141 -0
- flock/frontend/src/main.tsx +13 -0
- flock/frontend/src/services/api.ts +337 -0
- flock/frontend/src/services/graphService.test.ts +330 -0
- flock/frontend/src/services/graphService.ts +75 -0
- flock/frontend/src/services/indexeddb.test.ts +793 -0
- flock/frontend/src/services/indexeddb.ts +848 -0
- flock/frontend/src/services/layout.test.ts +437 -0
- flock/frontend/src/services/layout.ts +357 -0
- flock/frontend/src/services/themeApplicator.ts +140 -0
- flock/frontend/src/services/themeService.ts +77 -0
- flock/frontend/src/services/websocket.ts +650 -0
- flock/frontend/src/store/filterStore.test.ts +250 -0
- flock/frontend/src/store/filterStore.ts +272 -0
- flock/frontend/src/store/graphStore.test.ts +570 -0
- flock/frontend/src/store/graphStore.ts +462 -0
- flock/frontend/src/store/moduleStore.test.ts +253 -0
- flock/frontend/src/store/moduleStore.ts +75 -0
- flock/frontend/src/store/settingsStore.ts +188 -0
- flock/frontend/src/store/streamStore.ts +68 -0
- flock/frontend/src/store/uiStore.test.ts +54 -0
- flock/frontend/src/store/uiStore.ts +122 -0
- flock/frontend/src/store/wsStore.ts +34 -0
- flock/frontend/src/styles/index.css +15 -0
- flock/frontend/src/styles/scrollbar.css +47 -0
- flock/frontend/src/styles/variables.css +488 -0
- flock/frontend/src/test/setup.ts +1 -0
- flock/frontend/src/types/filters.ts +47 -0
- flock/frontend/src/types/graph.ts +95 -0
- flock/frontend/src/types/modules.ts +10 -0
- flock/frontend/src/types/theme.ts +55 -0
- flock/frontend/src/utils/artifacts.ts +24 -0
- flock/frontend/src/utils/mockData.ts +98 -0
- flock/frontend/src/utils/performance.ts +16 -0
- flock/frontend/src/vite-env.d.ts +17 -0
- flock/frontend/tsconfig.json +27 -0
- flock/frontend/tsconfig.node.json +11 -0
- flock/frontend/vite.config.ts +25 -0
- flock/frontend/vitest.config.ts +11 -0
- flock/{core/util → helper}/cli_helper.py +9 -5
- flock/{core/logging → logging}/__init__.py +2 -3
- flock/logging/auto_trace.py +159 -0
- flock/{core/logging → logging}/formatters/enum_builder.py +3 -4
- flock/{core/logging → logging}/formatters/theme_builder.py +19 -44
- flock/{core/logging → logging}/formatters/themed_formatter.py +69 -107
- flock/{core/logging → logging}/logging.py +78 -61
- flock/{core/logging → logging}/telemetry.py +66 -26
- flock/{core/logging → logging}/telemetry_exporter/base_exporter.py +2 -2
- flock/logging/telemetry_exporter/duckdb_exporter.py +216 -0
- flock/{core/logging → logging}/telemetry_exporter/file_exporter.py +13 -10
- flock/{core/logging → logging}/telemetry_exporter/sqlite_exporter.py +2 -3
- flock/logging/trace_and_logged.py +304 -0
- flock/mcp/__init__.py +91 -0
- flock/{core/mcp/mcp_client.py → mcp/client.py} +131 -158
- flock/{core/mcp/mcp_config.py → mcp/config.py} +86 -132
- flock/mcp/manager.py +286 -0
- flock/mcp/servers/sse/__init__.py +1 -1
- flock/mcp/servers/sse/flock_sse_server.py +16 -58
- flock/mcp/servers/stdio/__init__.py +1 -1
- flock/mcp/servers/stdio/flock_stdio_server.py +13 -53
- flock/mcp/servers/streamable_http/flock_streamable_http_server.py +22 -67
- flock/mcp/servers/websockets/flock_websocket_server.py +12 -45
- flock/{core/mcp/flock_mcp_tool_base.py → mcp/tool.py} +24 -78
- flock/mcp/types/__init__.py +42 -0
- flock/{core/mcp → mcp}/types/callbacks.py +12 -15
- flock/{core/mcp → mcp}/types/factories.py +7 -6
- flock/{core/mcp → mcp}/types/handlers.py +13 -18
- flock/{core/mcp → mcp}/types/types.py +70 -74
- flock/{core/mcp → mcp}/util/helpers.py +3 -3
- flock/orchestrator.py +970 -0
- flock/registry.py +148 -0
- flock/runtime.py +262 -0
- flock/service.py +277 -0
- flock/store.py +1214 -0
- flock/subscription.py +111 -0
- flock/themes/andromeda.toml +1 -1
- flock/themes/apple-system-colors.toml +1 -1
- flock/themes/arcoiris.toml +1 -1
- flock/themes/atomonelight.toml +1 -1
- flock/themes/ayu copy.toml +1 -1
- flock/themes/ayu-light.toml +1 -1
- flock/themes/belafonte-day.toml +1 -1
- flock/themes/belafonte-night.toml +1 -1
- flock/themes/blulocodark.toml +1 -1
- flock/themes/breeze.toml +1 -1
- flock/themes/broadcast.toml +1 -1
- flock/themes/brogrammer.toml +1 -1
- flock/themes/builtin-dark.toml +1 -1
- flock/themes/builtin-pastel-dark.toml +1 -1
- flock/themes/catppuccin-latte.toml +1 -1
- flock/themes/catppuccin-macchiato.toml +1 -1
- flock/themes/catppuccin-mocha.toml +1 -1
- flock/themes/cga.toml +1 -1
- flock/themes/chalk.toml +1 -1
- flock/themes/ciapre.toml +1 -1
- flock/themes/coffee-theme.toml +1 -1
- flock/themes/cyberpunkscarletprotocol.toml +1 -1
- flock/themes/dark+.toml +1 -1
- flock/themes/darkermatrix.toml +1 -1
- flock/themes/darkmatrix.toml +2 -2
- flock/themes/darkside.toml +1 -1
- flock/themes/deep.toml +2 -2
- flock/themes/desert.toml +1 -1
- flock/themes/django.toml +1 -1
- flock/themes/djangosmooth.toml +1 -1
- flock/themes/doomone.toml +1 -1
- flock/themes/dotgov.toml +1 -1
- flock/themes/dracula+.toml +1 -1
- flock/themes/duckbones.toml +1 -1
- flock/themes/encom.toml +1 -1
- flock/themes/espresso.toml +1 -1
- flock/themes/everblush.toml +1 -1
- flock/themes/fairyfloss.toml +1 -1
- flock/themes/fideloper.toml +1 -1
- flock/themes/fishtank.toml +1 -1
- flock/themes/flexoki-light.toml +1 -1
- flock/themes/floraverse.toml +1 -1
- flock/themes/framer.toml +1 -1
- flock/themes/galizur.toml +1 -1
- flock/themes/github.toml +1 -1
- flock/themes/grass.toml +1 -1
- flock/themes/grey-green.toml +1 -1
- flock/themes/gruvboxlight.toml +1 -1
- flock/themes/guezwhoz.toml +1 -1
- flock/themes/harper.toml +1 -1
- flock/themes/hax0r-blue.toml +1 -1
- flock/themes/hopscotch.256.toml +1 -1
- flock/themes/ic-green-ppl.toml +1 -1
- flock/themes/iceberg-dark.toml +1 -1
- flock/themes/japanesque.toml +1 -1
- flock/themes/jubi.toml +1 -1
- flock/themes/kibble.toml +1 -1
- flock/themes/kolorit.toml +1 -1
- flock/themes/kurokula.toml +1 -1
- flock/themes/materialdesigncolors.toml +1 -1
- flock/themes/matrix.toml +1 -1
- flock/themes/mellifluous.toml +1 -1
- flock/themes/midnight-in-mojave.toml +1 -1
- flock/themes/monokai-remastered.toml +1 -1
- flock/themes/monokai-soda.toml +1 -1
- flock/themes/neon.toml +1 -1
- flock/themes/neopolitan.toml +5 -5
- flock/themes/nord-light.toml +1 -1
- flock/themes/ocean.toml +1 -1
- flock/themes/onehalfdark.toml +1 -1
- flock/themes/onehalflight.toml +1 -1
- flock/themes/palenighthc.toml +1 -1
- flock/themes/paulmillr.toml +1 -1
- flock/themes/pencildark.toml +1 -1
- flock/themes/pnevma.toml +1 -1
- flock/themes/purple-rain.toml +1 -1
- flock/themes/purplepeter.toml +1 -1
- flock/themes/raycast-dark.toml +1 -1
- flock/themes/red-sands.toml +1 -1
- flock/themes/relaxed.toml +1 -1
- flock/themes/retro.toml +1 -1
- flock/themes/rose-pine.toml +1 -1
- flock/themes/royal.toml +1 -1
- flock/themes/ryuuko.toml +1 -1
- flock/themes/sakura.toml +1 -1
- flock/themes/scarlet-protocol.toml +1 -1
- flock/themes/seoulbones-dark.toml +1 -1
- flock/themes/shades-of-purple.toml +1 -1
- flock/themes/smyck.toml +1 -1
- flock/themes/softserver.toml +1 -1
- flock/themes/solarized-darcula.toml +1 -1
- flock/themes/square.toml +1 -1
- flock/themes/sugarplum.toml +1 -1
- flock/themes/thayer-bright.toml +1 -1
- flock/themes/tokyonight.toml +1 -1
- flock/themes/tomorrow.toml +1 -1
- flock/themes/ubuntu.toml +1 -1
- flock/themes/ultradark.toml +1 -1
- flock/themes/ultraviolent.toml +1 -1
- flock/themes/unikitty.toml +1 -1
- flock/themes/urple.toml +1 -1
- flock/themes/vesper.toml +1 -1
- flock/themes/vimbones.toml +1 -1
- flock/themes/wildcherry.toml +1 -1
- flock/themes/wilmersdorf.toml +1 -1
- flock/themes/wryan.toml +1 -1
- flock/themes/xcodedarkhc.toml +1 -1
- flock/themes/xcodelight.toml +1 -1
- flock/themes/zenbones-light.toml +1 -1
- flock/themes/zenwritten-dark.toml +1 -1
- flock/utilities.py +301 -0
- flock/utility/output_utility_component.py +226 -0
- flock/visibility.py +107 -0
- flock_core-0.5.0.dist-info/METADATA +964 -0
- flock_core-0.5.0.dist-info/RECORD +525 -0
- flock_core-0.5.0.dist-info/entry_points.txt +2 -0
- {flock_core-0.4.542.dist-info → flock_core-0.5.0.dist-info}/licenses/LICENSE +1 -1
- flock/adapter/__init__.py +0 -14
- flock/adapter/azure_adapter.py +0 -68
- flock/adapter/chroma_adapter.py +0 -73
- flock/adapter/faiss_adapter.py +0 -97
- flock/adapter/pinecone_adapter.py +0 -51
- flock/adapter/vector_base.py +0 -47
- flock/cli/assets/release_notes.md +0 -140
- flock/cli/config.py +0 -8
- flock/cli/constants.py +0 -36
- flock/cli/create_agent.py +0 -1
- flock/cli/create_flock.py +0 -280
- flock/cli/execute_flock.py +0 -620
- flock/cli/load_agent.py +0 -1
- flock/cli/load_examples.py +0 -1
- flock/cli/load_flock.py +0 -192
- flock/cli/load_release_notes.py +0 -20
- flock/cli/loaded_flock_cli.py +0 -254
- flock/cli/manage_agents.py +0 -459
- flock/cli/registry_management.py +0 -889
- flock/cli/runner.py +0 -41
- flock/cli/settings.py +0 -857
- flock/cli/utils.py +0 -135
- flock/cli/view_results.py +0 -29
- flock/cli/yaml_editor.py +0 -396
- flock/config.py +0 -56
- flock/core/__init__.py +0 -44
- flock/core/api/__init__.py +0 -10
- flock/core/api/custom_endpoint.py +0 -45
- flock/core/api/endpoints.py +0 -262
- flock/core/api/main.py +0 -162
- flock/core/api/models.py +0 -101
- flock/core/api/run_store.py +0 -224
- flock/core/api/runner.py +0 -44
- flock/core/api/service.py +0 -214
- flock/core/config/flock_agent_config.py +0 -11
- flock/core/config/scheduled_agent_config.py +0 -40
- flock/core/context/context.py +0 -214
- flock/core/context/context_manager.py +0 -40
- flock/core/context/context_vars.py +0 -11
- flock/core/evaluation/utils.py +0 -395
- flock/core/execution/batch_executor.py +0 -369
- flock/core/execution/evaluation_executor.py +0 -438
- flock/core/execution/local_executor.py +0 -31
- flock/core/execution/opik_executor.py +0 -103
- flock/core/execution/temporal_executor.py +0 -166
- flock/core/flock.py +0 -1003
- flock/core/flock_agent.py +0 -1258
- flock/core/flock_evaluator.py +0 -60
- flock/core/flock_factory.py +0 -513
- flock/core/flock_module.py +0 -207
- flock/core/flock_registry.py +0 -702
- flock/core/flock_router.py +0 -83
- flock/core/flock_scheduler.py +0 -166
- flock/core/flock_server_manager.py +0 -136
- flock/core/interpreter/python_interpreter.py +0 -689
- flock/core/logging/live_capture.py +0 -137
- flock/core/logging/trace_and_logged.py +0 -59
- flock/core/mcp/__init__.py +0 -1
- flock/core/mcp/flock_mcp_server.py +0 -640
- flock/core/mcp/mcp_client_manager.py +0 -201
- flock/core/mcp/types/__init__.py +0 -1
- flock/core/mixin/dspy_integration.py +0 -445
- flock/core/mixin/prompt_parser.py +0 -125
- flock/core/serialization/__init__.py +0 -13
- flock/core/serialization/callable_registry.py +0 -52
- flock/core/serialization/flock_serializer.py +0 -854
- flock/core/serialization/json_encoder.py +0 -41
- flock/core/serialization/secure_serializer.py +0 -175
- flock/core/serialization/serializable.py +0 -342
- flock/core/serialization/serialization_utils.py +0 -409
- flock/core/util/file_path_utils.py +0 -223
- flock/core/util/hydrator.py +0 -309
- flock/core/util/input_resolver.py +0 -141
- flock/core/util/loader.py +0 -59
- flock/core/util/splitter.py +0 -219
- flock/di.py +0 -41
- flock/evaluators/__init__.py +0 -1
- flock/evaluators/declarative/__init__.py +0 -1
- flock/evaluators/declarative/declarative_evaluator.py +0 -217
- flock/evaluators/memory/memory_evaluator.py +0 -90
- flock/evaluators/test/test_case_evaluator.py +0 -38
- flock/evaluators/zep/zep_evaluator.py +0 -59
- flock/modules/__init__.py +0 -1
- flock/modules/assertion/__init__.py +0 -1
- flock/modules/assertion/assertion_module.py +0 -286
- flock/modules/callback/__init__.py +0 -1
- flock/modules/callback/callback_module.py +0 -91
- flock/modules/enterprise_memory/README.md +0 -99
- flock/modules/enterprise_memory/enterprise_memory_module.py +0 -526
- flock/modules/mem0/__init__.py +0 -1
- flock/modules/mem0/mem0_module.py +0 -126
- flock/modules/mem0_async/__init__.py +0 -1
- flock/modules/mem0_async/async_mem0_module.py +0 -126
- flock/modules/memory/__init__.py +0 -1
- flock/modules/memory/memory_module.py +0 -429
- flock/modules/memory/memory_parser.py +0 -125
- flock/modules/memory/memory_storage.py +0 -736
- flock/modules/output/__init__.py +0 -1
- flock/modules/output/output_module.py +0 -196
- flock/modules/performance/__init__.py +0 -1
- flock/modules/performance/metrics_module.py +0 -678
- flock/modules/zep/__init__.py +0 -1
- flock/modules/zep/zep_module.py +0 -192
- flock/platform/docker_tools.py +0 -49
- flock/platform/jaeger_install.py +0 -86
- flock/routers/__init__.py +0 -1
- flock/routers/agent/__init__.py +0 -1
- flock/routers/agent/agent_router.py +0 -236
- flock/routers/agent/handoff_agent.py +0 -58
- flock/routers/conditional/conditional_router.py +0 -486
- flock/routers/default/__init__.py +0 -1
- flock/routers/default/default_router.py +0 -80
- flock/routers/feedback/feedback_router.py +0 -114
- flock/routers/list_generator/list_generator_router.py +0 -166
- flock/routers/llm/__init__.py +0 -1
- flock/routers/llm/llm_router.py +0 -365
- flock/tools/__init__.py +0 -0
- flock/tools/azure_tools.py +0 -781
- flock/tools/code_tools.py +0 -167
- flock/tools/file_tools.py +0 -149
- flock/tools/github_tools.py +0 -157
- flock/tools/markdown_tools.py +0 -205
- flock/tools/system_tools.py +0 -9
- flock/tools/text_tools.py +0 -810
- flock/tools/web_tools.py +0 -92
- flock/tools/zendesk_tools.py +0 -501
- flock/webapp/__init__.py +0 -1
- flock/webapp/app/__init__.py +0 -0
- flock/webapp/app/api/__init__.py +0 -0
- flock/webapp/app/api/agent_management.py +0 -237
- flock/webapp/app/api/execution.py +0 -503
- flock/webapp/app/api/flock_management.py +0 -125
- flock/webapp/app/api/registry_viewer.py +0 -29
- flock/webapp/app/chat.py +0 -662
- flock/webapp/app/config.py +0 -104
- flock/webapp/app/dependencies.py +0 -117
- flock/webapp/app/main.py +0 -1086
- flock/webapp/app/middleware.py +0 -113
- flock/webapp/app/models_ui.py +0 -7
- flock/webapp/app/services/__init__.py +0 -0
- flock/webapp/app/services/feedback_file_service.py +0 -363
- flock/webapp/app/services/flock_service.py +0 -345
- flock/webapp/app/services/sharing_models.py +0 -81
- flock/webapp/app/services/sharing_store.py +0 -597
- flock/webapp/app/templates/theme_mapper.html +0 -326
- flock/webapp/app/theme_mapper.py +0 -811
- flock/webapp/app/utils.py +0 -85
- flock/webapp/run.py +0 -219
- flock/webapp/static/css/chat.css +0 -301
- flock/webapp/static/css/components.css +0 -167
- flock/webapp/static/css/header.css +0 -39
- flock/webapp/static/css/layout.css +0 -281
- flock/webapp/static/css/sidebar.css +0 -127
- flock/webapp/static/css/two-pane.css +0 -48
- flock/webapp/templates/base.html +0 -389
- flock/webapp/templates/chat.html +0 -152
- flock/webapp/templates/chat_settings.html +0 -19
- flock/webapp/templates/flock_editor.html +0 -16
- flock/webapp/templates/index.html +0 -12
- flock/webapp/templates/partials/_agent_detail_form.html +0 -93
- flock/webapp/templates/partials/_agent_list.html +0 -18
- flock/webapp/templates/partials/_agent_manager_view.html +0 -51
- flock/webapp/templates/partials/_agent_tools_checklist.html +0 -14
- flock/webapp/templates/partials/_chat_container.html +0 -15
- flock/webapp/templates/partials/_chat_messages.html +0 -57
- flock/webapp/templates/partials/_chat_settings_form.html +0 -85
- flock/webapp/templates/partials/_create_flock_form.html +0 -50
- flock/webapp/templates/partials/_dashboard_flock_detail.html +0 -17
- flock/webapp/templates/partials/_dashboard_flock_file_list.html +0 -16
- flock/webapp/templates/partials/_dashboard_flock_properties_preview.html +0 -28
- flock/webapp/templates/partials/_dashboard_upload_flock_form.html +0 -16
- flock/webapp/templates/partials/_dynamic_input_form_content.html +0 -22
- flock/webapp/templates/partials/_env_vars_table.html +0 -23
- flock/webapp/templates/partials/_execution_form.html +0 -127
- flock/webapp/templates/partials/_execution_view_container.html +0 -28
- flock/webapp/templates/partials/_flock_file_list.html +0 -23
- flock/webapp/templates/partials/_flock_properties_form.html +0 -52
- flock/webapp/templates/partials/_flock_upload_form.html +0 -16
- flock/webapp/templates/partials/_header_flock_status.html +0 -5
- flock/webapp/templates/partials/_live_logs.html +0 -13
- flock/webapp/templates/partials/_load_manager_view.html +0 -49
- flock/webapp/templates/partials/_registry_table.html +0 -25
- flock/webapp/templates/partials/_registry_viewer_content.html +0 -70
- flock/webapp/templates/partials/_results_display.html +0 -78
- flock/webapp/templates/partials/_settings_env_content.html +0 -9
- flock/webapp/templates/partials/_settings_theme_content.html +0 -14
- flock/webapp/templates/partials/_settings_view.html +0 -36
- flock/webapp/templates/partials/_share_chat_link_snippet.html +0 -11
- flock/webapp/templates/partials/_share_link_snippet.html +0 -35
- flock/webapp/templates/partials/_sidebar.html +0 -74
- flock/webapp/templates/partials/_structured_data_view.html +0 -40
- flock/webapp/templates/partials/_theme_preview.html +0 -36
- flock/webapp/templates/registry_viewer.html +0 -84
- flock/webapp/templates/shared_run_page.html +0 -140
- flock/workflow/__init__.py +0 -0
- flock/workflow/activities.py +0 -237
- flock/workflow/agent_activities.py +0 -24
- flock/workflow/agent_execution_activity.py +0 -240
- flock/workflow/flock_workflow.py +0 -225
- flock/workflow/temporal_config.py +0 -96
- flock/workflow/temporal_setup.py +0 -60
- flock_core-0.4.542.dist-info/METADATA +0 -676
- flock_core-0.4.542.dist-info/RECORD +0 -572
- flock_core-0.4.542.dist-info/entry_points.txt +0 -2
- /flock/{core/logging → logging}/formatters/themes.py +0 -0
- /flock/{core/logging → logging}/span_middleware/baggage_span_processor.py +0 -0
- /flock/{core/mcp → mcp}/util/__init__.py +0 -0
- {flock_core-0.4.542.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
|
+
});
|