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

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

Potentially problematic release.


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

Files changed (469) hide show
  1. flock/__init__.py +12 -217
  2. flock/agent.py +678 -0
  3. flock/api/themes.py +71 -0
  4. flock/artifacts.py +79 -0
  5. flock/cli.py +75 -0
  6. flock/components.py +173 -0
  7. flock/dashboard/__init__.py +28 -0
  8. flock/dashboard/collector.py +283 -0
  9. flock/dashboard/events.py +182 -0
  10. flock/dashboard/launcher.py +230 -0
  11. flock/dashboard/service.py +537 -0
  12. flock/dashboard/websocket.py +235 -0
  13. flock/engines/__init__.py +6 -0
  14. flock/engines/dspy_engine.py +856 -0
  15. flock/examples.py +128 -0
  16. flock/frontend/README.md +678 -0
  17. flock/frontend/docs/DESIGN_SYSTEM.md +1980 -0
  18. flock/frontend/index.html +12 -0
  19. flock/frontend/package-lock.json +4347 -0
  20. flock/frontend/package.json +48 -0
  21. flock/frontend/src/App.tsx +79 -0
  22. flock/frontend/src/__tests__/e2e/critical-scenarios.test.tsx +587 -0
  23. flock/frontend/src/__tests__/integration/filtering-e2e.test.tsx +387 -0
  24. flock/frontend/src/__tests__/integration/graph-rendering.test.tsx +640 -0
  25. flock/frontend/src/__tests__/integration/indexeddb-persistence.test.tsx +699 -0
  26. flock/frontend/src/components/common/BuildInfo.tsx +39 -0
  27. flock/frontend/src/components/common/EmptyState.module.css +115 -0
  28. flock/frontend/src/components/common/EmptyState.tsx +128 -0
  29. flock/frontend/src/components/common/ErrorBoundary.module.css +169 -0
  30. flock/frontend/src/components/common/ErrorBoundary.tsx +118 -0
  31. flock/frontend/src/components/common/KeyboardShortcutsDialog.css +251 -0
  32. flock/frontend/src/components/common/KeyboardShortcutsDialog.tsx +151 -0
  33. flock/frontend/src/components/common/LoadingSpinner.module.css +97 -0
  34. flock/frontend/src/components/common/LoadingSpinner.tsx +29 -0
  35. flock/frontend/src/components/controls/PublishControl.css +547 -0
  36. flock/frontend/src/components/controls/PublishControl.test.tsx +543 -0
  37. flock/frontend/src/components/controls/PublishControl.tsx +432 -0
  38. flock/frontend/src/components/details/DetailWindowContainer.tsx +62 -0
  39. flock/frontend/src/components/details/LiveOutputTab.test.tsx +792 -0
  40. flock/frontend/src/components/details/LiveOutputTab.tsx +220 -0
  41. flock/frontend/src/components/details/MessageHistoryTab.tsx +299 -0
  42. flock/frontend/src/components/details/NodeDetailWindow.test.tsx +501 -0
  43. flock/frontend/src/components/details/NodeDetailWindow.tsx +218 -0
  44. flock/frontend/src/components/details/RunStatusTab.tsx +307 -0
  45. flock/frontend/src/components/details/tabs.test.tsx +1015 -0
  46. flock/frontend/src/components/filters/CorrelationIDFilter.module.css +102 -0
  47. flock/frontend/src/components/filters/CorrelationIDFilter.test.tsx +197 -0
  48. flock/frontend/src/components/filters/CorrelationIDFilter.tsx +121 -0
  49. flock/frontend/src/components/filters/FilterBar.module.css +29 -0
  50. flock/frontend/src/components/filters/FilterBar.test.tsx +133 -0
  51. flock/frontend/src/components/filters/FilterBar.tsx +33 -0
  52. flock/frontend/src/components/filters/FilterPills.module.css +79 -0
  53. flock/frontend/src/components/filters/FilterPills.test.tsx +173 -0
  54. flock/frontend/src/components/filters/FilterPills.tsx +67 -0
  55. flock/frontend/src/components/filters/TimeRangeFilter.module.css +91 -0
  56. flock/frontend/src/components/filters/TimeRangeFilter.test.tsx +154 -0
  57. flock/frontend/src/components/filters/TimeRangeFilter.tsx +105 -0
  58. flock/frontend/src/components/graph/AgentNode.test.tsx +75 -0
  59. flock/frontend/src/components/graph/AgentNode.tsx +322 -0
  60. flock/frontend/src/components/graph/GraphCanvas.tsx +406 -0
  61. flock/frontend/src/components/graph/MessageFlowEdge.tsx +128 -0
  62. flock/frontend/src/components/graph/MessageNode.test.tsx +62 -0
  63. flock/frontend/src/components/graph/MessageNode.tsx +116 -0
  64. flock/frontend/src/components/graph/MiniMap.tsx +47 -0
  65. flock/frontend/src/components/graph/TransformEdge.tsx +123 -0
  66. flock/frontend/src/components/layout/DashboardLayout.css +407 -0
  67. flock/frontend/src/components/layout/DashboardLayout.tsx +300 -0
  68. flock/frontend/src/components/layout/Header.module.css +88 -0
  69. flock/frontend/src/components/layout/Header.tsx +52 -0
  70. flock/frontend/src/components/modules/EventLogModule.test.tsx +401 -0
  71. flock/frontend/src/components/modules/EventLogModule.tsx +396 -0
  72. flock/frontend/src/components/modules/EventLogModuleWrapper.tsx +17 -0
  73. flock/frontend/src/components/modules/ModuleRegistry.test.ts +333 -0
  74. flock/frontend/src/components/modules/ModuleRegistry.ts +85 -0
  75. flock/frontend/src/components/modules/ModuleWindow.tsx +155 -0
  76. flock/frontend/src/components/modules/registerModules.ts +20 -0
  77. flock/frontend/src/components/settings/AdvancedSettings.tsx +175 -0
  78. flock/frontend/src/components/settings/AppearanceSettings.tsx +185 -0
  79. flock/frontend/src/components/settings/GraphSettings.tsx +110 -0
  80. flock/frontend/src/components/settings/SettingsPanel.css +327 -0
  81. flock/frontend/src/components/settings/SettingsPanel.tsx +131 -0
  82. flock/frontend/src/components/settings/ThemeSelector.tsx +298 -0
  83. flock/frontend/src/hooks/useKeyboardShortcuts.ts +148 -0
  84. flock/frontend/src/hooks/useModulePersistence.test.ts +442 -0
  85. flock/frontend/src/hooks/useModulePersistence.ts +154 -0
  86. flock/frontend/src/hooks/useModules.ts +139 -0
  87. flock/frontend/src/hooks/usePersistence.ts +139 -0
  88. flock/frontend/src/main.tsx +13 -0
  89. flock/frontend/src/services/api.ts +213 -0
  90. flock/frontend/src/services/indexeddb.test.ts +793 -0
  91. flock/frontend/src/services/indexeddb.ts +794 -0
  92. flock/frontend/src/services/layout.test.ts +437 -0
  93. flock/frontend/src/services/layout.ts +146 -0
  94. flock/frontend/src/services/themeApplicator.ts +140 -0
  95. flock/frontend/src/services/themeService.ts +77 -0
  96. flock/frontend/src/services/websocket.test.ts +595 -0
  97. flock/frontend/src/services/websocket.ts +685 -0
  98. flock/frontend/src/store/filterStore.test.ts +242 -0
  99. flock/frontend/src/store/filterStore.ts +103 -0
  100. flock/frontend/src/store/graphStore.test.ts +186 -0
  101. flock/frontend/src/store/graphStore.ts +414 -0
  102. flock/frontend/src/store/moduleStore.test.ts +253 -0
  103. flock/frontend/src/store/moduleStore.ts +57 -0
  104. flock/frontend/src/store/settingsStore.ts +188 -0
  105. flock/frontend/src/store/streamStore.ts +68 -0
  106. flock/frontend/src/store/uiStore.test.ts +54 -0
  107. flock/frontend/src/store/uiStore.ts +110 -0
  108. flock/frontend/src/store/wsStore.ts +34 -0
  109. flock/frontend/src/styles/index.css +15 -0
  110. flock/frontend/src/styles/scrollbar.css +47 -0
  111. flock/frontend/src/styles/variables.css +488 -0
  112. flock/frontend/src/test/setup.ts +1 -0
  113. flock/frontend/src/types/filters.ts +14 -0
  114. flock/frontend/src/types/graph.ts +55 -0
  115. flock/frontend/src/types/modules.ts +7 -0
  116. flock/frontend/src/types/theme.ts +55 -0
  117. flock/frontend/src/utils/mockData.ts +85 -0
  118. flock/frontend/src/utils/performance.ts +16 -0
  119. flock/frontend/src/utils/transforms.test.ts +860 -0
  120. flock/frontend/src/utils/transforms.ts +323 -0
  121. flock/frontend/src/vite-env.d.ts +17 -0
  122. flock/frontend/tsconfig.json +27 -0
  123. flock/frontend/tsconfig.node.json +11 -0
  124. flock/frontend/vite.config.ts +25 -0
  125. flock/frontend/vitest.config.ts +11 -0
  126. flock/{core/util → helper}/cli_helper.py +4 -3
  127. flock/{core/logging → logging}/__init__.py +2 -3
  128. flock/{core/logging → logging}/formatters/enum_builder.py +3 -4
  129. flock/{core/logging → logging}/formatters/theme_builder.py +19 -44
  130. flock/{core/logging → logging}/formatters/themed_formatter.py +69 -115
  131. flock/{core/logging → logging}/logging.py +77 -61
  132. flock/{core/logging → logging}/telemetry.py +20 -26
  133. flock/{core/logging → logging}/telemetry_exporter/base_exporter.py +2 -2
  134. flock/{core/logging → logging}/telemetry_exporter/file_exporter.py +6 -9
  135. flock/{core/logging → logging}/telemetry_exporter/sqlite_exporter.py +2 -3
  136. flock/{core/logging → logging}/trace_and_logged.py +20 -24
  137. flock/mcp/__init__.py +91 -0
  138. flock/{core/mcp/mcp_client.py → mcp/client.py} +103 -154
  139. flock/{core/mcp/mcp_config.py → mcp/config.py} +62 -117
  140. flock/mcp/manager.py +255 -0
  141. flock/mcp/servers/sse/__init__.py +1 -1
  142. flock/mcp/servers/sse/flock_sse_server.py +11 -53
  143. flock/mcp/servers/stdio/__init__.py +1 -1
  144. flock/mcp/servers/stdio/flock_stdio_server.py +8 -48
  145. flock/mcp/servers/streamable_http/flock_streamable_http_server.py +17 -62
  146. flock/mcp/servers/websockets/flock_websocket_server.py +7 -40
  147. flock/{core/mcp/flock_mcp_tool.py → mcp/tool.py} +16 -26
  148. flock/mcp/types/__init__.py +42 -0
  149. flock/{core/mcp → mcp}/types/callbacks.py +9 -15
  150. flock/{core/mcp → mcp}/types/factories.py +7 -6
  151. flock/{core/mcp → mcp}/types/handlers.py +13 -18
  152. flock/{core/mcp → mcp}/types/types.py +70 -74
  153. flock/{core/mcp → mcp}/util/helpers.py +1 -1
  154. flock/orchestrator.py +645 -0
  155. flock/registry.py +148 -0
  156. flock/runtime.py +262 -0
  157. flock/service.py +140 -0
  158. flock/store.py +69 -0
  159. flock/subscription.py +111 -0
  160. flock/themes/andromeda.toml +1 -1
  161. flock/themes/apple-system-colors.toml +1 -1
  162. flock/themes/arcoiris.toml +1 -1
  163. flock/themes/atomonelight.toml +1 -1
  164. flock/themes/ayu copy.toml +1 -1
  165. flock/themes/ayu-light.toml +1 -1
  166. flock/themes/belafonte-day.toml +1 -1
  167. flock/themes/belafonte-night.toml +1 -1
  168. flock/themes/blulocodark.toml +1 -1
  169. flock/themes/breeze.toml +1 -1
  170. flock/themes/broadcast.toml +1 -1
  171. flock/themes/brogrammer.toml +1 -1
  172. flock/themes/builtin-dark.toml +1 -1
  173. flock/themes/builtin-pastel-dark.toml +1 -1
  174. flock/themes/catppuccin-latte.toml +1 -1
  175. flock/themes/catppuccin-macchiato.toml +1 -1
  176. flock/themes/catppuccin-mocha.toml +1 -1
  177. flock/themes/cga.toml +1 -1
  178. flock/themes/chalk.toml +1 -1
  179. flock/themes/ciapre.toml +1 -1
  180. flock/themes/coffee-theme.toml +1 -1
  181. flock/themes/cyberpunkscarletprotocol.toml +1 -1
  182. flock/themes/dark+.toml +1 -1
  183. flock/themes/darkermatrix.toml +1 -1
  184. flock/themes/darkside.toml +1 -1
  185. flock/themes/desert.toml +1 -1
  186. flock/themes/django.toml +1 -1
  187. flock/themes/djangosmooth.toml +1 -1
  188. flock/themes/doomone.toml +1 -1
  189. flock/themes/dotgov.toml +1 -1
  190. flock/themes/dracula+.toml +1 -1
  191. flock/themes/duckbones.toml +1 -1
  192. flock/themes/encom.toml +1 -1
  193. flock/themes/espresso.toml +1 -1
  194. flock/themes/everblush.toml +1 -1
  195. flock/themes/fairyfloss.toml +1 -1
  196. flock/themes/fideloper.toml +1 -1
  197. flock/themes/fishtank.toml +1 -1
  198. flock/themes/flexoki-light.toml +1 -1
  199. flock/themes/floraverse.toml +1 -1
  200. flock/themes/framer.toml +1 -1
  201. flock/themes/galizur.toml +1 -1
  202. flock/themes/github.toml +1 -1
  203. flock/themes/grass.toml +1 -1
  204. flock/themes/grey-green.toml +1 -1
  205. flock/themes/gruvboxlight.toml +1 -1
  206. flock/themes/guezwhoz.toml +1 -1
  207. flock/themes/harper.toml +1 -1
  208. flock/themes/hax0r-blue.toml +1 -1
  209. flock/themes/hopscotch.256.toml +1 -1
  210. flock/themes/ic-green-ppl.toml +1 -1
  211. flock/themes/iceberg-dark.toml +1 -1
  212. flock/themes/japanesque.toml +1 -1
  213. flock/themes/jubi.toml +1 -1
  214. flock/themes/kibble.toml +1 -1
  215. flock/themes/kolorit.toml +1 -1
  216. flock/themes/kurokula.toml +1 -1
  217. flock/themes/materialdesigncolors.toml +1 -1
  218. flock/themes/matrix.toml +1 -1
  219. flock/themes/mellifluous.toml +1 -1
  220. flock/themes/midnight-in-mojave.toml +1 -1
  221. flock/themes/monokai-remastered.toml +1 -1
  222. flock/themes/monokai-soda.toml +1 -1
  223. flock/themes/neon.toml +1 -1
  224. flock/themes/neopolitan.toml +1 -1
  225. flock/themes/nord-light.toml +1 -1
  226. flock/themes/ocean.toml +1 -1
  227. flock/themes/onehalfdark.toml +1 -1
  228. flock/themes/onehalflight.toml +1 -1
  229. flock/themes/palenighthc.toml +1 -1
  230. flock/themes/paulmillr.toml +1 -1
  231. flock/themes/pencildark.toml +1 -1
  232. flock/themes/pnevma.toml +1 -1
  233. flock/themes/purple-rain.toml +1 -1
  234. flock/themes/purplepeter.toml +1 -1
  235. flock/themes/raycast-dark.toml +1 -1
  236. flock/themes/red-sands.toml +1 -1
  237. flock/themes/relaxed.toml +1 -1
  238. flock/themes/retro.toml +1 -1
  239. flock/themes/rose-pine.toml +1 -1
  240. flock/themes/royal.toml +1 -1
  241. flock/themes/ryuuko.toml +1 -1
  242. flock/themes/sakura.toml +1 -1
  243. flock/themes/scarlet-protocol.toml +1 -1
  244. flock/themes/seoulbones-dark.toml +1 -1
  245. flock/themes/shades-of-purple.toml +1 -1
  246. flock/themes/smyck.toml +1 -1
  247. flock/themes/softserver.toml +1 -1
  248. flock/themes/solarized-darcula.toml +1 -1
  249. flock/themes/square.toml +1 -1
  250. flock/themes/sugarplum.toml +1 -1
  251. flock/themes/thayer-bright.toml +1 -1
  252. flock/themes/tokyonight.toml +1 -1
  253. flock/themes/tomorrow.toml +1 -1
  254. flock/themes/ubuntu.toml +1 -1
  255. flock/themes/ultradark.toml +1 -1
  256. flock/themes/ultraviolent.toml +1 -1
  257. flock/themes/unikitty.toml +1 -1
  258. flock/themes/urple.toml +1 -1
  259. flock/themes/vesper.toml +1 -1
  260. flock/themes/vimbones.toml +1 -1
  261. flock/themes/wildcherry.toml +1 -1
  262. flock/themes/wilmersdorf.toml +1 -1
  263. flock/themes/wryan.toml +1 -1
  264. flock/themes/xcodedarkhc.toml +1 -1
  265. flock/themes/xcodelight.toml +1 -1
  266. flock/themes/zenbones-light.toml +1 -1
  267. flock/themes/zenwritten-dark.toml +1 -1
  268. flock/utilities.py +301 -0
  269. flock/{components/utility → utility}/output_utility_component.py +68 -53
  270. flock/visibility.py +107 -0
  271. flock_core-0.5.0b51.dist-info/METADATA +747 -0
  272. flock_core-0.5.0b51.dist-info/RECORD +508 -0
  273. flock_core-0.5.0b51.dist-info/entry_points.txt +2 -0
  274. {flock_core-0.5.0b28.dist-info → flock_core-0.5.0b51.dist-info}/licenses/LICENSE +1 -1
  275. flock/adapter/__init__.py +0 -14
  276. flock/adapter/azure_adapter.py +0 -68
  277. flock/adapter/chroma_adapter.py +0 -73
  278. flock/adapter/faiss_adapter.py +0 -97
  279. flock/adapter/pinecone_adapter.py +0 -51
  280. flock/adapter/vector_base.py +0 -47
  281. flock/cli/assets/release_notes.md +0 -140
  282. flock/cli/config.py +0 -8
  283. flock/cli/constants.py +0 -36
  284. flock/cli/create_agent.py +0 -1
  285. flock/cli/create_flock.py +0 -280
  286. flock/cli/execute_flock.py +0 -620
  287. flock/cli/load_agent.py +0 -1
  288. flock/cli/load_examples.py +0 -1
  289. flock/cli/load_flock.py +0 -192
  290. flock/cli/load_release_notes.py +0 -20
  291. flock/cli/loaded_flock_cli.py +0 -254
  292. flock/cli/manage_agents.py +0 -459
  293. flock/cli/registry_management.py +0 -889
  294. flock/cli/runner.py +0 -41
  295. flock/cli/settings.py +0 -857
  296. flock/cli/utils.py +0 -135
  297. flock/cli/view_results.py +0 -29
  298. flock/cli/yaml_editor.py +0 -396
  299. flock/components/__init__.py +0 -30
  300. flock/components/evaluation/__init__.py +0 -9
  301. flock/components/evaluation/declarative_evaluation_component.py +0 -606
  302. flock/components/routing/__init__.py +0 -15
  303. flock/components/routing/conditional_routing_component.py +0 -494
  304. flock/components/routing/default_routing_component.py +0 -103
  305. flock/components/routing/llm_routing_component.py +0 -206
  306. flock/components/utility/__init__.py +0 -22
  307. flock/components/utility/example_utility_component.py +0 -250
  308. flock/components/utility/feedback_utility_component.py +0 -206
  309. flock/components/utility/memory_utility_component.py +0 -550
  310. flock/components/utility/metrics_utility_component.py +0 -700
  311. flock/config.py +0 -61
  312. flock/core/__init__.py +0 -110
  313. flock/core/agent/__init__.py +0 -16
  314. flock/core/agent/default_agent.py +0 -216
  315. flock/core/agent/flock_agent_components.py +0 -104
  316. flock/core/agent/flock_agent_execution.py +0 -101
  317. flock/core/agent/flock_agent_integration.py +0 -260
  318. flock/core/agent/flock_agent_lifecycle.py +0 -186
  319. flock/core/agent/flock_agent_serialization.py +0 -381
  320. flock/core/api/__init__.py +0 -10
  321. flock/core/api/custom_endpoint.py +0 -45
  322. flock/core/api/endpoints.py +0 -254
  323. flock/core/api/main.py +0 -162
  324. flock/core/api/models.py +0 -97
  325. flock/core/api/run_store.py +0 -224
  326. flock/core/api/runner.py +0 -44
  327. flock/core/api/service.py +0 -214
  328. flock/core/component/__init__.py +0 -15
  329. flock/core/component/agent_component_base.py +0 -309
  330. flock/core/component/evaluation_component.py +0 -62
  331. flock/core/component/routing_component.py +0 -74
  332. flock/core/component/utility_component.py +0 -69
  333. flock/core/config/flock_agent_config.py +0 -58
  334. flock/core/config/scheduled_agent_config.py +0 -40
  335. flock/core/context/context.py +0 -213
  336. flock/core/context/context_manager.py +0 -37
  337. flock/core/context/context_vars.py +0 -10
  338. flock/core/evaluation/utils.py +0 -396
  339. flock/core/execution/batch_executor.py +0 -369
  340. flock/core/execution/evaluation_executor.py +0 -438
  341. flock/core/execution/local_executor.py +0 -31
  342. flock/core/execution/opik_executor.py +0 -103
  343. flock/core/execution/temporal_executor.py +0 -164
  344. flock/core/flock.py +0 -634
  345. flock/core/flock_agent.py +0 -336
  346. flock/core/flock_factory.py +0 -613
  347. flock/core/flock_scheduler.py +0 -166
  348. flock/core/flock_server_manager.py +0 -136
  349. flock/core/interpreter/python_interpreter.py +0 -689
  350. flock/core/mcp/__init__.py +0 -1
  351. flock/core/mcp/flock_mcp_server.py +0 -680
  352. flock/core/mcp/mcp_client_manager.py +0 -201
  353. flock/core/mcp/types/__init__.py +0 -1
  354. flock/core/mixin/dspy_integration.py +0 -403
  355. flock/core/mixin/prompt_parser.py +0 -125
  356. flock/core/orchestration/__init__.py +0 -15
  357. flock/core/orchestration/flock_batch_processor.py +0 -94
  358. flock/core/orchestration/flock_evaluator.py +0 -113
  359. flock/core/orchestration/flock_execution.py +0 -295
  360. flock/core/orchestration/flock_initialization.py +0 -149
  361. flock/core/orchestration/flock_server_manager.py +0 -67
  362. flock/core/orchestration/flock_web_server.py +0 -117
  363. flock/core/registry/__init__.py +0 -45
  364. flock/core/registry/agent_registry.py +0 -69
  365. flock/core/registry/callable_registry.py +0 -139
  366. flock/core/registry/component_discovery.py +0 -142
  367. flock/core/registry/component_registry.py +0 -64
  368. flock/core/registry/config_mapping.py +0 -64
  369. flock/core/registry/decorators.py +0 -137
  370. flock/core/registry/registry_hub.py +0 -205
  371. flock/core/registry/server_registry.py +0 -57
  372. flock/core/registry/type_registry.py +0 -86
  373. flock/core/serialization/__init__.py +0 -13
  374. flock/core/serialization/callable_registry.py +0 -52
  375. flock/core/serialization/flock_serializer.py +0 -832
  376. flock/core/serialization/json_encoder.py +0 -41
  377. flock/core/serialization/secure_serializer.py +0 -175
  378. flock/core/serialization/serializable.py +0 -342
  379. flock/core/serialization/serialization_utils.py +0 -412
  380. flock/core/util/file_path_utils.py +0 -223
  381. flock/core/util/hydrator.py +0 -309
  382. flock/core/util/input_resolver.py +0 -164
  383. flock/core/util/loader.py +0 -59
  384. flock/core/util/splitter.py +0 -219
  385. flock/di.py +0 -27
  386. flock/platform/docker_tools.py +0 -49
  387. flock/platform/jaeger_install.py +0 -86
  388. flock/webapp/__init__.py +0 -1
  389. flock/webapp/app/__init__.py +0 -0
  390. flock/webapp/app/api/__init__.py +0 -0
  391. flock/webapp/app/api/agent_management.py +0 -241
  392. flock/webapp/app/api/execution.py +0 -709
  393. flock/webapp/app/api/flock_management.py +0 -129
  394. flock/webapp/app/api/registry_viewer.py +0 -30
  395. flock/webapp/app/chat.py +0 -665
  396. flock/webapp/app/config.py +0 -104
  397. flock/webapp/app/dependencies.py +0 -117
  398. flock/webapp/app/main.py +0 -1070
  399. flock/webapp/app/middleware.py +0 -113
  400. flock/webapp/app/models_ui.py +0 -7
  401. flock/webapp/app/services/__init__.py +0 -0
  402. flock/webapp/app/services/feedback_file_service.py +0 -363
  403. flock/webapp/app/services/flock_service.py +0 -337
  404. flock/webapp/app/services/sharing_models.py +0 -81
  405. flock/webapp/app/services/sharing_store.py +0 -762
  406. flock/webapp/app/templates/theme_mapper.html +0 -326
  407. flock/webapp/app/theme_mapper.py +0 -812
  408. flock/webapp/app/utils.py +0 -85
  409. flock/webapp/run.py +0 -215
  410. flock/webapp/static/css/chat.css +0 -301
  411. flock/webapp/static/css/components.css +0 -167
  412. flock/webapp/static/css/header.css +0 -39
  413. flock/webapp/static/css/layout.css +0 -46
  414. flock/webapp/static/css/sidebar.css +0 -127
  415. flock/webapp/static/css/two-pane.css +0 -48
  416. flock/webapp/templates/base.html +0 -200
  417. flock/webapp/templates/chat.html +0 -152
  418. flock/webapp/templates/chat_settings.html +0 -19
  419. flock/webapp/templates/flock_editor.html +0 -16
  420. flock/webapp/templates/index.html +0 -12
  421. flock/webapp/templates/partials/_agent_detail_form.html +0 -93
  422. flock/webapp/templates/partials/_agent_list.html +0 -18
  423. flock/webapp/templates/partials/_agent_manager_view.html +0 -51
  424. flock/webapp/templates/partials/_agent_tools_checklist.html +0 -14
  425. flock/webapp/templates/partials/_chat_container.html +0 -15
  426. flock/webapp/templates/partials/_chat_messages.html +0 -57
  427. flock/webapp/templates/partials/_chat_settings_form.html +0 -85
  428. flock/webapp/templates/partials/_create_flock_form.html +0 -50
  429. flock/webapp/templates/partials/_dashboard_flock_detail.html +0 -17
  430. flock/webapp/templates/partials/_dashboard_flock_file_list.html +0 -16
  431. flock/webapp/templates/partials/_dashboard_flock_properties_preview.html +0 -28
  432. flock/webapp/templates/partials/_dashboard_upload_flock_form.html +0 -16
  433. flock/webapp/templates/partials/_dynamic_input_form_content.html +0 -22
  434. flock/webapp/templates/partials/_env_vars_table.html +0 -23
  435. flock/webapp/templates/partials/_execution_form.html +0 -118
  436. flock/webapp/templates/partials/_execution_view_container.html +0 -28
  437. flock/webapp/templates/partials/_flock_file_list.html +0 -23
  438. flock/webapp/templates/partials/_flock_properties_form.html +0 -52
  439. flock/webapp/templates/partials/_flock_upload_form.html +0 -16
  440. flock/webapp/templates/partials/_header_flock_status.html +0 -5
  441. flock/webapp/templates/partials/_load_manager_view.html +0 -49
  442. flock/webapp/templates/partials/_registry_table.html +0 -25
  443. flock/webapp/templates/partials/_registry_viewer_content.html +0 -70
  444. flock/webapp/templates/partials/_results_display.html +0 -78
  445. flock/webapp/templates/partials/_settings_env_content.html +0 -9
  446. flock/webapp/templates/partials/_settings_theme_content.html +0 -14
  447. flock/webapp/templates/partials/_settings_view.html +0 -36
  448. flock/webapp/templates/partials/_share_chat_link_snippet.html +0 -11
  449. flock/webapp/templates/partials/_share_link_snippet.html +0 -35
  450. flock/webapp/templates/partials/_sidebar.html +0 -74
  451. flock/webapp/templates/partials/_streaming_results_container.html +0 -195
  452. flock/webapp/templates/partials/_structured_data_view.html +0 -40
  453. flock/webapp/templates/partials/_theme_preview.html +0 -36
  454. flock/webapp/templates/registry_viewer.html +0 -84
  455. flock/webapp/templates/shared_run_page.html +0 -140
  456. flock/workflow/__init__.py +0 -0
  457. flock/workflow/activities.py +0 -196
  458. flock/workflow/agent_activities.py +0 -24
  459. flock/workflow/agent_execution_activity.py +0 -202
  460. flock/workflow/flock_workflow.py +0 -214
  461. flock/workflow/temporal_config.py +0 -96
  462. flock/workflow/temporal_setup.py +0 -68
  463. flock_core-0.5.0b28.dist-info/METADATA +0 -274
  464. flock_core-0.5.0b28.dist-info/RECORD +0 -561
  465. flock_core-0.5.0b28.dist-info/entry_points.txt +0 -2
  466. /flock/{core/logging → logging}/formatters/themes.py +0 -0
  467. /flock/{core/logging → logging}/span_middleware/baggage_span_processor.py +0 -0
  468. /flock/{core/mcp → mcp}/util/__init__.py +0 -0
  469. {flock_core-0.5.0b28.dist-info → flock_core-0.5.0b51.dist-info}/WHEEL +0 -0
@@ -0,0 +1,442 @@
1
+ import { describe, it, expect, beforeEach, vi, afterEach } from 'vitest';
2
+ import { renderHook, waitFor } from '@testing-library/react';
3
+ import { useModulePersistence } from './useModulePersistence';
4
+ import { useModuleStore } from '../store/moduleStore';
5
+ import { indexedDBService } from '../services/indexeddb';
6
+ import type { ModuleInstance } from '../types/modules';
7
+
8
+ // Mock IndexedDB service
9
+ vi.mock('../services/indexeddb', () => ({
10
+ indexedDBService: {
11
+ getAllModuleInstances: vi.fn(),
12
+ saveModuleInstance: vi.fn(),
13
+ deleteModuleInstance: vi.fn(),
14
+ },
15
+ }));
16
+
17
+ describe('useModulePersistence', () => {
18
+ beforeEach(() => {
19
+ vi.clearAllMocks();
20
+ // Reset module store
21
+ useModuleStore.setState({ instances: new Map() });
22
+ });
23
+
24
+ afterEach(() => {
25
+ vi.useRealTimers();
26
+ });
27
+
28
+ describe('Module Loading', () => {
29
+ it('should load saved module instances on mount', async () => {
30
+ const savedInstances = [
31
+ {
32
+ instance_id: 'module-1',
33
+ type: 'eventLog',
34
+ position: { x: 100, y: 200 },
35
+ size: { width: 600, height: 400 },
36
+ visible: true,
37
+ created_at: '2025-10-03T00:00:00Z',
38
+ updated_at: '2025-10-03T00:00:00Z',
39
+ },
40
+ {
41
+ instance_id: 'module-2',
42
+ type: 'eventLog',
43
+ position: { x: 300, y: 400 },
44
+ size: { width: 800, height: 600 },
45
+ visible: true,
46
+ created_at: '2025-10-03T00:00:00Z',
47
+ updated_at: '2025-10-03T00:00:00Z',
48
+ },
49
+ ];
50
+
51
+ vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue(savedInstances);
52
+
53
+ renderHook(() => useModulePersistence());
54
+
55
+ await waitFor(() => {
56
+ const instances = useModuleStore.getState().instances;
57
+ expect(instances.size).toBe(2);
58
+ expect(instances.get('module-1')).toEqual({
59
+ id: 'module-1',
60
+ type: 'eventLog',
61
+ position: { x: 100, y: 200 },
62
+ size: { width: 600, height: 400 },
63
+ visible: true,
64
+ });
65
+ expect(instances.get('module-2')).toEqual({
66
+ id: 'module-2',
67
+ type: 'eventLog',
68
+ position: { x: 300, y: 400 },
69
+ size: { width: 800, height: 600 },
70
+ visible: true,
71
+ });
72
+ });
73
+ });
74
+
75
+ it('should handle empty module instances on mount', async () => {
76
+ vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
77
+
78
+ renderHook(() => useModulePersistence());
79
+
80
+ await waitFor(() => {
81
+ expect(indexedDBService.getAllModuleInstances).toHaveBeenCalledOnce();
82
+ });
83
+
84
+ const instances = useModuleStore.getState().instances;
85
+ expect(instances.size).toBe(0);
86
+ });
87
+
88
+ it('should handle errors when loading instances', async () => {
89
+ const consoleErrorSpy = vi.spyOn(console, 'error').mockImplementation(() => {});
90
+ vi.mocked(indexedDBService.getAllModuleInstances).mockRejectedValue(
91
+ new Error('Database error')
92
+ );
93
+
94
+ renderHook(() => useModulePersistence());
95
+
96
+ await waitFor(() => {
97
+ expect(consoleErrorSpy).toHaveBeenCalledWith(
98
+ '[ModulePersistence] Failed to load module instances:',
99
+ expect.any(Error)
100
+ );
101
+ });
102
+
103
+ consoleErrorSpy.mockRestore();
104
+ });
105
+ });
106
+
107
+ describe('Module Saving', () => {
108
+ it('should save new module instance when added', async () => {
109
+ vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
110
+
111
+ renderHook(() => useModulePersistence());
112
+
113
+ await waitFor(() => {
114
+ expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
115
+ });
116
+
117
+ const newInstance: ModuleInstance = {
118
+ id: 'new-module',
119
+ type: 'eventLog',
120
+ position: { x: 150, y: 250 },
121
+ size: { width: 700, height: 500 },
122
+ visible: true,
123
+ };
124
+
125
+ useModuleStore.getState().addModule(newInstance);
126
+
127
+ // Wait for debounced save (300ms + buffer)
128
+ await new Promise((resolve) => setTimeout(resolve, 400));
129
+
130
+ await waitFor(() => {
131
+ expect(indexedDBService.saveModuleInstance).toHaveBeenCalledWith(
132
+ expect.objectContaining({
133
+ instance_id: 'new-module',
134
+ type: 'eventLog',
135
+ position: { x: 150, y: 250 },
136
+ size: { width: 700, height: 500 },
137
+ visible: true,
138
+ })
139
+ );
140
+ });
141
+ });
142
+
143
+ it('should debounce saves when position changes rapidly', async () => {
144
+ vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
145
+
146
+ renderHook(() => useModulePersistence());
147
+
148
+ await waitFor(() => {
149
+ expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
150
+ });
151
+
152
+ const instance: ModuleInstance = {
153
+ id: 'drag-module',
154
+ type: 'eventLog',
155
+ position: { x: 100, y: 100 },
156
+ size: { width: 600, height: 400 },
157
+ visible: true,
158
+ };
159
+
160
+ useModuleStore.getState().addModule(instance);
161
+
162
+ // Wait for initial add to save
163
+ await new Promise((resolve) => setTimeout(resolve, 400));
164
+
165
+ // Rapidly update position (simulating drag)
166
+ useModuleStore.getState().updateModule('drag-module', { position: { x: 110, y: 110 } });
167
+ await new Promise((resolve) => setTimeout(resolve, 100));
168
+
169
+ useModuleStore.getState().updateModule('drag-module', { position: { x: 120, y: 120 } });
170
+ await new Promise((resolve) => setTimeout(resolve, 100));
171
+
172
+ useModuleStore.getState().updateModule('drag-module', { position: { x: 130, y: 130 } });
173
+
174
+ // Wait for debounced save
175
+ await new Promise((resolve) => setTimeout(resolve, 400));
176
+
177
+ await waitFor(() => {
178
+ // Should have saved at least twice: initial add + final position
179
+ expect(indexedDBService.saveModuleInstance).toHaveBeenCalled();
180
+ expect(indexedDBService.saveModuleInstance).toHaveBeenLastCalledWith(
181
+ expect.objectContaining({
182
+ position: { x: 130, y: 130 },
183
+ })
184
+ );
185
+ });
186
+ });
187
+
188
+ it('should save when size changes', async () => {
189
+ vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
190
+
191
+ renderHook(() => useModulePersistence());
192
+
193
+ await waitFor(() => {
194
+ expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
195
+ });
196
+
197
+ const instance: ModuleInstance = {
198
+ id: 'resize-module',
199
+ type: 'eventLog',
200
+ position: { x: 100, y: 100 },
201
+ size: { width: 600, height: 400 },
202
+ visible: true,
203
+ };
204
+
205
+ useModuleStore.getState().addModule(instance);
206
+ await new Promise((resolve) => setTimeout(resolve, 400));
207
+
208
+ // Update size
209
+ useModuleStore.getState().updateModule('resize-module', {
210
+ size: { width: 800, height: 600 },
211
+ });
212
+
213
+ await new Promise((resolve) => setTimeout(resolve, 400));
214
+
215
+ await waitFor(() => {
216
+ expect(indexedDBService.saveModuleInstance).toHaveBeenCalledWith(
217
+ expect.objectContaining({
218
+ size: { width: 800, height: 600 },
219
+ })
220
+ );
221
+ });
222
+ });
223
+
224
+ it('should save when visibility changes', async () => {
225
+ vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
226
+
227
+ renderHook(() => useModulePersistence());
228
+
229
+ await waitFor(() => {
230
+ expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
231
+ });
232
+
233
+ const instance: ModuleInstance = {
234
+ id: 'toggle-module',
235
+ type: 'eventLog',
236
+ position: { x: 100, y: 100 },
237
+ size: { width: 600, height: 400 },
238
+ visible: true,
239
+ };
240
+
241
+ useModuleStore.getState().addModule(instance);
242
+ await new Promise((resolve) => setTimeout(resolve, 400));
243
+
244
+ // Toggle visibility
245
+ useModuleStore.getState().toggleVisibility('toggle-module');
246
+
247
+ await new Promise((resolve) => setTimeout(resolve, 400));
248
+
249
+ await waitFor(() => {
250
+ expect(indexedDBService.saveModuleInstance).toHaveBeenCalledWith(
251
+ expect.objectContaining({
252
+ visible: false,
253
+ })
254
+ );
255
+ });
256
+ });
257
+
258
+ it('should handle save errors gracefully', async () => {
259
+ const consoleErrorSpy = vi.spyOn(console, 'error').mockImplementation(() => {});
260
+ vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
261
+ vi.mocked(indexedDBService.saveModuleInstance).mockRejectedValue(
262
+ new Error('Save failed')
263
+ );
264
+
265
+ renderHook(() => useModulePersistence());
266
+
267
+ await waitFor(() => {
268
+ expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
269
+ });
270
+
271
+ const instance: ModuleInstance = {
272
+ id: 'error-module',
273
+ type: 'eventLog',
274
+ position: { x: 100, y: 100 },
275
+ size: { width: 600, height: 400 },
276
+ visible: true,
277
+ };
278
+
279
+ useModuleStore.getState().addModule(instance);
280
+ await new Promise((resolve) => setTimeout(resolve, 400));
281
+
282
+ await waitFor(() => {
283
+ expect(consoleErrorSpy).toHaveBeenCalledWith(
284
+ expect.stringContaining('Failed to save instance'),
285
+ expect.any(Error)
286
+ );
287
+ });
288
+
289
+ consoleErrorSpy.mockRestore();
290
+ });
291
+ });
292
+
293
+ describe('Module Deletion', () => {
294
+ it('should delete module instance when removed', async () => {
295
+ vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
296
+
297
+ renderHook(() => useModulePersistence());
298
+
299
+ await waitFor(() => {
300
+ expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
301
+ });
302
+
303
+ const instance: ModuleInstance = {
304
+ id: 'delete-module',
305
+ type: 'eventLog',
306
+ position: { x: 100, y: 100 },
307
+ size: { width: 600, height: 400 },
308
+ visible: true,
309
+ };
310
+
311
+ useModuleStore.getState().addModule(instance);
312
+ await new Promise((resolve) => setTimeout(resolve, 400));
313
+
314
+ // Remove the instance
315
+ useModuleStore.getState().removeModule('delete-module');
316
+
317
+ await waitFor(() => {
318
+ expect(indexedDBService.deleteModuleInstance).toHaveBeenCalledWith('delete-module');
319
+ });
320
+ });
321
+
322
+ it('should cancel pending save when module is deleted', async () => {
323
+ vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
324
+
325
+ renderHook(() => useModulePersistence());
326
+
327
+ await waitFor(() => {
328
+ expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
329
+ });
330
+
331
+ const instance: ModuleInstance = {
332
+ id: 'cancel-module',
333
+ type: 'eventLog',
334
+ position: { x: 100, y: 100 },
335
+ size: { width: 600, height: 400 },
336
+ visible: true,
337
+ };
338
+
339
+ useModuleStore.getState().addModule(instance);
340
+
341
+ // Wait for initial add to save
342
+ await new Promise((resolve) => setTimeout(resolve, 400));
343
+
344
+ const initialCallCount = vi.mocked(indexedDBService.saveModuleInstance).mock.calls.length;
345
+
346
+ // Update position (triggers debounced save)
347
+ useModuleStore.getState().updateModule('cancel-module', { position: { x: 200, y: 200 } });
348
+
349
+ // Remove before debounce timer fires (100ms < 300ms debounce)
350
+ await new Promise((resolve) => setTimeout(resolve, 100));
351
+ useModuleStore.getState().removeModule('cancel-module');
352
+
353
+ // Wait for potential saves
354
+ await new Promise((resolve) => setTimeout(resolve, 400));
355
+
356
+ await waitFor(() => {
357
+ expect(indexedDBService.deleteModuleInstance).toHaveBeenCalledWith('cancel-module');
358
+ });
359
+
360
+ // save should not have been called again after the update (timer was cancelled)
361
+ const finalCallCount = vi.mocked(indexedDBService.saveModuleInstance).mock.calls.length;
362
+ expect(finalCallCount).toBe(initialCallCount);
363
+ });
364
+
365
+ it('should handle delete errors gracefully', async () => {
366
+ const consoleErrorSpy = vi.spyOn(console, 'error').mockImplementation(() => {});
367
+ vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
368
+ vi.mocked(indexedDBService.deleteModuleInstance).mockRejectedValue(
369
+ new Error('Delete failed')
370
+ );
371
+
372
+ renderHook(() => useModulePersistence());
373
+
374
+ await waitFor(() => {
375
+ expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
376
+ });
377
+
378
+ const instance: ModuleInstance = {
379
+ id: 'error-delete-module',
380
+ type: 'eventLog',
381
+ position: { x: 100, y: 100 },
382
+ size: { width: 600, height: 400 },
383
+ visible: true,
384
+ };
385
+
386
+ useModuleStore.getState().addModule(instance);
387
+ await new Promise((resolve) => setTimeout(resolve, 400));
388
+
389
+ useModuleStore.getState().removeModule('error-delete-module');
390
+
391
+ await waitFor(() => {
392
+ expect(consoleErrorSpy).toHaveBeenCalledWith(
393
+ expect.stringContaining('Failed to delete instance'),
394
+ expect.any(Error)
395
+ );
396
+ });
397
+
398
+ consoleErrorSpy.mockRestore();
399
+ });
400
+ });
401
+
402
+ describe('Timer Cleanup', () => {
403
+ it('should cleanup timers on unmount', async () => {
404
+ vi.mocked(indexedDBService.getAllModuleInstances).mockResolvedValue([]);
405
+
406
+ const { unmount } = renderHook(() => useModulePersistence());
407
+
408
+ await waitFor(() => {
409
+ expect(indexedDBService.getAllModuleInstances).toHaveBeenCalled();
410
+ });
411
+
412
+ const instance: ModuleInstance = {
413
+ id: 'cleanup-module',
414
+ type: 'eventLog',
415
+ position: { x: 100, y: 100 },
416
+ size: { width: 600, height: 400 },
417
+ visible: true,
418
+ };
419
+
420
+ useModuleStore.getState().addModule(instance);
421
+
422
+ // Wait for initial add to potentially save
423
+ await new Promise((resolve) => setTimeout(resolve, 400));
424
+
425
+ const callCountBeforeUpdate = vi.mocked(indexedDBService.saveModuleInstance).mock.calls.length;
426
+
427
+ // Update position (triggers debounced save)
428
+ useModuleStore.getState().updateModule('cleanup-module', { position: { x: 200, y: 200 } });
429
+
430
+ // Unmount before timer fires (100ms < 300ms debounce)
431
+ await new Promise((resolve) => setTimeout(resolve, 100));
432
+ unmount();
433
+
434
+ // Wait for potential timer - save should not be called after unmount
435
+ await new Promise((resolve) => setTimeout(resolve, 400));
436
+
437
+ // The update should NOT have saved because we unmounted before debounce completed
438
+ const finalCallCount = vi.mocked(indexedDBService.saveModuleInstance).mock.calls.length;
439
+ expect(finalCallCount).toBe(callCountBeforeUpdate);
440
+ });
441
+ });
442
+ });
@@ -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
+ };