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,102 @@
1
+ .container {
2
+ position: relative;
3
+ width: 300px;
4
+ }
5
+
6
+ .inputWrapper {
7
+ position: relative;
8
+ }
9
+
10
+ .input {
11
+ width: 100%;
12
+ padding: var(--spacing-2) var(--spacing-8) var(--spacing-2) var(--spacing-3);
13
+ background-color: var(--color-bg-surface);
14
+ border: 1px solid var(--color-border-default);
15
+ border-radius: var(--radius-md);
16
+ font-size: var(--font-size-body-sm);
17
+ font-family: var(--font-family-sans);
18
+ color: var(--color-text-primary);
19
+ outline: none;
20
+ box-sizing: border-box;
21
+ transition: var(--transition-colors), box-shadow var(--duration-normal) var(--ease-smooth);
22
+ }
23
+
24
+ .input::placeholder {
25
+ color: var(--color-text-muted);
26
+ font-style: italic;
27
+ }
28
+
29
+ .input:focus {
30
+ border-color: var(--color-border-focus);
31
+ box-shadow: var(--shadow-glow-primary);
32
+ }
33
+
34
+ .clearButton {
35
+ position: absolute;
36
+ right: var(--spacing-2);
37
+ top: 50%;
38
+ transform: translateY(-50%);
39
+ background: none;
40
+ border: none;
41
+ cursor: pointer;
42
+ font-size: 18px;
43
+ color: var(--color-text-muted);
44
+ padding: 0 var(--spacing-1);
45
+ line-height: 1;
46
+ transition: var(--transition-colors);
47
+ }
48
+
49
+ .clearButton:hover {
50
+ color: var(--color-text-secondary);
51
+ }
52
+
53
+ .dropdown {
54
+ position: absolute;
55
+ top: 100%;
56
+ left: 0;
57
+ right: 0;
58
+ margin-top: var(--spacing-1);
59
+ background-color: var(--color-bg-surface);
60
+ border: 1px solid var(--color-border-default);
61
+ border-radius: var(--radius-md);
62
+ box-shadow: var(--shadow-lg);
63
+ max-height: 300px;
64
+ overflow-y: auto;
65
+ z-index: 1000;
66
+ }
67
+
68
+ .dropdownEmpty {
69
+ padding: var(--spacing-3);
70
+ color: var(--color-text-muted);
71
+ font-size: var(--font-size-body-sm);
72
+ text-align: center;
73
+ }
74
+
75
+ .dropdownItem {
76
+ padding: var(--spacing-2-5) var(--spacing-3);
77
+ cursor: pointer;
78
+ border-bottom: 1px solid var(--color-border-subtle);
79
+ transition: var(--transition-colors);
80
+ }
81
+
82
+ .dropdownItem:last-child {
83
+ border-bottom: none;
84
+ }
85
+
86
+ .dropdownItem:hover {
87
+ background-color: var(--color-bg-elevated);
88
+ }
89
+
90
+ .correlationId {
91
+ font-family: var(--font-family-mono);
92
+ font-size: var(--font-size-body-xs);
93
+ font-weight: var(--font-weight-medium);
94
+ color: var(--color-text-primary);
95
+ margin-bottom: var(--spacing-1);
96
+ word-break: break-all;
97
+ }
98
+
99
+ .metadata {
100
+ font-size: var(--font-size-caption);
101
+ color: var(--color-text-tertiary);
102
+ }
@@ -0,0 +1,197 @@
1
+ import { describe, it, expect, vi, beforeEach } from 'vitest';
2
+ import { render, screen, fireEvent } from '@testing-library/react';
3
+ import CorrelationIDFilter from './CorrelationIDFilter';
4
+ import { useFilterStore } from '../../store/filterStore';
5
+
6
+ vi.mock('../../store/filterStore');
7
+
8
+ describe('CorrelationIDFilter', () => {
9
+ const mockSetCorrelationId = vi.fn();
10
+ const mockAvailableIds = [
11
+ {
12
+ correlation_id: 'abc12345',
13
+ first_seen: Date.now() - 120000,
14
+ artifact_count: 5,
15
+ run_count: 2,
16
+ },
17
+ {
18
+ correlation_id: 'def67890',
19
+ first_seen: Date.now() - 60000,
20
+ artifact_count: 3,
21
+ run_count: 1,
22
+ },
23
+ {
24
+ correlation_id: 'ghi11111',
25
+ first_seen: Date.now() - 300000,
26
+ artifact_count: 10,
27
+ run_count: 3,
28
+ },
29
+ ];
30
+
31
+ beforeEach(() => {
32
+ vi.clearAllMocks();
33
+ vi.mocked(useFilterStore).mockImplementation((selector: any) => {
34
+ const state = {
35
+ correlationId: null,
36
+ availableCorrelationIds: mockAvailableIds,
37
+ setCorrelationId: mockSetCorrelationId,
38
+ };
39
+ return selector(state);
40
+ });
41
+ });
42
+
43
+ it('should render correlation ID filter input', () => {
44
+ render(<CorrelationIDFilter />);
45
+ expect(screen.getByPlaceholderText(/Search correlation ID/i)).toBeInTheDocument();
46
+ });
47
+
48
+ it('should show dropdown when input is focused', () => {
49
+ render(<CorrelationIDFilter />);
50
+ const input = screen.getByPlaceholderText(/Search correlation ID/i);
51
+
52
+ fireEvent.focus(input);
53
+
54
+ // Should show all available IDs
55
+ expect(screen.getByText(/abc12345/)).toBeInTheDocument();
56
+ expect(screen.getByText(/def67890/)).toBeInTheDocument();
57
+ expect(screen.getByText(/ghi11111/)).toBeInTheDocument();
58
+ });
59
+
60
+ it('should display metadata for each correlation ID', () => {
61
+ render(<CorrelationIDFilter />);
62
+ const input = screen.getByPlaceholderText(/Search correlation ID/i);
63
+
64
+ fireEvent.focus(input);
65
+
66
+ // Should show artifact count and time ago
67
+ expect(screen.getByText(/5 messages/)).toBeInTheDocument();
68
+ expect(screen.getByText(/3 messages/)).toBeInTheDocument();
69
+ expect(screen.getByText(/10 messages/)).toBeInTheDocument();
70
+ });
71
+
72
+ it('should filter dropdown items based on input', () => {
73
+ render(<CorrelationIDFilter />);
74
+ const input = screen.getByPlaceholderText(/Search correlation ID/i) as HTMLInputElement;
75
+
76
+ fireEvent.focus(input);
77
+ fireEvent.change(input, { target: { value: 'abc' } });
78
+
79
+ // Should only show matching ID
80
+ expect(screen.getByText(/abc12345/)).toBeInTheDocument();
81
+ expect(screen.queryByText(/def67890/)).not.toBeInTheDocument();
82
+ expect(screen.queryByText(/ghi11111/)).not.toBeInTheDocument();
83
+ });
84
+
85
+ it('should call setCorrelationId when an option is selected', () => {
86
+ render(<CorrelationIDFilter />);
87
+ const input = screen.getByPlaceholderText(/Search correlation ID/i);
88
+
89
+ fireEvent.focus(input);
90
+ const option = screen.getByText(/abc12345/);
91
+ fireEvent.click(option);
92
+
93
+ expect(mockSetCorrelationId).toHaveBeenCalledWith('abc12345');
94
+ });
95
+
96
+ it('should display selected correlation ID in input', () => {
97
+ vi.mocked(useFilterStore).mockImplementation((selector: any) => {
98
+ const state = {
99
+ correlationId: 'abc12345',
100
+ availableCorrelationIds: mockAvailableIds,
101
+ setCorrelationId: mockSetCorrelationId,
102
+ };
103
+ return selector(state);
104
+ });
105
+
106
+ render(<CorrelationIDFilter />);
107
+ const input = screen.getByPlaceholderText(/Search correlation ID/i) as HTMLInputElement;
108
+
109
+ expect(input.value).toBe('abc12345');
110
+ });
111
+
112
+ it('should clear selection when clear button is clicked', () => {
113
+ vi.mocked(useFilterStore).mockImplementation((selector: any) => {
114
+ const state = {
115
+ correlationId: 'abc12345',
116
+ availableCorrelationIds: mockAvailableIds,
117
+ setCorrelationId: mockSetCorrelationId,
118
+ };
119
+ return selector(state);
120
+ });
121
+
122
+ render(<CorrelationIDFilter />);
123
+ const clearButton = screen.getByRole('button', { name: /clear/i });
124
+
125
+ fireEvent.click(clearButton);
126
+
127
+ expect(mockSetCorrelationId).toHaveBeenCalledWith(null);
128
+ });
129
+
130
+ it('should hide dropdown when clicking outside', () => {
131
+ render(<CorrelationIDFilter />);
132
+ const input = screen.getByPlaceholderText(/Search correlation ID/i);
133
+
134
+ fireEvent.focus(input);
135
+ expect(screen.getByText(/abc12345/)).toBeInTheDocument();
136
+
137
+ // Simulate click outside
138
+ fireEvent.blur(input);
139
+
140
+ // Dropdown should be hidden (use setTimeout to wait for blur)
141
+ setTimeout(() => {
142
+ expect(screen.queryByText(/abc12345/)).not.toBeInTheDocument();
143
+ }, 100);
144
+ });
145
+
146
+ it('should show "No correlation IDs found" when list is empty', () => {
147
+ vi.mocked(useFilterStore).mockImplementation((selector: any) => {
148
+ const state = {
149
+ correlationId: null,
150
+ availableCorrelationIds: [],
151
+ setCorrelationId: mockSetCorrelationId,
152
+ };
153
+ return selector(state);
154
+ });
155
+
156
+ render(<CorrelationIDFilter />);
157
+ const input = screen.getByPlaceholderText(/Search correlation ID/i);
158
+
159
+ fireEvent.focus(input);
160
+
161
+ expect(screen.getByText(/No correlation IDs found/i)).toBeInTheDocument();
162
+ });
163
+
164
+ it('should format time ago correctly', () => {
165
+ const now = Date.now();
166
+ vi.mocked(useFilterStore).mockImplementation((selector: any) => {
167
+ const state = {
168
+ correlationId: null,
169
+ availableCorrelationIds: [
170
+ {
171
+ correlation_id: 'recent',
172
+ first_seen: now - 30000, // 30 seconds ago
173
+ artifact_count: 1,
174
+ run_count: 1,
175
+ },
176
+ {
177
+ correlation_id: 'old',
178
+ first_seen: now - 3600000, // 1 hour ago
179
+ artifact_count: 1,
180
+ run_count: 1,
181
+ },
182
+ ],
183
+ setCorrelationId: mockSetCorrelationId,
184
+ };
185
+ return selector(state);
186
+ });
187
+
188
+ render(<CorrelationIDFilter />);
189
+ const input = screen.getByPlaceholderText(/Search correlation ID/i);
190
+
191
+ fireEvent.focus(input);
192
+
193
+ // Should show relative time
194
+ expect(screen.getByText(/30s ago/i)).toBeInTheDocument();
195
+ expect(screen.getByText(/1h ago/i)).toBeInTheDocument();
196
+ });
197
+ });
@@ -0,0 +1,121 @@
1
+ import React, { useState, useRef, useEffect } from 'react';
2
+ import { useFilterStore } from '../../store/filterStore';
3
+ import styles from './CorrelationIDFilter.module.css';
4
+
5
+ const formatTimeAgo = (timestamp: number): string => {
6
+ const seconds = Math.floor((Date.now() - timestamp) / 1000);
7
+ if (seconds < 60) return `${seconds}s ago`;
8
+ const minutes = Math.floor(seconds / 60);
9
+ if (minutes < 60) return `${minutes}m ago`;
10
+ const hours = Math.floor(minutes / 60);
11
+ if (hours < 24) return `${hours}h ago`;
12
+ const days = Math.floor(hours / 24);
13
+ return `${days}d ago`;
14
+ };
15
+
16
+ const CorrelationIDFilter: React.FC = () => {
17
+ const correlationId = useFilterStore((state) => state.correlationId);
18
+ const availableCorrelationIds = useFilterStore((state) => state.availableCorrelationIds);
19
+ const setCorrelationId = useFilterStore((state) => state.setCorrelationId);
20
+
21
+ const [inputValue, setInputValue] = useState(correlationId || '');
22
+ const [isOpen, setIsOpen] = useState(false);
23
+ const containerRef = useRef<HTMLDivElement>(null);
24
+
25
+ useEffect(() => {
26
+ setInputValue(correlationId || '');
27
+ }, [correlationId]);
28
+
29
+ useEffect(() => {
30
+ const handleClickOutside = (event: MouseEvent) => {
31
+ if (containerRef.current && !containerRef.current.contains(event.target as Node)) {
32
+ setIsOpen(false);
33
+ }
34
+ };
35
+
36
+ document.addEventListener('mousedown', handleClickOutside);
37
+ return () => document.removeEventListener('mousedown', handleClickOutside);
38
+ }, []);
39
+
40
+ const filteredIds = availableCorrelationIds.filter((item) =>
41
+ item.correlation_id.toLowerCase().includes(inputValue.toLowerCase())
42
+ );
43
+
44
+ const handleSelect = (id: string) => {
45
+ setCorrelationId(id);
46
+ setInputValue(id);
47
+ setIsOpen(false);
48
+ };
49
+
50
+ const handleClear = () => {
51
+ setCorrelationId(null);
52
+ setInputValue('');
53
+ setIsOpen(false);
54
+ };
55
+
56
+ const handleInputChange = (e: React.ChangeEvent<HTMLInputElement>) => {
57
+ setInputValue(e.target.value);
58
+ setIsOpen(true);
59
+ };
60
+
61
+ const handleFocus = () => {
62
+ setIsOpen(true);
63
+ };
64
+
65
+ const handleBlur = () => {
66
+ // Delay to allow click on dropdown items
67
+ setTimeout(() => setIsOpen(false), 200);
68
+ };
69
+
70
+ return (
71
+ <div ref={containerRef} className={styles.container}>
72
+ <div className={styles.inputWrapper}>
73
+ <input
74
+ type="text"
75
+ value={inputValue}
76
+ onChange={handleInputChange}
77
+ onFocus={handleFocus}
78
+ onBlur={handleBlur}
79
+ placeholder="Search correlation ID..."
80
+ className={styles.input}
81
+ />
82
+ {correlationId && (
83
+ <button
84
+ onClick={handleClear}
85
+ aria-label="Clear"
86
+ className={styles.clearButton}
87
+ >
88
+ ×
89
+ </button>
90
+ )}
91
+ </div>
92
+
93
+ {isOpen && (
94
+ <div className={styles.dropdown}>
95
+ {filteredIds.length === 0 ? (
96
+ <div className={styles.dropdownEmpty}>
97
+ No correlation IDs found
98
+ </div>
99
+ ) : (
100
+ filteredIds.map((item) => (
101
+ <div
102
+ key={item.correlation_id}
103
+ onClick={() => handleSelect(item.correlation_id)}
104
+ className={styles.dropdownItem}
105
+ >
106
+ <div className={styles.correlationId}>
107
+ {item.correlation_id}
108
+ </div>
109
+ <div className={styles.metadata}>
110
+ {item.artifact_count} messages, {formatTimeAgo(item.first_seen)}
111
+ </div>
112
+ </div>
113
+ ))
114
+ )}
115
+ </div>
116
+ )}
117
+ </div>
118
+ );
119
+ };
120
+
121
+ export default CorrelationIDFilter;
@@ -0,0 +1,29 @@
1
+ .filterBar {
2
+ display: flex;
3
+ flex-direction: column;
4
+ gap: var(--space-layout-sm);
5
+ padding: var(--space-layout-sm) var(--space-layout-md);
6
+ background-color: var(--color-bg-elevated);
7
+ border-bottom: 1px solid var(--color-border-subtle);
8
+ }
9
+
10
+ .filterControls {
11
+ display: flex;
12
+ gap: var(--space-layout-sm);
13
+ align-items: flex-start;
14
+ flex-wrap: wrap;
15
+ }
16
+
17
+ .filterGroup {
18
+ display: flex;
19
+ flex-direction: column;
20
+ gap: var(--gap-md);
21
+ }
22
+
23
+ .filterLabel {
24
+ font-size: var(--font-size-overline);
25
+ font-weight: var(--font-weight-semibold);
26
+ color: var(--color-text-tertiary);
27
+ text-transform: uppercase;
28
+ letter-spacing: var(--letter-spacing-wider);
29
+ }
@@ -0,0 +1,133 @@
1
+ import { describe, it, expect, vi } from 'vitest';
2
+ import { render, screen } from '@testing-library/react';
3
+ import FilterBar from './FilterBar';
4
+ import { useFilterStore } from '../../store/filterStore';
5
+
6
+ vi.mock('../../store/filterStore');
7
+ vi.mock('./CorrelationIDFilter', () => ({
8
+ default: () => <div data-testid="correlation-id-filter">CorrelationIDFilter</div>,
9
+ }));
10
+ vi.mock('./TimeRangeFilter', () => ({
11
+ default: () => <div data-testid="time-range-filter">TimeRangeFilter</div>,
12
+ }));
13
+ vi.mock('./FilterPills', () => ({
14
+ default: () => <div data-testid="filter-pills">FilterPills</div>,
15
+ }));
16
+
17
+ describe('FilterBar', () => {
18
+ it('should render all filter components', () => {
19
+ vi.mocked(useFilterStore).mockImplementation((selector: any) => {
20
+ const state = {
21
+ correlationId: null,
22
+ timeRange: { preset: 'last10min' },
23
+ availableCorrelationIds: [],
24
+ getActiveFilters: () => [],
25
+ };
26
+ return selector(state);
27
+ });
28
+
29
+ render(<FilterBar />);
30
+
31
+ expect(screen.getByTestId('correlation-id-filter')).toBeInTheDocument();
32
+ expect(screen.getByTestId('time-range-filter')).toBeInTheDocument();
33
+ expect(screen.getByTestId('filter-pills')).toBeInTheDocument();
34
+ });
35
+
36
+ it('should have proper layout structure', () => {
37
+ vi.mocked(useFilterStore).mockImplementation((selector: any) => {
38
+ const state = {
39
+ correlationId: null,
40
+ timeRange: { preset: 'last10min' },
41
+ availableCorrelationIds: [],
42
+ getActiveFilters: () => [],
43
+ };
44
+ return selector(state);
45
+ });
46
+
47
+ const { container } = render(<FilterBar />);
48
+
49
+ // Should have a container with CSS module class (hashed)
50
+ const filterBar = container.firstChild as HTMLElement;
51
+ expect(filterBar.className).toMatch(/filterBar/);
52
+ });
53
+
54
+ it('should render correlation ID filter and time range filter in top row', () => {
55
+ vi.mocked(useFilterStore).mockImplementation((selector: any) => {
56
+ const state = {
57
+ correlationId: null,
58
+ timeRange: { preset: 'last10min' },
59
+ availableCorrelationIds: [],
60
+ getActiveFilters: () => [],
61
+ };
62
+ return selector(state);
63
+ });
64
+
65
+ render(<FilterBar />);
66
+
67
+ const correlationFilter = screen.getByTestId('correlation-id-filter');
68
+ const timeRangeFilter = screen.getByTestId('time-range-filter');
69
+
70
+ // Both should be present
71
+ expect(correlationFilter).toBeInTheDocument();
72
+ expect(timeRangeFilter).toBeInTheDocument();
73
+ });
74
+
75
+ it('should render filter pills below filter controls', () => {
76
+ vi.mocked(useFilterStore).mockImplementation((selector: any) => {
77
+ const state = {
78
+ correlationId: 'test-123',
79
+ timeRange: { preset: 'last5min' },
80
+ availableCorrelationIds: [],
81
+ getActiveFilters: () => [
82
+ {
83
+ type: 'correlationId',
84
+ value: 'test-123',
85
+ label: 'Correlation ID: test-123',
86
+ },
87
+ ],
88
+ };
89
+ return selector(state);
90
+ });
91
+
92
+ render(<FilterBar />);
93
+
94
+ expect(screen.getByTestId('filter-pills')).toBeInTheDocument();
95
+ });
96
+
97
+ it('should have appropriate spacing between components', () => {
98
+ vi.mocked(useFilterStore).mockImplementation((selector: any) => {
99
+ const state = {
100
+ correlationId: null,
101
+ timeRange: { preset: 'last10min' },
102
+ availableCorrelationIds: [],
103
+ getActiveFilters: () => [],
104
+ };
105
+ return selector(state);
106
+ });
107
+
108
+ const { container } = render(<FilterBar />);
109
+ const filterBar = container.firstChild as HTMLElement;
110
+
111
+ // Should have filter controls with CSS module class (hashed)
112
+ const filterControls = filterBar.querySelector('[class*="filterControls"]');
113
+ expect(filterControls).toBeInTheDocument();
114
+ });
115
+
116
+ it('should maintain consistent styling with dashboard theme', () => {
117
+ vi.mocked(useFilterStore).mockImplementation((selector: any) => {
118
+ const state = {
119
+ correlationId: null,
120
+ timeRange: { preset: 'last10min' },
121
+ availableCorrelationIds: [],
122
+ getActiveFilters: () => [],
123
+ };
124
+ return selector(state);
125
+ });
126
+
127
+ const { container } = render(<FilterBar />);
128
+ const filterBar = container.firstChild as HTMLElement;
129
+
130
+ // Should have padding and background consistent with dashboard
131
+ expect(filterBar).toBeDefined();
132
+ });
133
+ });
@@ -0,0 +1,33 @@
1
+ import React from 'react';
2
+ import CorrelationIDFilter from './CorrelationIDFilter';
3
+ import TimeRangeFilter from './TimeRangeFilter';
4
+ import FilterPills from './FilterPills';
5
+ import styles from './FilterBar.module.css';
6
+
7
+ const FilterBar: React.FC = () => {
8
+ return (
9
+ <div className={styles.filterBar}>
10
+ {/* Filter Controls */}
11
+ <div className={styles.filterControls}>
12
+ <div className={styles.filterGroup}>
13
+ <label className={styles.filterLabel}>
14
+ Correlation ID
15
+ </label>
16
+ <CorrelationIDFilter />
17
+ </div>
18
+
19
+ <div className={styles.filterGroup}>
20
+ <label className={styles.filterLabel}>
21
+ Time Range
22
+ </label>
23
+ <TimeRangeFilter />
24
+ </div>
25
+ </div>
26
+
27
+ {/* Active Filter Pills */}
28
+ <FilterPills />
29
+ </div>
30
+ );
31
+ };
32
+
33
+ export default FilterBar;
@@ -0,0 +1,79 @@
1
+ .container {
2
+ display: flex;
3
+ gap: var(--spacing-2);
4
+ align-items: center;
5
+ flex-wrap: wrap;
6
+ }
7
+
8
+ .pill {
9
+ display: flex;
10
+ align-items: center;
11
+ gap: var(--gap-md);
12
+ padding: var(--spacing-1-5) var(--spacing-3);
13
+ background: linear-gradient(135deg, var(--color-primary-600) 0%, var(--color-primary-700) 100%);
14
+ border: 1px solid var(--color-primary-500);
15
+ border-radius: var(--radius-full);
16
+ font-size: var(--font-size-body-xs);
17
+ font-weight: var(--font-weight-medium);
18
+ color: var(--color-text-on-primary);
19
+ box-shadow: var(--shadow-xs);
20
+ transition: var(--transition-all);
21
+ animation: slideIn var(--duration-normal) var(--ease-out);
22
+ }
23
+
24
+ .pill:hover {
25
+ transform: translateY(-1px);
26
+ box-shadow: var(--shadow-sm);
27
+ }
28
+
29
+ .pillSecondary {
30
+ background: linear-gradient(135deg, var(--color-secondary-600) 0%, var(--color-secondary-700) 100%);
31
+ border-color: var(--color-secondary-500);
32
+ }
33
+
34
+ .pillLabel {
35
+ user-select: none;
36
+ }
37
+
38
+ .removeButton {
39
+ background: none;
40
+ border: none;
41
+ cursor: pointer;
42
+ padding: 0 var(--spacing-0-5);
43
+ font-size: 16px;
44
+ color: var(--color-text-on-primary);
45
+ line-height: 1;
46
+ display: flex;
47
+ align-items: center;
48
+ justify-content: center;
49
+ transition: var(--transition-transform);
50
+ opacity: 0.8;
51
+ }
52
+
53
+ .removeButton:hover {
54
+ opacity: 1;
55
+ transform: scale(1.2);
56
+ }
57
+
58
+ @keyframes slideIn {
59
+ from {
60
+ opacity: 0;
61
+ transform: translateX(-10px);
62
+ }
63
+ to {
64
+ opacity: 1;
65
+ transform: translateX(0);
66
+ }
67
+ }
68
+
69
+ /* Animation when pill is removed */
70
+ .pill.removing {
71
+ animation: slideOut var(--duration-fast) var(--ease-in) forwards;
72
+ }
73
+
74
+ @keyframes slideOut {
75
+ to {
76
+ opacity: 0;
77
+ transform: scale(0.8) translateX(-10px);
78
+ }
79
+ }