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,154 @@
|
|
|
1
|
+
import { useEffect, useRef, useCallback } from 'react';
|
|
2
|
+
import { useModuleStore } from '../store/moduleStore';
|
|
3
|
+
import { indexedDBService } from '../services/indexeddb';
|
|
4
|
+
import type { ModuleInstance } from '../types/modules';
|
|
5
|
+
|
|
6
|
+
/**
|
|
7
|
+
* Hook to persist module instances to IndexedDB
|
|
8
|
+
*
|
|
9
|
+
* Features:
|
|
10
|
+
* - Loads all module instances on mount
|
|
11
|
+
* - Saves instances when added/updated (with debouncing for drag/resize)
|
|
12
|
+
* - Deletes instances when removed
|
|
13
|
+
* - Debounce: 300ms (same as node position persistence)
|
|
14
|
+
*/
|
|
15
|
+
export const useModulePersistence = () => {
|
|
16
|
+
const instances = useModuleStore((state) => state.instances);
|
|
17
|
+
const addModule = useModuleStore((state) => state.addModule);
|
|
18
|
+
|
|
19
|
+
// Track previous instances to detect changes
|
|
20
|
+
const previousInstancesRef = useRef<Map<string, ModuleInstance>>(new Map());
|
|
21
|
+
|
|
22
|
+
// Debounce timer for saves
|
|
23
|
+
const saveTimerRef = useRef<{ [key: string]: number }>({});
|
|
24
|
+
|
|
25
|
+
// Load module instances on mount
|
|
26
|
+
useEffect(() => {
|
|
27
|
+
const loadModuleInstances = async () => {
|
|
28
|
+
try {
|
|
29
|
+
const savedInstances = await indexedDBService.getAllModuleInstances();
|
|
30
|
+
|
|
31
|
+
// Restore each module instance to the store
|
|
32
|
+
savedInstances.forEach((record) => {
|
|
33
|
+
const instance: ModuleInstance = {
|
|
34
|
+
id: record.instance_id,
|
|
35
|
+
type: record.type,
|
|
36
|
+
position: record.position,
|
|
37
|
+
size: record.size,
|
|
38
|
+
visible: record.visible,
|
|
39
|
+
};
|
|
40
|
+
addModule(instance);
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
console.log(`[ModulePersistence] Restored ${savedInstances.length} module instances`);
|
|
44
|
+
} catch (error) {
|
|
45
|
+
console.error('[ModulePersistence] Failed to load module instances:', error);
|
|
46
|
+
}
|
|
47
|
+
};
|
|
48
|
+
|
|
49
|
+
loadModuleInstances();
|
|
50
|
+
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
51
|
+
}, []); // Only run on mount
|
|
52
|
+
|
|
53
|
+
// Save module instances when they change (debounced)
|
|
54
|
+
const saveModuleInstance = useCallback(async (instance: ModuleInstance) => {
|
|
55
|
+
// Clear existing timer for this instance
|
|
56
|
+
if (saveTimerRef.current[instance.id]) {
|
|
57
|
+
clearTimeout(saveTimerRef.current[instance.id]);
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
// Debounce: wait 300ms before saving (handles rapid drag/resize updates)
|
|
61
|
+
saveTimerRef.current[instance.id] = setTimeout(async () => {
|
|
62
|
+
try {
|
|
63
|
+
const now = new Date().toISOString();
|
|
64
|
+
const record = {
|
|
65
|
+
instance_id: instance.id,
|
|
66
|
+
type: instance.type,
|
|
67
|
+
position: instance.position,
|
|
68
|
+
size: instance.size,
|
|
69
|
+
visible: instance.visible,
|
|
70
|
+
created_at: now, // Will be overwritten if updating existing
|
|
71
|
+
updated_at: now,
|
|
72
|
+
};
|
|
73
|
+
|
|
74
|
+
await indexedDBService.saveModuleInstance(record);
|
|
75
|
+
console.log(`[ModulePersistence] Saved instance: ${instance.id}`);
|
|
76
|
+
} catch (error) {
|
|
77
|
+
console.error(`[ModulePersistence] Failed to save instance ${instance.id}:`, error);
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
delete saveTimerRef.current[instance.id];
|
|
81
|
+
}, 300);
|
|
82
|
+
}, []);
|
|
83
|
+
|
|
84
|
+
// Delete module instance from IndexedDB
|
|
85
|
+
const deleteModuleInstance = useCallback(async (instanceId: string) => {
|
|
86
|
+
// Clear any pending save timer
|
|
87
|
+
if (saveTimerRef.current[instanceId]) {
|
|
88
|
+
clearTimeout(saveTimerRef.current[instanceId]);
|
|
89
|
+
delete saveTimerRef.current[instanceId];
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
try {
|
|
93
|
+
await indexedDBService.deleteModuleInstance(instanceId);
|
|
94
|
+
console.log(`[ModulePersistence] Deleted instance: ${instanceId}`);
|
|
95
|
+
} catch (error) {
|
|
96
|
+
console.error(`[ModulePersistence] Failed to delete instance ${instanceId}:`, error);
|
|
97
|
+
}
|
|
98
|
+
}, []);
|
|
99
|
+
|
|
100
|
+
// Detect changes in module instances
|
|
101
|
+
useEffect(() => {
|
|
102
|
+
const currentIds = new Set(instances.keys());
|
|
103
|
+
const previousIds = new Set(previousInstancesRef.current.keys());
|
|
104
|
+
|
|
105
|
+
// Detect added instances
|
|
106
|
+
currentIds.forEach((id) => {
|
|
107
|
+
if (!previousIds.has(id)) {
|
|
108
|
+
const instance = instances.get(id);
|
|
109
|
+
if (instance) {
|
|
110
|
+
console.log(`[ModulePersistence] Detected new instance: ${id}`);
|
|
111
|
+
saveModuleInstance(instance);
|
|
112
|
+
}
|
|
113
|
+
}
|
|
114
|
+
});
|
|
115
|
+
|
|
116
|
+
// Detect updated instances
|
|
117
|
+
instances.forEach((instance, id) => {
|
|
118
|
+
const previous = previousInstancesRef.current.get(id);
|
|
119
|
+
if (previous) {
|
|
120
|
+
// Check if position, size, or visibility changed
|
|
121
|
+
const positionChanged =
|
|
122
|
+
previous.position.x !== instance.position.x ||
|
|
123
|
+
previous.position.y !== instance.position.y;
|
|
124
|
+
const sizeChanged =
|
|
125
|
+
previous.size.width !== instance.size.width ||
|
|
126
|
+
previous.size.height !== instance.size.height;
|
|
127
|
+
const visibilityChanged = previous.visible !== instance.visible;
|
|
128
|
+
|
|
129
|
+
if (positionChanged || sizeChanged || visibilityChanged) {
|
|
130
|
+
console.log(`[ModulePersistence] Detected update to instance: ${id}`);
|
|
131
|
+
saveModuleInstance(instance);
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
});
|
|
135
|
+
|
|
136
|
+
// Detect removed instances
|
|
137
|
+
previousIds.forEach((id) => {
|
|
138
|
+
if (!currentIds.has(id)) {
|
|
139
|
+
console.log(`[ModulePersistence] Detected removed instance: ${id}`);
|
|
140
|
+
deleteModuleInstance(id);
|
|
141
|
+
}
|
|
142
|
+
});
|
|
143
|
+
|
|
144
|
+
// Update ref for next comparison
|
|
145
|
+
previousInstancesRef.current = new Map(instances);
|
|
146
|
+
}, [instances, saveModuleInstance, deleteModuleInstance]);
|
|
147
|
+
|
|
148
|
+
// Cleanup timers on unmount
|
|
149
|
+
useEffect(() => {
|
|
150
|
+
return () => {
|
|
151
|
+
Object.values(saveTimerRef.current).forEach(clearTimeout);
|
|
152
|
+
};
|
|
153
|
+
}, []);
|
|
154
|
+
};
|
|
@@ -0,0 +1,157 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* useModules Hook - Module Lifecycle Management
|
|
3
|
+
*
|
|
4
|
+
* Provides ModuleContext to modules and manages lifecycle hooks.
|
|
5
|
+
* Aggregates data from multiple stores into a unified context.
|
|
6
|
+
* Calls module onMount/onUnmount lifecycle hooks when instances change.
|
|
7
|
+
*
|
|
8
|
+
* SPECIFICATION: docs/specs/003-real-time-dashboard/FRONTEND_ARCHITECTURE.md Section 7.4
|
|
9
|
+
* - Build ModuleContext from store data (events, filters)
|
|
10
|
+
* - Call module onMount lifecycle hooks when instances added
|
|
11
|
+
* - Call module onUnmount lifecycle hooks when instances removed
|
|
12
|
+
* - Provide publish and invoke actions in context
|
|
13
|
+
*
|
|
14
|
+
* UI Optimization Migration (Phase 4.1 - Spec 002):
|
|
15
|
+
* - agents/messages Maps are DEPRECATED (Phase 1 architecture)
|
|
16
|
+
* - Modules should use events array instead
|
|
17
|
+
* - Empty Maps provided for backward compatibility
|
|
18
|
+
*/
|
|
19
|
+
|
|
20
|
+
import { useEffect, useMemo, useRef } from 'react';
|
|
21
|
+
import { useModuleStore } from '../store/moduleStore';
|
|
22
|
+
import { useGraphStore } from '../store/graphStore';
|
|
23
|
+
import { useFilterStore } from '../store/filterStore';
|
|
24
|
+
import { moduleRegistry, type ModuleContext } from '../components/modules/ModuleRegistry';
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Custom hook for module lifecycle management
|
|
28
|
+
*
|
|
29
|
+
* Features:
|
|
30
|
+
* - Builds ModuleContext from graphStore, filterStore
|
|
31
|
+
* - Tracks module instances and calls lifecycle hooks
|
|
32
|
+
* - Calls onMount when new instances are added
|
|
33
|
+
* - Calls onUnmount when instances are removed
|
|
34
|
+
* - Provides publish/invoke actions (placeholder for now)
|
|
35
|
+
*
|
|
36
|
+
* @returns {Object} Hook return value
|
|
37
|
+
* @returns {ModuleContext} context - The module context object
|
|
38
|
+
*/
|
|
39
|
+
export function useModules() {
|
|
40
|
+
// Subscribe to store state
|
|
41
|
+
const instances = useModuleStore((state) => state.instances);
|
|
42
|
+
const events = useGraphStore((state) => state.events);
|
|
43
|
+
|
|
44
|
+
// UI Optimization Migration (Phase 4.1): Provide empty Maps for deprecated fields
|
|
45
|
+
const agents = useMemo(() => new Map(), []);
|
|
46
|
+
const messages = useMemo(() => new Map(), []);
|
|
47
|
+
const correlationId = useFilterStore((state) => state.correlationId);
|
|
48
|
+
const timeRange = useFilterStore((state) => state.timeRange);
|
|
49
|
+
const artifactTypes = useFilterStore((state) => state.selectedArtifactTypes);
|
|
50
|
+
const producers = useFilterStore((state) => state.selectedProducers);
|
|
51
|
+
const tags = useFilterStore((state) => state.selectedTags);
|
|
52
|
+
const visibility = useFilterStore((state) => state.selectedVisibility);
|
|
53
|
+
const summary = useFilterStore((state) => state.summary);
|
|
54
|
+
|
|
55
|
+
// Track previous instances to detect changes
|
|
56
|
+
const prevInstancesRef = useRef<Map<string, any>>(new Map());
|
|
57
|
+
|
|
58
|
+
// Build ModuleContext
|
|
59
|
+
const context: ModuleContext = useMemo(
|
|
60
|
+
() => ({
|
|
61
|
+
agents,
|
|
62
|
+
messages,
|
|
63
|
+
events,
|
|
64
|
+
filters: {
|
|
65
|
+
correlationId,
|
|
66
|
+
timeRange,
|
|
67
|
+
artifactTypes,
|
|
68
|
+
producers,
|
|
69
|
+
tags,
|
|
70
|
+
visibility,
|
|
71
|
+
},
|
|
72
|
+
summary,
|
|
73
|
+
publish: (artifact: any) => {
|
|
74
|
+
// Placeholder: In production, this would dispatch to WebSocket
|
|
75
|
+
console.log('[Module Context] Publish artifact:', artifact);
|
|
76
|
+
},
|
|
77
|
+
invoke: (agentName: string, inputs: any[]) => {
|
|
78
|
+
// Placeholder: In production, this would dispatch to WebSocket
|
|
79
|
+
console.log('[Module Context] Invoke agent:', agentName, 'with inputs:', inputs);
|
|
80
|
+
},
|
|
81
|
+
}),
|
|
82
|
+
// Note: agents and messages are stable empty Maps, so excluded from deps
|
|
83
|
+
[events, correlationId, timeRange, artifactTypes, producers, tags, visibility, summary]
|
|
84
|
+
);
|
|
85
|
+
|
|
86
|
+
/**
|
|
87
|
+
* Lifecycle effect: Call onMount/onUnmount hooks when instances change
|
|
88
|
+
*/
|
|
89
|
+
useEffect(() => {
|
|
90
|
+
const currentInstances = instances;
|
|
91
|
+
const prevInstances = prevInstancesRef.current;
|
|
92
|
+
|
|
93
|
+
// Detect added instances (in current but not in previous)
|
|
94
|
+
const addedInstances = new Map<string, any>();
|
|
95
|
+
currentInstances.forEach((instance, id) => {
|
|
96
|
+
if (!prevInstances.has(id)) {
|
|
97
|
+
addedInstances.set(id, instance);
|
|
98
|
+
}
|
|
99
|
+
});
|
|
100
|
+
|
|
101
|
+
// Detect removed instances (in previous but not in current)
|
|
102
|
+
const removedInstances = new Map<string, any>();
|
|
103
|
+
prevInstances.forEach((instance, id) => {
|
|
104
|
+
if (!currentInstances.has(id)) {
|
|
105
|
+
removedInstances.set(id, instance);
|
|
106
|
+
}
|
|
107
|
+
});
|
|
108
|
+
|
|
109
|
+
// Call onMount for added instances
|
|
110
|
+
addedInstances.forEach((instance) => {
|
|
111
|
+
const module = moduleRegistry.get(instance.type);
|
|
112
|
+
if (module?.onMount) {
|
|
113
|
+
console.log(`[useModules] Calling onMount for module: ${module.name} (${instance.id})`);
|
|
114
|
+
try {
|
|
115
|
+
module.onMount(context);
|
|
116
|
+
} catch (error) {
|
|
117
|
+
console.error(`[useModules] Error in onMount for module ${module.name}:`, error);
|
|
118
|
+
}
|
|
119
|
+
}
|
|
120
|
+
});
|
|
121
|
+
|
|
122
|
+
// Call onUnmount for removed instances
|
|
123
|
+
removedInstances.forEach((instance) => {
|
|
124
|
+
const module = moduleRegistry.get(instance.type);
|
|
125
|
+
if (module?.onUnmount) {
|
|
126
|
+
console.log(`[useModules] Calling onUnmount for module: ${module.name} (${instance.id})`);
|
|
127
|
+
try {
|
|
128
|
+
module.onUnmount();
|
|
129
|
+
} catch (error) {
|
|
130
|
+
console.error(`[useModules] Error in onUnmount for module ${module.name}:`, error);
|
|
131
|
+
}
|
|
132
|
+
}
|
|
133
|
+
});
|
|
134
|
+
|
|
135
|
+
// Update previous instances ref for next comparison
|
|
136
|
+
prevInstancesRef.current = new Map(currentInstances);
|
|
137
|
+
|
|
138
|
+
// Cleanup: Call onUnmount for all instances when hook unmounts
|
|
139
|
+
return () => {
|
|
140
|
+
currentInstances.forEach((instance) => {
|
|
141
|
+
const module = moduleRegistry.get(instance.type);
|
|
142
|
+
if (module?.onUnmount) {
|
|
143
|
+
console.log(`[useModules] Cleanup: Calling onUnmount for module: ${module.name} (${instance.id})`);
|
|
144
|
+
try {
|
|
145
|
+
module.onUnmount();
|
|
146
|
+
} catch (error) {
|
|
147
|
+
console.error(`[useModules] Error in cleanup onUnmount for module ${module.name}:`, error);
|
|
148
|
+
}
|
|
149
|
+
}
|
|
150
|
+
});
|
|
151
|
+
};
|
|
152
|
+
}, [instances, context]);
|
|
153
|
+
|
|
154
|
+
return {
|
|
155
|
+
context,
|
|
156
|
+
};
|
|
157
|
+
}
|
|
@@ -0,0 +1,141 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* usePersistence Hook - Node Position Persistence
|
|
3
|
+
*
|
|
4
|
+
* Provides position persistence for graph nodes using IndexedDB.
|
|
5
|
+
* Manages loading and saving of node positions with debouncing.
|
|
6
|
+
* Handles separate layouts for Agent View and Blackboard View.
|
|
7
|
+
*
|
|
8
|
+
* SPECIFICATION: docs/specs/003-real-time-dashboard/FRONTEND_ARCHITECTURE.md
|
|
9
|
+
* - Debounce: 300ms delay on drag stop
|
|
10
|
+
* - Performance: Position save <50ms (after debounce)
|
|
11
|
+
* - Separate layouts for Agent View vs Blackboard View
|
|
12
|
+
*/
|
|
13
|
+
|
|
14
|
+
import { useEffect, useCallback, useRef } from 'react';
|
|
15
|
+
import { useGraphStore } from '../store/graphStore';
|
|
16
|
+
import { useUIStore, VisualizationMode } from '../store/uiStore';
|
|
17
|
+
import { indexedDBService } from '../services/indexeddb';
|
|
18
|
+
|
|
19
|
+
interface Position {
|
|
20
|
+
x: number;
|
|
21
|
+
y: number;
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
/**
|
|
25
|
+
* Custom debounce implementation
|
|
26
|
+
* Creates a debounced function that delays invoking func until after delay milliseconds
|
|
27
|
+
*/
|
|
28
|
+
function debounce<T extends (...args: any[]) => void>(
|
|
29
|
+
func: T,
|
|
30
|
+
delay: number
|
|
31
|
+
): (...args: Parameters<T>) => void {
|
|
32
|
+
let timeoutId: ReturnType<typeof setTimeout> | null = null;
|
|
33
|
+
|
|
34
|
+
return (...args: Parameters<T>) => {
|
|
35
|
+
if (timeoutId) {
|
|
36
|
+
clearTimeout(timeoutId);
|
|
37
|
+
}
|
|
38
|
+
|
|
39
|
+
timeoutId = setTimeout(() => {
|
|
40
|
+
func(...args);
|
|
41
|
+
timeoutId = null;
|
|
42
|
+
}, delay);
|
|
43
|
+
};
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
/**
|
|
47
|
+
* React hook for node position persistence
|
|
48
|
+
*
|
|
49
|
+
* Features:
|
|
50
|
+
* - Loads node positions from IndexedDB on mount
|
|
51
|
+
* - Applies loaded positions to graphStore
|
|
52
|
+
* - Returns debounced saveNodePosition function (300ms delay)
|
|
53
|
+
* - Handles mode switching (reloads positions for new mode)
|
|
54
|
+
*
|
|
55
|
+
* @returns {Object} Hook return value
|
|
56
|
+
* @returns {Function} saveNodePosition - Debounced function to save node position
|
|
57
|
+
*/
|
|
58
|
+
export function usePersistence() {
|
|
59
|
+
const mode = useUIStore((state) => state.mode);
|
|
60
|
+
|
|
61
|
+
// Use ref to maintain debounced function identity across renders
|
|
62
|
+
const debouncedSaveRef = useRef<((nodeId: string, mode: VisualizationMode, position: Position) => void) | null>(null);
|
|
63
|
+
|
|
64
|
+
// Initialize debounced save function once
|
|
65
|
+
if (!debouncedSaveRef.current) {
|
|
66
|
+
debouncedSaveRef.current = debounce(
|
|
67
|
+
async (nodeId: string, mode: VisualizationMode, position: Position) => {
|
|
68
|
+
try {
|
|
69
|
+
console.log(`[usePersistence] Saving position for ${nodeId} in ${mode} view:`, position);
|
|
70
|
+
const layoutRecord = {
|
|
71
|
+
node_id: nodeId,
|
|
72
|
+
x: position.x,
|
|
73
|
+
y: position.y,
|
|
74
|
+
last_updated: new Date().toISOString(),
|
|
75
|
+
};
|
|
76
|
+
|
|
77
|
+
if (mode === 'agent') {
|
|
78
|
+
await indexedDBService.saveAgentViewLayout(layoutRecord);
|
|
79
|
+
console.log(`[usePersistence] ✓ Saved to agent view layout`);
|
|
80
|
+
} else {
|
|
81
|
+
await indexedDBService.saveBlackboardViewLayout(layoutRecord);
|
|
82
|
+
console.log(`[usePersistence] ✓ Saved to blackboard view layout`);
|
|
83
|
+
}
|
|
84
|
+
} catch (error) {
|
|
85
|
+
console.error(`[usePersistence] Failed to save node position for ${nodeId}:`, error);
|
|
86
|
+
}
|
|
87
|
+
},
|
|
88
|
+
300 // 300ms debounce delay as per specification
|
|
89
|
+
);
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
/**
|
|
93
|
+
* Load node positions from IndexedDB for current mode
|
|
94
|
+
* Note: Deliberately excludes updateNodePosition from dependencies to prevent infinite loops
|
|
95
|
+
* The function is stable from zustand, so we can safely use it without re-creating the callback
|
|
96
|
+
*/
|
|
97
|
+
const loadNodePositions = useCallback(async (currentMode: VisualizationMode) => {
|
|
98
|
+
try {
|
|
99
|
+
let layouts: Array<{ node_id: string; x: number; y: number; last_updated: string }> = [];
|
|
100
|
+
|
|
101
|
+
if (currentMode === 'agent') {
|
|
102
|
+
layouts = await indexedDBService.getAllAgentViewLayouts();
|
|
103
|
+
} else {
|
|
104
|
+
layouts = await indexedDBService.getAllBlackboardViewLayouts();
|
|
105
|
+
}
|
|
106
|
+
|
|
107
|
+
// Apply loaded positions to graph store
|
|
108
|
+
// Use graphStore directly to avoid dependency on updateNodePosition selector
|
|
109
|
+
layouts.forEach((layout) => {
|
|
110
|
+
useGraphStore.getState().updateNodePosition(layout.node_id, { x: layout.x, y: layout.y });
|
|
111
|
+
});
|
|
112
|
+
|
|
113
|
+
console.log(`[usePersistence] Loaded ${layouts.length} node positions for ${currentMode} view`);
|
|
114
|
+
} catch (error) {
|
|
115
|
+
console.error(`[usePersistence] Failed to load node positions for ${currentMode} view:`, error);
|
|
116
|
+
}
|
|
117
|
+
}, []); // Empty deps - function is now stable!
|
|
118
|
+
|
|
119
|
+
/**
|
|
120
|
+
* Load positions on mount and when mode changes
|
|
121
|
+
*/
|
|
122
|
+
useEffect(() => {
|
|
123
|
+
loadNodePositions(mode);
|
|
124
|
+
}, [mode, loadNodePositions]);
|
|
125
|
+
|
|
126
|
+
/**
|
|
127
|
+
* Public API: Save node position with debouncing
|
|
128
|
+
*/
|
|
129
|
+
const saveNodePosition = useCallback(
|
|
130
|
+
(nodeId: string, position: Position) => {
|
|
131
|
+
if (debouncedSaveRef.current) {
|
|
132
|
+
debouncedSaveRef.current(nodeId, mode, position);
|
|
133
|
+
}
|
|
134
|
+
},
|
|
135
|
+
[mode]
|
|
136
|
+
);
|
|
137
|
+
|
|
138
|
+
return {
|
|
139
|
+
saveNodePosition,
|
|
140
|
+
};
|
|
141
|
+
}
|
|
@@ -0,0 +1,13 @@
|
|
|
1
|
+
import ReactDOM from 'react-dom/client';
|
|
2
|
+
import App from './App';
|
|
3
|
+
import './styles/index.css';
|
|
4
|
+
|
|
5
|
+
// Mock data removed - dashboard now starts empty and shows only real WebSocket data
|
|
6
|
+
|
|
7
|
+
const rootElement = document.getElementById('root');
|
|
8
|
+
|
|
9
|
+
if (!rootElement) {
|
|
10
|
+
throw new Error('Root element not found');
|
|
11
|
+
}
|
|
12
|
+
|
|
13
|
+
ReactDOM.createRoot(rootElement).render(<App />);
|