flock-core 0.4.543__py3-none-any.whl → 0.5.0__py3-none-any.whl

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

Potentially problematic release.


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

Files changed (501) hide show
  1. flock/__init__.py +12 -217
  2. flock/agent.py +1079 -0
  3. flock/api/themes.py +71 -0
  4. flock/artifacts.py +86 -0
  5. flock/cli.py +147 -0
  6. flock/components.py +189 -0
  7. flock/dashboard/__init__.py +30 -0
  8. flock/dashboard/collector.py +559 -0
  9. flock/dashboard/events.py +188 -0
  10. flock/dashboard/graph_builder.py +563 -0
  11. flock/dashboard/launcher.py +235 -0
  12. flock/dashboard/models/graph.py +156 -0
  13. flock/dashboard/service.py +991 -0
  14. flock/dashboard/static_v2/assets/index-DFRnI_mt.js +111 -0
  15. flock/dashboard/static_v2/assets/index-fPLNdmp1.css +1 -0
  16. flock/dashboard/static_v2/index.html +13 -0
  17. flock/dashboard/websocket.py +246 -0
  18. flock/engines/__init__.py +6 -0
  19. flock/engines/dspy_engine.py +932 -0
  20. flock/examples.py +131 -0
  21. flock/frontend/README.md +778 -0
  22. flock/frontend/docs/DESIGN_SYSTEM.md +1980 -0
  23. flock/frontend/index.html +12 -0
  24. flock/frontend/package-lock.json +4337 -0
  25. flock/frontend/package.json +48 -0
  26. flock/frontend/src/App.tsx +139 -0
  27. flock/frontend/src/__tests__/integration/graph-snapshot.test.tsx +647 -0
  28. flock/frontend/src/__tests__/integration/indexeddb-persistence.test.tsx +699 -0
  29. flock/frontend/src/components/common/BuildInfo.tsx +39 -0
  30. flock/frontend/src/components/common/EmptyState.module.css +115 -0
  31. flock/frontend/src/components/common/EmptyState.tsx +128 -0
  32. flock/frontend/src/components/common/ErrorBoundary.module.css +169 -0
  33. flock/frontend/src/components/common/ErrorBoundary.tsx +118 -0
  34. flock/frontend/src/components/common/KeyboardShortcutsDialog.css +251 -0
  35. flock/frontend/src/components/common/KeyboardShortcutsDialog.tsx +151 -0
  36. flock/frontend/src/components/common/LoadingSpinner.module.css +97 -0
  37. flock/frontend/src/components/common/LoadingSpinner.tsx +29 -0
  38. flock/frontend/src/components/controls/PublishControl.css +547 -0
  39. flock/frontend/src/components/controls/PublishControl.test.tsx +543 -0
  40. flock/frontend/src/components/controls/PublishControl.tsx +432 -0
  41. flock/frontend/src/components/details/DetailWindowContainer.tsx +58 -0
  42. flock/frontend/src/components/details/LiveOutputTab.test.tsx +792 -0
  43. flock/frontend/src/components/details/LiveOutputTab.tsx +220 -0
  44. flock/frontend/src/components/details/MessageDetailWindow.tsx +439 -0
  45. flock/frontend/src/components/details/MessageHistoryTab.tsx +374 -0
  46. flock/frontend/src/components/details/NodeDetailWindow.test.tsx +501 -0
  47. flock/frontend/src/components/details/NodeDetailWindow.tsx +218 -0
  48. flock/frontend/src/components/details/RunStatusTab.tsx +348 -0
  49. flock/frontend/src/components/details/tabs.test.tsx +1015 -0
  50. flock/frontend/src/components/filters/ArtifactTypeFilter.tsx +21 -0
  51. flock/frontend/src/components/filters/CorrelationIDFilter.module.css +102 -0
  52. flock/frontend/src/components/filters/CorrelationIDFilter.test.tsx +197 -0
  53. flock/frontend/src/components/filters/CorrelationIDFilter.tsx +121 -0
  54. flock/frontend/src/components/filters/FilterFlyout.module.css +104 -0
  55. flock/frontend/src/components/filters/FilterFlyout.tsx +80 -0
  56. flock/frontend/src/components/filters/FilterPills.module.css +220 -0
  57. flock/frontend/src/components/filters/FilterPills.test.tsx +189 -0
  58. flock/frontend/src/components/filters/FilterPills.tsx +143 -0
  59. flock/frontend/src/components/filters/ProducerFilter.tsx +21 -0
  60. flock/frontend/src/components/filters/SavedFiltersControl.module.css +60 -0
  61. flock/frontend/src/components/filters/SavedFiltersControl.test.tsx +158 -0
  62. flock/frontend/src/components/filters/SavedFiltersControl.tsx +159 -0
  63. flock/frontend/src/components/filters/TagFilter.tsx +21 -0
  64. flock/frontend/src/components/filters/TimeRangeFilter.module.css +115 -0
  65. flock/frontend/src/components/filters/TimeRangeFilter.test.tsx +154 -0
  66. flock/frontend/src/components/filters/TimeRangeFilter.tsx +110 -0
  67. flock/frontend/src/components/filters/VisibilityFilter.tsx +21 -0
  68. flock/frontend/src/components/graph/AgentNode.test.tsx +77 -0
  69. flock/frontend/src/components/graph/AgentNode.tsx +324 -0
  70. flock/frontend/src/components/graph/GraphCanvas.tsx +613 -0
  71. flock/frontend/src/components/graph/MessageFlowEdge.tsx +128 -0
  72. flock/frontend/src/components/graph/MessageNode.test.tsx +64 -0
  73. flock/frontend/src/components/graph/MessageNode.tsx +129 -0
  74. flock/frontend/src/components/graph/MiniMap.tsx +47 -0
  75. flock/frontend/src/components/graph/TransformEdge.tsx +123 -0
  76. flock/frontend/src/components/layout/DashboardLayout.css +420 -0
  77. flock/frontend/src/components/layout/DashboardLayout.tsx +287 -0
  78. flock/frontend/src/components/layout/Header.module.css +88 -0
  79. flock/frontend/src/components/layout/Header.tsx +52 -0
  80. flock/frontend/src/components/modules/HistoricalArtifactsModule.module.css +288 -0
  81. flock/frontend/src/components/modules/HistoricalArtifactsModule.tsx +450 -0
  82. flock/frontend/src/components/modules/HistoricalArtifactsModuleWrapper.tsx +13 -0
  83. flock/frontend/src/components/modules/JsonAttributeRenderer.tsx +140 -0
  84. flock/frontend/src/components/modules/ModuleRegistry.test.ts +333 -0
  85. flock/frontend/src/components/modules/ModuleRegistry.ts +93 -0
  86. flock/frontend/src/components/modules/ModuleWindow.tsx +223 -0
  87. flock/frontend/src/components/modules/TraceModuleJaeger.tsx +1971 -0
  88. flock/frontend/src/components/modules/TraceModuleJaegerWrapper.tsx +13 -0
  89. flock/frontend/src/components/modules/registerModules.ts +29 -0
  90. flock/frontend/src/components/settings/AdvancedSettings.tsx +175 -0
  91. flock/frontend/src/components/settings/AppearanceSettings.tsx +185 -0
  92. flock/frontend/src/components/settings/GraphSettings.tsx +110 -0
  93. flock/frontend/src/components/settings/MultiSelect.tsx +235 -0
  94. flock/frontend/src/components/settings/SettingsPanel.css +327 -0
  95. flock/frontend/src/components/settings/SettingsPanel.tsx +131 -0
  96. flock/frontend/src/components/settings/ThemeSelector.tsx +298 -0
  97. flock/frontend/src/components/settings/TracingSettings.tsx +404 -0
  98. flock/frontend/src/hooks/useKeyboardShortcuts.ts +148 -0
  99. flock/frontend/src/hooks/useModulePersistence.test.ts +442 -0
  100. flock/frontend/src/hooks/useModulePersistence.ts +154 -0
  101. flock/frontend/src/hooks/useModules.ts +157 -0
  102. flock/frontend/src/hooks/usePersistence.ts +141 -0
  103. flock/frontend/src/main.tsx +13 -0
  104. flock/frontend/src/services/api.ts +337 -0
  105. flock/frontend/src/services/graphService.test.ts +330 -0
  106. flock/frontend/src/services/graphService.ts +75 -0
  107. flock/frontend/src/services/indexeddb.test.ts +793 -0
  108. flock/frontend/src/services/indexeddb.ts +848 -0
  109. flock/frontend/src/services/layout.test.ts +437 -0
  110. flock/frontend/src/services/layout.ts +357 -0
  111. flock/frontend/src/services/themeApplicator.ts +140 -0
  112. flock/frontend/src/services/themeService.ts +77 -0
  113. flock/frontend/src/services/websocket.ts +650 -0
  114. flock/frontend/src/store/filterStore.test.ts +250 -0
  115. flock/frontend/src/store/filterStore.ts +272 -0
  116. flock/frontend/src/store/graphStore.test.ts +570 -0
  117. flock/frontend/src/store/graphStore.ts +462 -0
  118. flock/frontend/src/store/moduleStore.test.ts +253 -0
  119. flock/frontend/src/store/moduleStore.ts +75 -0
  120. flock/frontend/src/store/settingsStore.ts +188 -0
  121. flock/frontend/src/store/streamStore.ts +68 -0
  122. flock/frontend/src/store/uiStore.test.ts +54 -0
  123. flock/frontend/src/store/uiStore.ts +122 -0
  124. flock/frontend/src/store/wsStore.ts +34 -0
  125. flock/frontend/src/styles/index.css +15 -0
  126. flock/frontend/src/styles/scrollbar.css +47 -0
  127. flock/frontend/src/styles/variables.css +488 -0
  128. flock/frontend/src/test/setup.ts +1 -0
  129. flock/frontend/src/types/filters.ts +47 -0
  130. flock/frontend/src/types/graph.ts +95 -0
  131. flock/frontend/src/types/modules.ts +10 -0
  132. flock/frontend/src/types/theme.ts +55 -0
  133. flock/frontend/src/utils/artifacts.ts +24 -0
  134. flock/frontend/src/utils/mockData.ts +98 -0
  135. flock/frontend/src/utils/performance.ts +16 -0
  136. flock/frontend/src/vite-env.d.ts +17 -0
  137. flock/frontend/tsconfig.json +27 -0
  138. flock/frontend/tsconfig.node.json +11 -0
  139. flock/frontend/vite.config.ts +25 -0
  140. flock/frontend/vitest.config.ts +11 -0
  141. flock/{core/util → helper}/cli_helper.py +9 -5
  142. flock/{core/logging → logging}/__init__.py +2 -3
  143. flock/logging/auto_trace.py +159 -0
  144. flock/{core/logging → logging}/formatters/enum_builder.py +3 -4
  145. flock/{core/logging → logging}/formatters/theme_builder.py +19 -44
  146. flock/{core/logging → logging}/formatters/themed_formatter.py +69 -107
  147. flock/{core/logging → logging}/logging.py +78 -61
  148. flock/{core/logging → logging}/telemetry.py +66 -26
  149. flock/{core/logging → logging}/telemetry_exporter/base_exporter.py +2 -2
  150. flock/logging/telemetry_exporter/duckdb_exporter.py +216 -0
  151. flock/{core/logging → logging}/telemetry_exporter/file_exporter.py +13 -10
  152. flock/{core/logging → logging}/telemetry_exporter/sqlite_exporter.py +2 -3
  153. flock/logging/trace_and_logged.py +304 -0
  154. flock/mcp/__init__.py +91 -0
  155. flock/{core/mcp/mcp_client.py → mcp/client.py} +131 -158
  156. flock/{core/mcp/mcp_config.py → mcp/config.py} +86 -132
  157. flock/mcp/manager.py +286 -0
  158. flock/mcp/servers/sse/__init__.py +1 -1
  159. flock/mcp/servers/sse/flock_sse_server.py +16 -58
  160. flock/mcp/servers/stdio/__init__.py +1 -1
  161. flock/mcp/servers/stdio/flock_stdio_server.py +13 -53
  162. flock/mcp/servers/streamable_http/flock_streamable_http_server.py +22 -67
  163. flock/mcp/servers/websockets/flock_websocket_server.py +12 -45
  164. flock/{core/mcp/flock_mcp_tool_base.py → mcp/tool.py} +24 -78
  165. flock/mcp/types/__init__.py +42 -0
  166. flock/{core/mcp → mcp}/types/callbacks.py +12 -15
  167. flock/{core/mcp → mcp}/types/factories.py +7 -6
  168. flock/{core/mcp → mcp}/types/handlers.py +13 -18
  169. flock/{core/mcp → mcp}/types/types.py +70 -74
  170. flock/{core/mcp → mcp}/util/helpers.py +3 -3
  171. flock/orchestrator.py +970 -0
  172. flock/registry.py +148 -0
  173. flock/runtime.py +262 -0
  174. flock/service.py +277 -0
  175. flock/store.py +1214 -0
  176. flock/subscription.py +111 -0
  177. flock/themes/andromeda.toml +1 -1
  178. flock/themes/apple-system-colors.toml +1 -1
  179. flock/themes/arcoiris.toml +1 -1
  180. flock/themes/atomonelight.toml +1 -1
  181. flock/themes/ayu copy.toml +1 -1
  182. flock/themes/ayu-light.toml +1 -1
  183. flock/themes/belafonte-day.toml +1 -1
  184. flock/themes/belafonte-night.toml +1 -1
  185. flock/themes/blulocodark.toml +1 -1
  186. flock/themes/breeze.toml +1 -1
  187. flock/themes/broadcast.toml +1 -1
  188. flock/themes/brogrammer.toml +1 -1
  189. flock/themes/builtin-dark.toml +1 -1
  190. flock/themes/builtin-pastel-dark.toml +1 -1
  191. flock/themes/catppuccin-latte.toml +1 -1
  192. flock/themes/catppuccin-macchiato.toml +1 -1
  193. flock/themes/catppuccin-mocha.toml +1 -1
  194. flock/themes/cga.toml +1 -1
  195. flock/themes/chalk.toml +1 -1
  196. flock/themes/ciapre.toml +1 -1
  197. flock/themes/coffee-theme.toml +1 -1
  198. flock/themes/cyberpunkscarletprotocol.toml +1 -1
  199. flock/themes/dark+.toml +1 -1
  200. flock/themes/darkermatrix.toml +1 -1
  201. flock/themes/darkmatrix.toml +2 -2
  202. flock/themes/darkside.toml +1 -1
  203. flock/themes/deep.toml +2 -2
  204. flock/themes/desert.toml +1 -1
  205. flock/themes/django.toml +1 -1
  206. flock/themes/djangosmooth.toml +1 -1
  207. flock/themes/doomone.toml +1 -1
  208. flock/themes/dotgov.toml +1 -1
  209. flock/themes/dracula+.toml +1 -1
  210. flock/themes/duckbones.toml +1 -1
  211. flock/themes/encom.toml +1 -1
  212. flock/themes/espresso.toml +1 -1
  213. flock/themes/everblush.toml +1 -1
  214. flock/themes/fairyfloss.toml +1 -1
  215. flock/themes/fideloper.toml +1 -1
  216. flock/themes/fishtank.toml +1 -1
  217. flock/themes/flexoki-light.toml +1 -1
  218. flock/themes/floraverse.toml +1 -1
  219. flock/themes/framer.toml +1 -1
  220. flock/themes/galizur.toml +1 -1
  221. flock/themes/github.toml +1 -1
  222. flock/themes/grass.toml +1 -1
  223. flock/themes/grey-green.toml +1 -1
  224. flock/themes/gruvboxlight.toml +1 -1
  225. flock/themes/guezwhoz.toml +1 -1
  226. flock/themes/harper.toml +1 -1
  227. flock/themes/hax0r-blue.toml +1 -1
  228. flock/themes/hopscotch.256.toml +1 -1
  229. flock/themes/ic-green-ppl.toml +1 -1
  230. flock/themes/iceberg-dark.toml +1 -1
  231. flock/themes/japanesque.toml +1 -1
  232. flock/themes/jubi.toml +1 -1
  233. flock/themes/kibble.toml +1 -1
  234. flock/themes/kolorit.toml +1 -1
  235. flock/themes/kurokula.toml +1 -1
  236. flock/themes/materialdesigncolors.toml +1 -1
  237. flock/themes/matrix.toml +1 -1
  238. flock/themes/mellifluous.toml +1 -1
  239. flock/themes/midnight-in-mojave.toml +1 -1
  240. flock/themes/monokai-remastered.toml +1 -1
  241. flock/themes/monokai-soda.toml +1 -1
  242. flock/themes/neon.toml +1 -1
  243. flock/themes/neopolitan.toml +5 -5
  244. flock/themes/nord-light.toml +1 -1
  245. flock/themes/ocean.toml +1 -1
  246. flock/themes/onehalfdark.toml +1 -1
  247. flock/themes/onehalflight.toml +1 -1
  248. flock/themes/palenighthc.toml +1 -1
  249. flock/themes/paulmillr.toml +1 -1
  250. flock/themes/pencildark.toml +1 -1
  251. flock/themes/pnevma.toml +1 -1
  252. flock/themes/purple-rain.toml +1 -1
  253. flock/themes/purplepeter.toml +1 -1
  254. flock/themes/raycast-dark.toml +1 -1
  255. flock/themes/red-sands.toml +1 -1
  256. flock/themes/relaxed.toml +1 -1
  257. flock/themes/retro.toml +1 -1
  258. flock/themes/rose-pine.toml +1 -1
  259. flock/themes/royal.toml +1 -1
  260. flock/themes/ryuuko.toml +1 -1
  261. flock/themes/sakura.toml +1 -1
  262. flock/themes/scarlet-protocol.toml +1 -1
  263. flock/themes/seoulbones-dark.toml +1 -1
  264. flock/themes/shades-of-purple.toml +1 -1
  265. flock/themes/smyck.toml +1 -1
  266. flock/themes/softserver.toml +1 -1
  267. flock/themes/solarized-darcula.toml +1 -1
  268. flock/themes/square.toml +1 -1
  269. flock/themes/sugarplum.toml +1 -1
  270. flock/themes/thayer-bright.toml +1 -1
  271. flock/themes/tokyonight.toml +1 -1
  272. flock/themes/tomorrow.toml +1 -1
  273. flock/themes/ubuntu.toml +1 -1
  274. flock/themes/ultradark.toml +1 -1
  275. flock/themes/ultraviolent.toml +1 -1
  276. flock/themes/unikitty.toml +1 -1
  277. flock/themes/urple.toml +1 -1
  278. flock/themes/vesper.toml +1 -1
  279. flock/themes/vimbones.toml +1 -1
  280. flock/themes/wildcherry.toml +1 -1
  281. flock/themes/wilmersdorf.toml +1 -1
  282. flock/themes/wryan.toml +1 -1
  283. flock/themes/xcodedarkhc.toml +1 -1
  284. flock/themes/xcodelight.toml +1 -1
  285. flock/themes/zenbones-light.toml +1 -1
  286. flock/themes/zenwritten-dark.toml +1 -1
  287. flock/utilities.py +301 -0
  288. flock/utility/output_utility_component.py +226 -0
  289. flock/visibility.py +107 -0
  290. flock_core-0.5.0.dist-info/METADATA +964 -0
  291. flock_core-0.5.0.dist-info/RECORD +525 -0
  292. flock_core-0.5.0.dist-info/entry_points.txt +2 -0
  293. {flock_core-0.4.543.dist-info → flock_core-0.5.0.dist-info}/licenses/LICENSE +1 -1
  294. flock/adapter/__init__.py +0 -14
  295. flock/adapter/azure_adapter.py +0 -68
  296. flock/adapter/chroma_adapter.py +0 -73
  297. flock/adapter/faiss_adapter.py +0 -97
  298. flock/adapter/pinecone_adapter.py +0 -51
  299. flock/adapter/vector_base.py +0 -47
  300. flock/cli/assets/release_notes.md +0 -140
  301. flock/cli/config.py +0 -8
  302. flock/cli/constants.py +0 -36
  303. flock/cli/create_agent.py +0 -1
  304. flock/cli/create_flock.py +0 -280
  305. flock/cli/execute_flock.py +0 -620
  306. flock/cli/load_agent.py +0 -1
  307. flock/cli/load_examples.py +0 -1
  308. flock/cli/load_flock.py +0 -192
  309. flock/cli/load_release_notes.py +0 -20
  310. flock/cli/loaded_flock_cli.py +0 -254
  311. flock/cli/manage_agents.py +0 -459
  312. flock/cli/registry_management.py +0 -889
  313. flock/cli/runner.py +0 -41
  314. flock/cli/settings.py +0 -857
  315. flock/cli/utils.py +0 -135
  316. flock/cli/view_results.py +0 -29
  317. flock/cli/yaml_editor.py +0 -396
  318. flock/config.py +0 -56
  319. flock/core/__init__.py +0 -44
  320. flock/core/api/__init__.py +0 -10
  321. flock/core/api/custom_endpoint.py +0 -45
  322. flock/core/api/endpoints.py +0 -262
  323. flock/core/api/main.py +0 -162
  324. flock/core/api/models.py +0 -101
  325. flock/core/api/run_store.py +0 -224
  326. flock/core/api/runner.py +0 -44
  327. flock/core/api/service.py +0 -214
  328. flock/core/config/flock_agent_config.py +0 -11
  329. flock/core/config/scheduled_agent_config.py +0 -40
  330. flock/core/context/context.py +0 -214
  331. flock/core/context/context_manager.py +0 -40
  332. flock/core/context/context_vars.py +0 -11
  333. flock/core/evaluation/utils.py +0 -395
  334. flock/core/execution/batch_executor.py +0 -369
  335. flock/core/execution/evaluation_executor.py +0 -438
  336. flock/core/execution/local_executor.py +0 -31
  337. flock/core/execution/opik_executor.py +0 -103
  338. flock/core/execution/temporal_executor.py +0 -166
  339. flock/core/flock.py +0 -1003
  340. flock/core/flock_agent.py +0 -1258
  341. flock/core/flock_evaluator.py +0 -60
  342. flock/core/flock_factory.py +0 -513
  343. flock/core/flock_module.py +0 -207
  344. flock/core/flock_registry.py +0 -702
  345. flock/core/flock_router.py +0 -83
  346. flock/core/flock_scheduler.py +0 -166
  347. flock/core/flock_server_manager.py +0 -136
  348. flock/core/interpreter/python_interpreter.py +0 -689
  349. flock/core/logging/live_capture.py +0 -137
  350. flock/core/logging/trace_and_logged.py +0 -59
  351. flock/core/mcp/__init__.py +0 -1
  352. flock/core/mcp/flock_mcp_server.py +0 -640
  353. flock/core/mcp/mcp_client_manager.py +0 -201
  354. flock/core/mcp/types/__init__.py +0 -1
  355. flock/core/mixin/dspy_integration.py +0 -445
  356. flock/core/mixin/prompt_parser.py +0 -125
  357. flock/core/serialization/__init__.py +0 -13
  358. flock/core/serialization/callable_registry.py +0 -52
  359. flock/core/serialization/flock_serializer.py +0 -854
  360. flock/core/serialization/json_encoder.py +0 -41
  361. flock/core/serialization/secure_serializer.py +0 -175
  362. flock/core/serialization/serializable.py +0 -342
  363. flock/core/serialization/serialization_utils.py +0 -409
  364. flock/core/util/file_path_utils.py +0 -223
  365. flock/core/util/hydrator.py +0 -309
  366. flock/core/util/input_resolver.py +0 -141
  367. flock/core/util/loader.py +0 -59
  368. flock/core/util/splitter.py +0 -219
  369. flock/di.py +0 -41
  370. flock/evaluators/__init__.py +0 -1
  371. flock/evaluators/declarative/__init__.py +0 -1
  372. flock/evaluators/declarative/declarative_evaluator.py +0 -217
  373. flock/evaluators/memory/memory_evaluator.py +0 -90
  374. flock/evaluators/test/test_case_evaluator.py +0 -38
  375. flock/evaluators/zep/zep_evaluator.py +0 -59
  376. flock/modules/__init__.py +0 -1
  377. flock/modules/assertion/__init__.py +0 -1
  378. flock/modules/assertion/assertion_module.py +0 -286
  379. flock/modules/callback/__init__.py +0 -1
  380. flock/modules/callback/callback_module.py +0 -91
  381. flock/modules/enterprise_memory/README.md +0 -99
  382. flock/modules/enterprise_memory/enterprise_memory_module.py +0 -526
  383. flock/modules/mem0/__init__.py +0 -1
  384. flock/modules/mem0/mem0_module.py +0 -126
  385. flock/modules/mem0_async/__init__.py +0 -1
  386. flock/modules/mem0_async/async_mem0_module.py +0 -126
  387. flock/modules/memory/__init__.py +0 -1
  388. flock/modules/memory/memory_module.py +0 -429
  389. flock/modules/memory/memory_parser.py +0 -125
  390. flock/modules/memory/memory_storage.py +0 -736
  391. flock/modules/output/__init__.py +0 -1
  392. flock/modules/output/output_module.py +0 -196
  393. flock/modules/performance/__init__.py +0 -1
  394. flock/modules/performance/metrics_module.py +0 -678
  395. flock/modules/zep/__init__.py +0 -1
  396. flock/modules/zep/zep_module.py +0 -192
  397. flock/platform/docker_tools.py +0 -49
  398. flock/platform/jaeger_install.py +0 -86
  399. flock/routers/__init__.py +0 -1
  400. flock/routers/agent/__init__.py +0 -1
  401. flock/routers/agent/agent_router.py +0 -236
  402. flock/routers/agent/handoff_agent.py +0 -58
  403. flock/routers/conditional/conditional_router.py +0 -486
  404. flock/routers/default/__init__.py +0 -1
  405. flock/routers/default/default_router.py +0 -80
  406. flock/routers/feedback/feedback_router.py +0 -114
  407. flock/routers/list_generator/list_generator_router.py +0 -166
  408. flock/routers/llm/__init__.py +0 -1
  409. flock/routers/llm/llm_router.py +0 -365
  410. flock/tools/__init__.py +0 -0
  411. flock/tools/azure_tools.py +0 -781
  412. flock/tools/code_tools.py +0 -167
  413. flock/tools/file_tools.py +0 -149
  414. flock/tools/github_tools.py +0 -157
  415. flock/tools/markdown_tools.py +0 -205
  416. flock/tools/system_tools.py +0 -9
  417. flock/tools/text_tools.py +0 -810
  418. flock/tools/web_tools.py +0 -92
  419. flock/tools/zendesk_tools.py +0 -501
  420. flock/webapp/__init__.py +0 -1
  421. flock/webapp/app/__init__.py +0 -0
  422. flock/webapp/app/api/__init__.py +0 -0
  423. flock/webapp/app/api/agent_management.py +0 -237
  424. flock/webapp/app/api/execution.py +0 -503
  425. flock/webapp/app/api/flock_management.py +0 -125
  426. flock/webapp/app/api/registry_viewer.py +0 -29
  427. flock/webapp/app/chat.py +0 -662
  428. flock/webapp/app/config.py +0 -104
  429. flock/webapp/app/dependencies.py +0 -117
  430. flock/webapp/app/main.py +0 -1086
  431. flock/webapp/app/middleware.py +0 -113
  432. flock/webapp/app/models_ui.py +0 -7
  433. flock/webapp/app/services/__init__.py +0 -0
  434. flock/webapp/app/services/feedback_file_service.py +0 -363
  435. flock/webapp/app/services/flock_service.py +0 -345
  436. flock/webapp/app/services/sharing_models.py +0 -81
  437. flock/webapp/app/services/sharing_store.py +0 -597
  438. flock/webapp/app/templates/theme_mapper.html +0 -326
  439. flock/webapp/app/theme_mapper.py +0 -811
  440. flock/webapp/app/utils.py +0 -85
  441. flock/webapp/run.py +0 -219
  442. flock/webapp/static/css/chat.css +0 -301
  443. flock/webapp/static/css/components.css +0 -167
  444. flock/webapp/static/css/header.css +0 -39
  445. flock/webapp/static/css/layout.css +0 -281
  446. flock/webapp/static/css/sidebar.css +0 -127
  447. flock/webapp/static/css/two-pane.css +0 -48
  448. flock/webapp/templates/base.html +0 -389
  449. flock/webapp/templates/chat.html +0 -152
  450. flock/webapp/templates/chat_settings.html +0 -19
  451. flock/webapp/templates/flock_editor.html +0 -16
  452. flock/webapp/templates/index.html +0 -12
  453. flock/webapp/templates/partials/_agent_detail_form.html +0 -93
  454. flock/webapp/templates/partials/_agent_list.html +0 -18
  455. flock/webapp/templates/partials/_agent_manager_view.html +0 -51
  456. flock/webapp/templates/partials/_agent_tools_checklist.html +0 -14
  457. flock/webapp/templates/partials/_chat_container.html +0 -15
  458. flock/webapp/templates/partials/_chat_messages.html +0 -57
  459. flock/webapp/templates/partials/_chat_settings_form.html +0 -85
  460. flock/webapp/templates/partials/_create_flock_form.html +0 -50
  461. flock/webapp/templates/partials/_dashboard_flock_detail.html +0 -17
  462. flock/webapp/templates/partials/_dashboard_flock_file_list.html +0 -16
  463. flock/webapp/templates/partials/_dashboard_flock_properties_preview.html +0 -28
  464. flock/webapp/templates/partials/_dashboard_upload_flock_form.html +0 -16
  465. flock/webapp/templates/partials/_dynamic_input_form_content.html +0 -22
  466. flock/webapp/templates/partials/_env_vars_table.html +0 -23
  467. flock/webapp/templates/partials/_execution_form.html +0 -127
  468. flock/webapp/templates/partials/_execution_view_container.html +0 -28
  469. flock/webapp/templates/partials/_flock_file_list.html +0 -23
  470. flock/webapp/templates/partials/_flock_properties_form.html +0 -52
  471. flock/webapp/templates/partials/_flock_upload_form.html +0 -16
  472. flock/webapp/templates/partials/_header_flock_status.html +0 -5
  473. flock/webapp/templates/partials/_live_logs.html +0 -13
  474. flock/webapp/templates/partials/_load_manager_view.html +0 -49
  475. flock/webapp/templates/partials/_registry_table.html +0 -25
  476. flock/webapp/templates/partials/_registry_viewer_content.html +0 -70
  477. flock/webapp/templates/partials/_results_display.html +0 -78
  478. flock/webapp/templates/partials/_settings_env_content.html +0 -9
  479. flock/webapp/templates/partials/_settings_theme_content.html +0 -14
  480. flock/webapp/templates/partials/_settings_view.html +0 -36
  481. flock/webapp/templates/partials/_share_chat_link_snippet.html +0 -11
  482. flock/webapp/templates/partials/_share_link_snippet.html +0 -35
  483. flock/webapp/templates/partials/_sidebar.html +0 -74
  484. flock/webapp/templates/partials/_structured_data_view.html +0 -40
  485. flock/webapp/templates/partials/_theme_preview.html +0 -36
  486. flock/webapp/templates/registry_viewer.html +0 -84
  487. flock/webapp/templates/shared_run_page.html +0 -140
  488. flock/workflow/__init__.py +0 -0
  489. flock/workflow/activities.py +0 -237
  490. flock/workflow/agent_activities.py +0 -24
  491. flock/workflow/agent_execution_activity.py +0 -240
  492. flock/workflow/flock_workflow.py +0 -225
  493. flock/workflow/temporal_config.py +0 -96
  494. flock/workflow/temporal_setup.py +0 -60
  495. flock_core-0.4.543.dist-info/METADATA +0 -676
  496. flock_core-0.4.543.dist-info/RECORD +0 -572
  497. flock_core-0.4.543.dist-info/entry_points.txt +0 -2
  498. /flock/{core/logging → logging}/formatters/themes.py +0 -0
  499. /flock/{core/logging → logging}/span_middleware/baggage_span_processor.py +0 -0
  500. /flock/{core/mcp → mcp}/util/__init__.py +0 -0
  501. {flock_core-0.4.543.dist-info → flock_core-0.5.0.dist-info}/WHEEL +0 -0
@@ -0,0 +1,543 @@
1
+ import { describe, it, expect, vi, beforeEach } from 'vitest';
2
+ import { render, screen, fireEvent, waitFor } from '@testing-library/react';
3
+ import PublishControl from './PublishControl';
4
+ import { useFilterStore } from '../../store/filterStore';
5
+
6
+ // Mock fetch globally
7
+ const mockFetch = vi.fn();
8
+ globalThis.fetch = mockFetch;
9
+
10
+ describe('PublishControl', () => {
11
+ const mockArtifactTypes = [
12
+ { name: 'Idea', schema: { type: 'object', properties: { content: { type: 'string' } } } },
13
+ { name: 'Movie', schema: { type: 'object', properties: { title: { type: 'string' }, plot: { type: 'string' } } } },
14
+ { name: 'Tagline', schema: { type: 'object', properties: { text: { type: 'string' } } } },
15
+ ];
16
+
17
+ beforeEach(() => {
18
+ vi.clearAllMocks();
19
+ mockFetch.mockReset();
20
+ });
21
+
22
+ it('should render publish form with artifact type dropdown and content textarea', async () => {
23
+ mockFetch.mockResolvedValueOnce({
24
+ ok: true,
25
+ json: async () => ({ artifact_types: mockArtifactTypes }),
26
+ });
27
+
28
+ render(<PublishControl />);
29
+
30
+ await waitFor(() => {
31
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
32
+ });
33
+
34
+ // Select artifact type to trigger dynamic field rendering
35
+ const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
36
+ fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
37
+
38
+ // Now the content field should appear
39
+ expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
40
+ expect(screen.getByRole('button', { name: /publish artifact/i })).toBeInTheDocument();
41
+ });
42
+
43
+ it('should populate artifact type dropdown from available types', async () => {
44
+ mockFetch.mockResolvedValueOnce({
45
+ ok: true,
46
+ json: async () => ({ artifact_types: mockArtifactTypes }),
47
+ });
48
+
49
+ render(<PublishControl />);
50
+
51
+ await waitFor(() => {
52
+ const dropdown = screen.getByLabelText(/artifact type/i) as HTMLSelectElement;
53
+ expect(dropdown.options.length).toBeGreaterThan(0);
54
+ });
55
+
56
+ const dropdown = screen.getByLabelText(/artifact type/i) as HTMLSelectElement;
57
+ const optionValues = Array.from(dropdown.options).map(opt => opt.value);
58
+
59
+ expect(optionValues).toContain('Idea');
60
+ expect(optionValues).toContain('Movie');
61
+ expect(optionValues).toContain('Tagline');
62
+ });
63
+
64
+ it('should validate required fields before submission', async () => {
65
+ mockFetch
66
+ .mockResolvedValueOnce({
67
+ ok: true,
68
+ json: async () => ({ artifact_types: mockArtifactTypes }),
69
+ })
70
+ .mockResolvedValueOnce({
71
+ ok: true,
72
+ json: async () => ({
73
+ status: 'success',
74
+ correlation_id: 'should-not-be-called',
75
+ }),
76
+ });
77
+
78
+ render(<PublishControl />);
79
+
80
+ await waitFor(() => {
81
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
82
+ });
83
+
84
+ const submitButton = screen.getByRole('button', { name: /publish artifact/i });
85
+
86
+ // Test 1: Try to submit without selecting type - should NOT call API
87
+ fireEvent.click(submitButton);
88
+
89
+ // Give it a moment, then verify API was NOT called (only 1 call for artifact types)
90
+ await new Promise(resolve => setTimeout(resolve, 100));
91
+ expect(mockFetch).toHaveBeenCalledTimes(1); // Only the initial artifact types fetch
92
+
93
+ // Test 2: Select artifact type but leave content empty - should NOT call API
94
+ const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
95
+ fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
96
+
97
+ await waitFor(() => {
98
+ expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
99
+ });
100
+
101
+ // Try to submit with empty content
102
+ fireEvent.click(submitButton);
103
+
104
+ // Give it a moment, then verify API was still NOT called
105
+ await new Promise(resolve => setTimeout(resolve, 100));
106
+ expect(mockFetch).toHaveBeenCalledTimes(1); // Still only 1 call
107
+ });
108
+
109
+ it('should validate content as valid JSON', async () => {
110
+ mockFetch.mockResolvedValueOnce({
111
+ ok: true,
112
+ json: async () => ({ artifact_types: mockArtifactTypes }),
113
+ });
114
+
115
+ render(<PublishControl />);
116
+
117
+ await waitFor(() => {
118
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
119
+ });
120
+
121
+ const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
122
+ fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
123
+
124
+ // Wait for content field to appear after selecting type
125
+ await waitFor(() => {
126
+ expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
127
+ });
128
+
129
+ const contentInput = screen.getByLabelText(/content/i);
130
+ const submitButton = screen.getByRole('button', { name: /publish artifact/i });
131
+
132
+ // Content field is now a text input, not textarea, and accepts any string
133
+ // The validation is done on the backend, so this test is no longer relevant
134
+ // for JSON validation. Skip JSON validation test.
135
+ fireEvent.change(contentInput, { target: { value: 'Test content' } });
136
+ fireEvent.click(submitButton);
137
+
138
+ // Should not show JSON validation error for simple text
139
+ await waitFor(() => {
140
+ expect(mockFetch).toHaveBeenCalled();
141
+ });
142
+ });
143
+
144
+ it('should successfully publish and call API endpoint', async () => {
145
+ mockFetch
146
+ .mockResolvedValueOnce({
147
+ ok: true,
148
+ json: async () => ({ artifact_types: mockArtifactTypes }),
149
+ })
150
+ .mockResolvedValueOnce({
151
+ ok: true,
152
+ json: async () => ({
153
+ status: 'success',
154
+ correlation_id: 'test-correlation-123',
155
+ message: 'Artifact published successfully'
156
+ }),
157
+ });
158
+
159
+ render(<PublishControl />);
160
+
161
+ await waitFor(() => {
162
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
163
+ });
164
+
165
+ const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
166
+ fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
167
+
168
+ await waitFor(() => {
169
+ expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
170
+ });
171
+
172
+ const contentInput = screen.getByLabelText(/content/i);
173
+ const submitButton = screen.getByRole('button', { name: /publish artifact/i });
174
+
175
+ fireEvent.change(contentInput, { target: { value: 'Test idea' } });
176
+ fireEvent.click(submitButton);
177
+
178
+ await waitFor(() => {
179
+ expect(mockFetch).toHaveBeenCalledWith(
180
+ '/api/control/publish',
181
+ expect.objectContaining({
182
+ method: 'POST',
183
+ headers: { 'Content-Type': 'application/json' },
184
+ body: JSON.stringify({
185
+ artifact_type: 'Idea',
186
+ content: { content: 'Test idea' }, // Schema-based: { content: string }
187
+ }),
188
+ })
189
+ );
190
+ });
191
+ });
192
+
193
+ it('should display success message with correlation ID after successful publish', async () => {
194
+ mockFetch
195
+ .mockResolvedValueOnce({
196
+ ok: true,
197
+ json: async () => ({ artifact_types: mockArtifactTypes }),
198
+ })
199
+ .mockResolvedValueOnce({
200
+ ok: true,
201
+ json: async () => ({
202
+ status: 'success',
203
+ correlation_id: 'test-correlation-456',
204
+ message: 'Artifact published successfully'
205
+ }),
206
+ });
207
+
208
+ render(<PublishControl />);
209
+
210
+ await waitFor(() => {
211
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
212
+ });
213
+
214
+ const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
215
+ fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
216
+
217
+ await waitFor(() => {
218
+ expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
219
+ });
220
+
221
+ const contentInput = screen.getByLabelText(/content/i);
222
+ const submitButton = screen.getByRole('button', { name: /publish artifact/i });
223
+
224
+ fireEvent.change(contentInput, { target: { value: 'Success test' } });
225
+ fireEvent.click(submitButton);
226
+
227
+ await waitFor(() => {
228
+ expect(screen.getByText(/successfully published/i)).toBeInTheDocument();
229
+ expect(screen.getByText(/test-correlation-456/)).toBeInTheDocument();
230
+ });
231
+ });
232
+
233
+ it('should display error message on API failure', async () => {
234
+ mockFetch
235
+ .mockResolvedValueOnce({
236
+ ok: true,
237
+ json: async () => ({ artifact_types: mockArtifactTypes }),
238
+ })
239
+ .mockResolvedValueOnce({
240
+ ok: false,
241
+ status: 500,
242
+ json: async () => ({
243
+ error: 'Internal server error',
244
+ message: 'Failed to publish artifact'
245
+ }),
246
+ });
247
+
248
+ render(<PublishControl />);
249
+
250
+ await waitFor(() => {
251
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
252
+ });
253
+
254
+ const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
255
+ fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
256
+
257
+ await waitFor(() => {
258
+ expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
259
+ });
260
+
261
+ const contentInput = screen.getByLabelText(/content/i);
262
+ const submitButton = screen.getByRole('button', { name: /publish artifact/i });
263
+
264
+ fireEvent.change(contentInput, { target: { value: 'Error test' } });
265
+ fireEvent.click(submitButton);
266
+
267
+ await waitFor(() => {
268
+ expect(screen.getByText(/failed to publish/i)).toBeInTheDocument();
269
+ });
270
+ });
271
+
272
+ it('should reset form after successful publish', async () => {
273
+ mockFetch
274
+ .mockResolvedValueOnce({
275
+ ok: true,
276
+ json: async () => ({ artifact_types: mockArtifactTypes }),
277
+ })
278
+ .mockResolvedValueOnce({
279
+ ok: true,
280
+ json: async () => ({
281
+ status: 'success',
282
+ correlation_id: 'test-correlation-789',
283
+ message: 'Artifact published successfully'
284
+ }),
285
+ });
286
+
287
+ render(<PublishControl />);
288
+
289
+ await waitFor(() => {
290
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
291
+ });
292
+
293
+ const artifactTypeSelect = screen.getByLabelText(/artifact type/i) as HTMLSelectElement;
294
+ fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
295
+
296
+ await waitFor(() => {
297
+ expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
298
+ });
299
+
300
+ const contentInput = screen.getByLabelText(/content/i) as HTMLInputElement;
301
+ const submitButton = screen.getByRole('button', { name: /publish artifact/i });
302
+
303
+ fireEvent.change(contentInput, { target: { value: 'Reset test' } });
304
+ fireEvent.click(submitButton);
305
+
306
+ await waitFor(() => {
307
+ expect(screen.getByText(/successfully published/i)).toBeInTheDocument();
308
+ });
309
+
310
+ // Form should be reset - artifact type cleared means fields disappear
311
+ expect(artifactTypeSelect.value).toBe('');
312
+ });
313
+
314
+ it('should disable submit button during API call', async () => {
315
+ let resolvePublish: (value: any) => void;
316
+ const publishPromise = new Promise(resolve => {
317
+ resolvePublish = resolve;
318
+ });
319
+
320
+ mockFetch
321
+ .mockResolvedValueOnce({
322
+ ok: true,
323
+ json: async () => ({ artifact_types: mockArtifactTypes }),
324
+ })
325
+ .mockReturnValueOnce(publishPromise as any);
326
+
327
+ render(<PublishControl />);
328
+
329
+ await waitFor(() => {
330
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
331
+ });
332
+
333
+ const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
334
+ fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
335
+
336
+ await waitFor(() => {
337
+ expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
338
+ });
339
+
340
+ const contentInput = screen.getByLabelText(/content/i);
341
+ const submitButton = screen.getByRole('button', { name: /publish artifact/i });
342
+
343
+ fireEvent.change(contentInput, { target: { value: 'Loading test' } });
344
+ fireEvent.click(submitButton);
345
+
346
+ // Button should be disabled during API call
347
+ await waitFor(() => {
348
+ expect(submitButton).toBeDisabled();
349
+ expect(screen.getByText(/publishing/i)).toBeInTheDocument();
350
+ });
351
+
352
+ // Resolve the promise
353
+ resolvePublish!({
354
+ ok: true,
355
+ json: async () => ({
356
+ status: 'success',
357
+ correlation_id: 'test-correlation-999',
358
+ message: 'Artifact published successfully'
359
+ }),
360
+ });
361
+
362
+ // Button should be enabled after API call completes
363
+ await waitFor(() => {
364
+ expect(submitButton).not.toBeDisabled();
365
+ });
366
+ });
367
+
368
+ it('should handle network errors gracefully', async () => {
369
+ mockFetch
370
+ .mockResolvedValueOnce({
371
+ ok: true,
372
+ json: async () => ({ artifact_types: mockArtifactTypes }),
373
+ })
374
+ .mockRejectedValueOnce(new Error('Network error'));
375
+
376
+ render(<PublishControl />);
377
+
378
+ await waitFor(() => {
379
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
380
+ });
381
+
382
+ const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
383
+ fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
384
+
385
+ await waitFor(() => {
386
+ expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
387
+ });
388
+
389
+ const contentInput = screen.getByLabelText(/content/i);
390
+ const submitButton = screen.getByRole('button', { name: /publish artifact/i });
391
+
392
+ fireEvent.change(contentInput, { target: { value: 'Network error test' } });
393
+ fireEvent.click(submitButton);
394
+
395
+ await waitFor(() => {
396
+ expect(screen.getByText(/network error|failed to connect/i)).toBeInTheDocument();
397
+ });
398
+ });
399
+
400
+ // Auto-filter checkbox tests
401
+ it('should render auto-set filter checkbox checked by default', async () => {
402
+ mockFetch.mockResolvedValueOnce({
403
+ ok: true,
404
+ json: async () => ({ artifact_types: mockArtifactTypes }),
405
+ });
406
+
407
+ render(<PublishControl />);
408
+
409
+ await waitFor(() => {
410
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
411
+ });
412
+
413
+ const checkbox = screen.getByLabelText(/set filter to correlation id/i) as HTMLInputElement;
414
+ expect(checkbox).toBeInTheDocument();
415
+ expect(checkbox.checked).toBe(true);
416
+ });
417
+
418
+ it('should set filter to correlation ID when checkbox is checked and publish succeeds', async () => {
419
+ mockFetch
420
+ .mockResolvedValueOnce({
421
+ ok: true,
422
+ json: async () => ({ artifact_types: mockArtifactTypes }),
423
+ })
424
+ .mockResolvedValueOnce({
425
+ ok: true,
426
+ json: async () => ({
427
+ status: 'success',
428
+ correlation_id: 'auto-filter-123',
429
+ message: 'Artifact published successfully'
430
+ }),
431
+ });
432
+
433
+ render(<PublishControl />);
434
+
435
+ await waitFor(() => {
436
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
437
+ });
438
+
439
+ const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
440
+ const checkbox = screen.getByLabelText(/set filter to correlation id/i) as HTMLInputElement;
441
+
442
+ // Checkbox should be checked by default
443
+ expect(checkbox.checked).toBe(true);
444
+
445
+ fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
446
+
447
+ await waitFor(() => {
448
+ expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
449
+ });
450
+
451
+ const contentInput = screen.getByLabelText(/content/i);
452
+ const submitButton = screen.getByRole('button', { name: /publish artifact/i });
453
+
454
+ fireEvent.change(contentInput, { target: { value: 'Auto-filter test' } });
455
+ fireEvent.click(submitButton);
456
+
457
+ await waitFor(() => {
458
+ expect(screen.getByText(/filter set to: auto-filter-123/i)).toBeInTheDocument();
459
+ });
460
+
461
+ // Verify filter was set in store
462
+ const filterState = useFilterStore.getState();
463
+ expect(filterState.correlationId).toBe('auto-filter-123');
464
+ });
465
+
466
+ it('should NOT set filter when checkbox is unchecked', async () => {
467
+ mockFetch
468
+ .mockResolvedValueOnce({
469
+ ok: true,
470
+ json: async () => ({ artifact_types: mockArtifactTypes }),
471
+ })
472
+ .mockResolvedValueOnce({
473
+ ok: true,
474
+ json: async () => ({
475
+ status: 'success',
476
+ correlation_id: 'no-filter-456',
477
+ message: 'Artifact published successfully'
478
+ }),
479
+ });
480
+
481
+ render(<PublishControl />);
482
+
483
+ await waitFor(() => {
484
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
485
+ });
486
+
487
+ const artifactTypeSelect = screen.getByLabelText(/artifact type/i);
488
+ const checkbox = screen.getByLabelText(/set filter to correlation id/i) as HTMLInputElement;
489
+
490
+ // Uncheck the checkbox
491
+ fireEvent.click(checkbox);
492
+ expect(checkbox.checked).toBe(false);
493
+
494
+ // Clear any existing filter
495
+ useFilterStore.setState({ correlationId: undefined });
496
+
497
+ fireEvent.change(artifactTypeSelect, { target: { value: 'Idea' } });
498
+
499
+ await waitFor(() => {
500
+ expect(screen.getByLabelText(/content/i)).toBeInTheDocument();
501
+ });
502
+
503
+ const contentInput = screen.getByLabelText(/content/i);
504
+ const submitButton = screen.getByRole('button', { name: /publish artifact/i });
505
+
506
+ fireEvent.change(contentInput, { target: { value: 'No filter test' } });
507
+ fireEvent.click(submitButton);
508
+
509
+ await waitFor(() => {
510
+ expect(screen.getByText(/correlation id: no-filter-456/i)).toBeInTheDocument();
511
+ });
512
+
513
+ // Verify filter was NOT set in store
514
+ const filterState = useFilterStore.getState();
515
+ expect(filterState.correlationId).toBeUndefined();
516
+ });
517
+
518
+ it('should allow toggling checkbox state', async () => {
519
+ mockFetch.mockResolvedValueOnce({
520
+ ok: true,
521
+ json: async () => ({ artifact_types: mockArtifactTypes }),
522
+ });
523
+
524
+ render(<PublishControl />);
525
+
526
+ await waitFor(() => {
527
+ expect(screen.getByLabelText(/artifact type/i)).toBeInTheDocument();
528
+ });
529
+
530
+ const checkbox = screen.getByLabelText(/set filter to correlation id/i) as HTMLInputElement;
531
+
532
+ // Initially checked
533
+ expect(checkbox.checked).toBe(true);
534
+
535
+ // Uncheck
536
+ fireEvent.click(checkbox);
537
+ expect(checkbox.checked).toBe(false);
538
+
539
+ // Check again
540
+ fireEvent.click(checkbox);
541
+ expect(checkbox.checked).toBe(true);
542
+ });
543
+ });