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

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

Potentially problematic release.


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

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.542.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.542.dist-info/METADATA +0 -676
  496. flock_core-0.4.542.dist-info/RECORD +0 -572
  497. flock_core-0.4.542.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.542.dist-info → flock_core-0.5.0.dist-info}/WHEEL +0 -0
flock/webapp/app/chat.py DELETED
@@ -1,662 +0,0 @@
1
- from __future__ import annotations
2
-
3
- import ast # Add import ast
4
- import json
5
- from datetime import datetime
6
- from typing import Literal
7
- from uuid import uuid4
8
-
9
- import markdown2 # Added for Markdown to HTML conversion
10
- from fastapi import APIRouter, Depends, Form, Request, Response
11
- from fastapi.responses import FileResponse, HTMLResponse
12
- from pydantic import BaseModel
13
-
14
- from flock.core.flock import Flock
15
- from flock.core.logging.logging import get_logger
16
- from flock.webapp.app.dependencies import get_shared_link_store
17
- from flock.webapp.app.main import get_base_context_web, templates
18
- from flock.webapp.app.services.feedback_file_service import (
19
- create_csv_feedback_file,
20
- create_csv_feedback_file_for_agent,
21
- create_xlsx_feedback_file,
22
- create_xlsx_feedback_file_for_agent,
23
- )
24
- from flock.webapp.app.services.sharing_models import (
25
- FeedbackRecord,
26
- SharedLinkConfig,
27
- )
28
- from flock.webapp.app.services.sharing_store import SharedLinkStoreInterface
29
-
30
- router = APIRouter()
31
- logger = get_logger("webapp.chat")
32
-
33
- # ---------------------------------------------------------------------------
34
- # In-memory session store (cookie-based). Not suitable for production scale.
35
- # ---------------------------------------------------------------------------
36
- _chat_sessions: dict[str, list[dict[str, str]]] = {}
37
-
38
- COOKIE_NAME = "chat_sid"
39
-
40
-
41
- def _ensure_session(request: Request) -> tuple[str, list[dict[str, str]]]:
42
- """Returns (sid, history_list) tuple and guarantees cookie presence."""
43
- sid: str | None = request.cookies.get(COOKIE_NAME)
44
- if not sid:
45
- sid = uuid4().hex
46
- if sid not in _chat_sessions:
47
- _chat_sessions[sid] = []
48
- return sid, _chat_sessions[sid]
49
-
50
-
51
- def _get_history_for_shared_chat(request: Request, share_id: str) -> list[dict[str, str]]:
52
- """Manages history for a shared chat session, namespaced by share_id and user's session_id."""
53
- user_sid: str | None = request.cookies.get(COOKIE_NAME)
54
- if not user_sid: # Should have been set by _ensure_session on page load
55
- user_sid = uuid4().hex
56
- # Note: This history will be ephemeral if the cookie isn't set back to the client,
57
- # but _ensure_session on the shared chat page load should handle cookie setting.
58
-
59
- # Composite key for shared chat history
60
- shared_session_key = f"shared_{share_id}_{user_sid}"
61
- if shared_session_key not in _chat_sessions:
62
- _chat_sessions[shared_session_key] = []
63
- return _chat_sessions[shared_session_key]
64
-
65
-
66
- # ---------------------------------------------------------------------------
67
- # Chat configuration (per app instance for non-shared, or from SharedLinkConfig for shared)
68
- # ---------------------------------------------------------------------------
69
-
70
-
71
- class ChatConfig(BaseModel):
72
- agent_name: str | None = None # Name of the Flock agent to chat with
73
- message_key: str = "message"
74
- history_key: str = "history"
75
- response_key: str = "response"
76
-
77
-
78
- # Store a single global chat config on the FastAPI app state for non-shared chat
79
- def get_chat_config(request: Request) -> ChatConfig:
80
- if not hasattr(request.app.state, "chat_config"):
81
- request.app.state.chat_config = ChatConfig()
82
- return request.app.state.chat_config
83
-
84
-
85
- # ---------------------------------------------------------------------------
86
- # Helper for Shared Chat Context
87
- # ---------------------------------------------------------------------------
88
- async def _get_shared_chat_context(
89
- request: Request,
90
- share_id: str,
91
- store: SharedLinkStoreInterface = Depends(get_shared_link_store)
92
- ) -> tuple[ChatConfig | None, Flock | None, SharedLinkConfig | None]:
93
- shared_config_db = await store.get_config(share_id)
94
-
95
- if not shared_config_db or shared_config_db.share_type != "chat":
96
- logger.warning(f"Shared chat link {share_id} not found or not a chat share type.")
97
- return None, None, None
98
-
99
- # Retrieve the pre-loaded Flock instance for this share_id
100
- # This is loaded by the /chat/shared/{share_id} endpoint in main.py (or will be)
101
- # For chat.py, we will create a specific /chat/shared/{share_id} endpoint
102
-
103
- loaded_flock: Flock | None = None
104
- if hasattr(request.app.state, 'shared_flocks') and share_id in request.app.state.shared_flocks:
105
- loaded_flock = request.app.state.shared_flocks[share_id]
106
- else:
107
- # Attempt to load on-the-fly if not found (e.g., direct API call without page load)
108
- # This is a fallback and might be slower if the Flock definition is large.
109
- # The main /chat/shared/{share_id} page route should pre-load this.
110
- try:
111
- from flock.core.flock import Flock as ConcreteFlock # Local import
112
- loaded_flock = ConcreteFlock.from_yaml(shared_config_db.flock_definition)
113
- if not hasattr(request.app.state, 'shared_flocks'):
114
- request.app.state.shared_flocks = {}
115
- request.app.state.shared_flocks[share_id] = loaded_flock # Cache it
116
- logger.info(f"On-the-fly load of Flock for shared chat {share_id}.")
117
- except Exception as e_load:
118
- logger.error(f"Failed to load Flock from definition for shared chat {share_id}: {e_load}", exc_info=True)
119
- return None, None, shared_config_db
120
-
121
-
122
- frozen_chat_cfg = ChatConfig(
123
- agent_name=shared_config_db.agent_name, # agent_name from SharedLinkConfig is the chat agent
124
- message_key=shared_config_db.chat_message_key or "message",
125
- history_key=shared_config_db.chat_history_key or "history",
126
- response_key=shared_config_db.chat_response_key or "response",
127
- )
128
- return frozen_chat_cfg, loaded_flock, shared_config_db
129
-
130
-
131
- # ---------------------------------------------------------------------------
132
- # Routes
133
- # ---------------------------------------------------------------------------
134
-
135
-
136
- @router.get("/chat", response_class=HTMLResponse, tags=["Chat"])
137
- async def chat_page(request: Request):
138
- """Full-page chat UI (works even when the main UI is disabled)."""
139
- sid, history = _ensure_session(request)
140
- cfg = get_chat_config(request)
141
- context = get_base_context_web(request, ui_mode="standalone")
142
- context.update({"history": history, "chat_cfg": cfg, "chat_subtitle": f"Agent: {cfg.agent_name}" if cfg.agent_name else "Echo demo", "is_shared_chat": False, "share_id": None})
143
- response = templates.TemplateResponse(request, "chat.html", context)
144
- # Set cookie if not already present
145
- if COOKIE_NAME not in request.cookies:
146
- response.set_cookie(COOKIE_NAME, sid, max_age=60 * 60 * 24 * 7)
147
- return response
148
-
149
-
150
- @router.get("/chat/messages", response_class=HTMLResponse, tags=["Chat"], include_in_schema=False)
151
- async def chat_history_partial(request: Request):
152
- """HTMX endpoint that returns the rendered message list."""
153
- _, history = _ensure_session(request)
154
- return templates.TemplateResponse(
155
- request,
156
- "partials/_chat_messages.html",
157
- {"request": request, "history": history, "now": datetime.now}
158
- )
159
-
160
-
161
- @router.post("/chat/send", response_class=HTMLResponse, tags=["Chat"])
162
- async def chat_send(request: Request, message: str = Form(...)):
163
- """Echo-back mock implementation. Adds user msg + bot reply to history."""
164
- _, history = _ensure_session(request)
165
- current_time = datetime.now().strftime('%H:%M')
166
- cfg = get_chat_config(request)
167
- history.append({"role": "user", "text": message, "timestamp": current_time})
168
- start_time = datetime.now()
169
- is_error = False # Initialize is_error
170
-
171
- flock_inst = getattr(request.app.state, "flock_instance", None)
172
- bot_agent = cfg.agent_name if cfg.agent_name else None
173
- bot_text: str
174
-
175
- if bot_agent and flock_inst and bot_agent in getattr(flock_inst, "agents", {}):
176
- run_input: dict = {}
177
- if cfg.message_key: run_input[cfg.message_key] = message
178
- if cfg.history_key: run_input[cfg.history_key] = [h["text"] for h in history if h.get("role") == "user" or h.get("role") == "bot"] # Simple text history
179
-
180
- try:
181
- result_dict = await flock_inst.run_async(start_agent=bot_agent, input=run_input, box_result=False)
182
- # Assuming result_dict might be the actual dict, or its string representation is what we need.
183
- # For now, we work with bot_text derived from it.
184
- if cfg.response_key:
185
- bot_text = str(result_dict.get(cfg.response_key, result_dict))
186
- else:
187
- bot_text = str(result_dict)
188
-
189
- except Exception as e:
190
- bot_text = f"Error: {e}"
191
- is_error = True
192
-
193
- if not is_error:
194
- original_bot_text = bot_text # Keep a copy
195
- formatted_as_json = False
196
-
197
- stripped_text = bot_text.strip()
198
- if (stripped_text.startswith('{') and stripped_text.endswith('}')) or \
199
- (stripped_text.startswith('[') and stripped_text.endswith(']')):
200
- try:
201
- parsed_obj = json.loads(bot_text)
202
- if isinstance(parsed_obj, (dict, list)):
203
- pretty_json = json.dumps(parsed_obj, indent=2).replace('\\n', '\n')
204
- bot_text = f'''<pre><code class="language-json">{pretty_json}</code></pre>'''
205
- formatted_as_json = True
206
- except json.JSONDecodeError:
207
- try:
208
- evaluated_obj = ast.literal_eval(bot_text)
209
- if isinstance(evaluated_obj, (dict, list)):
210
- pretty_json = json.dumps(evaluated_obj, indent=2).replace('\\n', '\n')
211
- bot_text = f'''<pre><code class="language-json">{pretty_json}</code></pre>'''
212
- formatted_as_json = True
213
- except (ValueError, SyntaxError, TypeError):
214
- pass # Fall through
215
- except Exception as e_json_fmt:
216
- logger.error(f"Error formatting likely JSON: {e_json_fmt}. Original: {original_bot_text[:200]}", exc_info=True)
217
- bot_text = original_bot_text
218
-
219
- if not formatted_as_json:
220
- try:
221
- bot_text = markdown2.markdown(original_bot_text, extras=["fenced-code-blocks", "tables", "break-on-newline"])
222
- except Exception:
223
- logger.error(f"Error during Markdown conversion for bot_text. Original: {original_bot_text[:200]}", exc_info=True)
224
- bot_text = original_bot_text # Fallback to original text, will be HTML escaped by Jinja if not |safe
225
-
226
- else:
227
- # Fallback echo behavior or agent not found messages
228
- is_error = True # Treat these as plain text, no special formatting
229
- if bot_agent and not flock_inst:
230
- bot_text = f"Agent '{bot_agent}' configured, but no Flock loaded."
231
- elif bot_agent and flock_inst and bot_agent not in getattr(flock_inst, "agents", {}):
232
- bot_text = f"Agent '{bot_agent}' configured, but not found in the loaded Flock."
233
- else: # No agent configured
234
- bot_text = f"Echo: {message}"
235
- # If even echo should be markdown, remove is_error=True here and let it pass through. For now, plain.
236
-
237
- duration_ms = int((datetime.now() - start_time).total_seconds() * 1000)
238
- history.append({
239
- "role": "bot",
240
- "text": bot_text,
241
- "timestamp": current_time,
242
- "agent": bot_agent or "echo",
243
- "duration_ms": duration_ms,
244
- "raw_json": original_bot_text if 'original_bot_text' in locals() else bot_text,
245
- "flock_yaml": getattr(flock_inst, 'to_yaml', lambda: "")()
246
- })
247
- # Return updated history partial
248
- return templates.TemplateResponse(request, "partials/_chat_messages.html",
249
- {"request": request, "history": history, "now": datetime.now}
250
- )
251
-
252
-
253
- @router.get("/ui/htmx/chat-view", response_class=HTMLResponse, tags=["Chat"], include_in_schema=False)
254
- async def chat_container_partial(request: Request):
255
- _ensure_session(request)
256
- return templates.TemplateResponse(request, "partials/_chat_container.html", {"request": request})
257
-
258
-
259
- # ---------------------------------------------------------------------------
260
- # Chat settings management
261
- # ---------------------------------------------------------------------------
262
-
263
-
264
- @router.get("/ui/htmx/chat-settings-form", response_class=HTMLResponse, include_in_schema=False)
265
- async def chat_settings_form(request: Request):
266
- """Returns the form for configuring chat behaviour (HTMX partial)."""
267
- cfg = get_chat_config(request)
268
- flock_inst = getattr(request.app.state, "flock_instance", None)
269
- input_fields, output_fields = [], []
270
- if cfg.agent_name and flock_inst and cfg.agent_name in flock_inst.agents:
271
- agent_obj = flock_inst.agents[cfg.agent_name]
272
- # Expect signatures like "field: type | desc, ..." or "field: type" etc.
273
- def _extract(sig: str):
274
- fields = []
275
- for seg in sig.split(','):
276
- parts = seg.strip().split(':')
277
- if parts:
278
- fields.append(parts[0].strip())
279
- return [f for f in fields if f]
280
- input_fields = _extract(agent_obj.input) if getattr(agent_obj, 'input', '') else []
281
- output_fields = _extract(agent_obj.output) if getattr(agent_obj, 'output', '') else []
282
-
283
- context = get_base_context_web(request)
284
- context.update({
285
- "chat_cfg": cfg,
286
- "current_flock": flock_inst,
287
- "input_fields": input_fields,
288
- "output_fields": output_fields,
289
- })
290
- return templates.TemplateResponse(request, "partials/_chat_settings_form.html", context)
291
-
292
-
293
- @router.post("/chat/settings", include_in_schema=False)
294
- async def chat_settings_submit(
295
- request: Request,
296
- agent_name: str | None = Form(default=None),
297
- message_key: str = Form("message"),
298
- history_key: str = Form("history"),
299
- response_key: str = Form("response"),
300
- ):
301
- """Handles chat settings submission and triggers a toast notification."""
302
- cfg = get_chat_config(request)
303
- cfg.agent_name = agent_name
304
- cfg.message_key = message_key
305
- cfg.history_key = history_key
306
- cfg.response_key = response_key
307
-
308
- logger.info(f"Chat settings updated: Agent: {cfg.agent_name}, MsgKey: {cfg.message_key}, HistKey: {cfg.history_key}, RespKey: {cfg.response_key}")
309
-
310
- toast_event = {
311
- "showGlobalToast": {
312
- "message": "Chat settings saved successfully!",
313
- "type": "success"
314
- }
315
- }
316
- headers = {"HX-Trigger": json.dumps(toast_event)}
317
- return Response(status_code=204, headers=headers)
318
-
319
-
320
- # --- Stand-alone Chat HTML page access to settings --------------------------
321
-
322
-
323
- @router.get("/chat/settings-standalone", response_class=HTMLResponse, tags=["Chat"], include_in_schema=False)
324
- async def chat_settings_standalone(request: Request):
325
- """Standalone page to render chat settings (used by full-page chat HTML)."""
326
- cfg = get_chat_config(request)
327
- context = get_base_context_web(request, ui_mode="standalone")
328
- context.update({
329
- "chat_cfg": cfg,
330
- "current_flock": getattr(request.app.state, "flock_instance", None),
331
- })
332
- return templates.TemplateResponse(request, "chat_settings.html", context)
333
-
334
-
335
- # ---------------------------------------------------------------------------
336
- # Stand-alone HTMX partials (chat view & settings) for in-page swapping
337
- # ---------------------------------------------------------------------------
338
-
339
-
340
- @router.get("/chat/htmx/chat-view", response_class=HTMLResponse, include_in_schema=False)
341
- async def htmx_chat_view(request: Request):
342
- """Return chat container partial for standalone page reload via HTMX."""
343
- _ensure_session(request)
344
- return templates.TemplateResponse(request, "partials/_chat_container.html", {"request": request})
345
-
346
-
347
- @router.get("/chat/htmx/settings-form", response_class=HTMLResponse, include_in_schema=False)
348
- async def htmx_chat_settings_partial(request: Request):
349
- cfg = get_chat_config(request)
350
- # Allow temporarily selecting agent via query param without saving
351
- agent_override = request.query_params.get("agent_name")
352
- if agent_override is not None:
353
- cfg = cfg.copy()
354
- cfg.agent_name = agent_override or None
355
-
356
- flock_inst = getattr(request.app.state, "flock_instance", None)
357
- input_fields, output_fields = [], []
358
- if cfg.agent_name and flock_inst and cfg.agent_name in flock_inst.agents:
359
- agent_obj = flock_inst.agents[cfg.agent_name]
360
- def _extract(sig: str):
361
- return [seg.strip().split(':')[0].strip() for seg in sig.split(',') if seg.strip()]
362
- input_fields = _extract(agent_obj.input) if getattr(agent_obj, 'input', '') else []
363
- output_fields = _extract(agent_obj.output) if getattr(agent_obj, 'output', '') else []
364
-
365
- context = {"request": request, "chat_cfg": cfg, "current_flock": flock_inst, "input_fields": input_fields, "output_fields": output_fields}
366
- return templates.TemplateResponse(request, "partials/_chat_settings_form.html", context)
367
-
368
-
369
- # ---------------------------------------------------------------------------
370
- # Shared Chat Routes
371
- # ---------------------------------------------------------------------------
372
-
373
- @router.get("/chat/shared/{share_id}", response_class=HTMLResponse, tags=["Chat Sharing"])
374
- async def page_shared_chat(
375
- request: Request,
376
- share_id: str,
377
- store: SharedLinkStoreInterface = Depends(get_shared_link_store)
378
- ):
379
- """Serves the chat page for a shared chat session."""
380
- logger.info(f"Accessing shared chat page for share_id: {share_id}")
381
-
382
- sid, _ = _ensure_session(request) # Ensures user has a session for history tracking
383
-
384
- shared_config_db = await store.get_config(share_id)
385
-
386
- if not shared_config_db or shared_config_db.share_type != "chat":
387
- logger.warning(f"Shared chat link {share_id} not found or not a chat share type.")
388
- # Consider rendering an error template or redirecting
389
- error_context = get_base_context_web(request, ui_mode="standalone", error="Shared chat link is invalid or has expired.")
390
- return templates.TemplateResponse(request, "error_page.html", {**error_context, "error_title": "Invalid Link"}, status_code=404)
391
-
392
- # Load Flock from definition and cache in app.state.shared_flocks
393
- loaded_flock: Flock | None = None
394
- if hasattr(request.app.state, 'shared_flocks') and share_id in request.app.state.shared_flocks:
395
- loaded_flock = request.app.state.shared_flocks[share_id]
396
- else:
397
- try:
398
- from flock.core.flock import Flock as ConcreteFlock
399
- loaded_flock = ConcreteFlock.from_yaml(shared_config_db.flock_definition)
400
- if not hasattr(request.app.state, 'shared_flocks'):
401
- request.app.state.shared_flocks = {}
402
- request.app.state.shared_flocks[share_id] = loaded_flock
403
- logger.info(f"Loaded and cached Flock for shared chat {share_id} in app.state.shared_flocks.")
404
- except Exception as e_load:
405
- logger.error(f"Fatal: Could not load Flock from definition for shared chat {share_id}: {e_load}", exc_info=True)
406
- error_context = get_base_context_web(request, ui_mode="standalone", error=f"Could not load the shared Flock configuration: {e_load!s}")
407
- return templates.TemplateResponse(request, "error_page.html", {**error_context, "error_title": "Configuration Error"}, status_code=500)
408
-
409
- frozen_chat_cfg = ChatConfig(
410
- agent_name=shared_config_db.agent_name,
411
- message_key=shared_config_db.chat_message_key or "message",
412
- history_key=shared_config_db.chat_history_key or "history",
413
- response_key=shared_config_db.chat_response_key or "response",
414
- )
415
-
416
- # Get history specific to this user and this shared chat
417
- history = _get_history_for_shared_chat(request, share_id)
418
-
419
- context = get_base_context_web(request, ui_mode="standalone")
420
- context.update({
421
- "history": history, # User-specific history for this shared chat
422
- "chat_cfg": frozen_chat_cfg, # The "frozen" config from the share link
423
- "chat_subtitle": f"Shared Chat - Agent: {frozen_chat_cfg.agent_name}" if frozen_chat_cfg.agent_name else "Shared Echo Chat",
424
- "is_shared_chat": True,
425
- "share_id": share_id,
426
- "flock": loaded_flock # Pass flock for potential display, though backend uses cached one
427
- })
428
-
429
- response = templates.TemplateResponse(request, "chat.html", context)
430
- if COOKIE_NAME not in request.cookies: # Ensure cookie is set if _ensure_session created a new one
431
- response.set_cookie(COOKIE_NAME, sid, max_age=60 * 60 * 24 * 7)
432
- return response
433
-
434
- @router.get("/chat/messages-shared/{share_id}", response_class=HTMLResponse, tags=["Chat Sharing"], include_in_schema=False)
435
- async def chat_history_partial_shared(request: Request, share_id: str):
436
- """HTMX endpoint that returns the rendered message list for a shared chat."""
437
- # _ensure_session called on page load, so cookie should exist for history keying
438
- history = _get_history_for_shared_chat(request, share_id)
439
- return templates.TemplateResponse(request, "partials/_chat_messages.html",
440
- {"request": request, "history": history, "now": datetime.now}
441
- )
442
-
443
- @router.post("/chat/send-shared", response_class=HTMLResponse, tags=["Chat Sharing"])
444
- async def chat_send_shared(
445
- request: Request,
446
- share_id: str = Form(...),
447
- message: str = Form(...),
448
- # Note: Dependencies need to be declared at the route level for FastAPI to inject them.
449
- # So, we re-declare get_shared_link_store here or pass it to the helper if FastAPI handles sub-dependencies.
450
- # For simplicity with current structure, let _get_shared_chat_context handle its own dependency.
451
- # We can also make _get_shared_chat_context a Depends() if preferred.
452
- store: SharedLinkStoreInterface = Depends(get_shared_link_store)
453
- ):
454
- """Handles message sending for a shared chat session."""
455
- frozen_chat_cfg, flock_inst, _ = await _get_shared_chat_context(request, share_id, store)
456
- is_error = False # Initialize is_error
457
-
458
- if not frozen_chat_cfg or not flock_inst:
459
- # Error response if config or flock couldn't be loaded
460
- # This history is ephemeral as it won't be saved if the config is bad
461
- error_history = [{"role": "bot", "text": "Error: Shared chat configuration is invalid or Flock not found.", "timestamp": datetime.now().strftime('%H:%M')}]
462
- return templates.TemplateResponse(request, "partials/_chat_messages.html",
463
- {"request": request, "history": error_history, "now": datetime.now},
464
- status_code=404
465
- )
466
-
467
- history = _get_history_for_shared_chat(request, share_id)
468
- current_time = datetime.now().strftime('%H:%M')
469
- history.append({"role": "user", "text": message, "timestamp": current_time})
470
- start_time = datetime.now()
471
-
472
- bot_agent = frozen_chat_cfg.agent_name
473
- bot_text: str
474
-
475
- if bot_agent and bot_agent in getattr(flock_inst, "agents", {}):
476
- run_input: dict = {}
477
- if frozen_chat_cfg.message_key: run_input[frozen_chat_cfg.message_key] = message
478
- if frozen_chat_cfg.history_key: run_input[frozen_chat_cfg.history_key] = [h["text"] for h in history if h.get("role") == "user" or h.get("role") == "bot"]
479
-
480
- try:
481
- result_dict = await flock_inst.run_async(start_agent=bot_agent, input=run_input, box_result=False)
482
- if frozen_chat_cfg.response_key:
483
- bot_text = str(result_dict.get(frozen_chat_cfg.response_key, result_dict))
484
- else:
485
- bot_text = str(result_dict)
486
-
487
- except Exception as e:
488
- bot_text = f"Error running agent {bot_agent} in shared chat: {e}"
489
- is_error = True
490
- logger.error(f"Error in /chat/send-shared (agent: {bot_agent}, share: {share_id}): {e}", exc_info=True)
491
-
492
- if not is_error:
493
- original_bot_text = bot_text # Keep a copy
494
- formatted_as_json = False
495
-
496
- stripped_text = bot_text.strip()
497
- if (stripped_text.startswith('{') and stripped_text.endswith('}')) or \
498
- (stripped_text.startswith('[') and stripped_text.endswith(']')):
499
- try:
500
- parsed_obj = json.loads(bot_text)
501
- if isinstance(parsed_obj, (dict, list)):
502
- pretty_json = json.dumps(parsed_obj, indent=2).replace('\\n', '\n')
503
- bot_text = f'''<pre><code class="language-json">{pretty_json}</code></pre>'''
504
- formatted_as_json = True
505
- except json.JSONDecodeError:
506
- try:
507
- evaluated_obj = ast.literal_eval(bot_text)
508
- if isinstance(evaluated_obj, (dict, list)):
509
- pretty_json = json.dumps(evaluated_obj, indent=2).replace('\\n', '\n')
510
- bot_text = f'''<pre><code class="language-json">{pretty_json}</code></pre>'''
511
- formatted_as_json = True
512
- except (ValueError, SyntaxError, TypeError):
513
- pass # Fall through
514
- except Exception as e_json_fmt:
515
- logger.error(f"Error formatting likely JSON (shared chat): {e_json_fmt}. Original: {original_bot_text[:200]}", exc_info=True)
516
- bot_text = original_bot_text
517
-
518
- if not formatted_as_json:
519
- try:
520
- bot_text = markdown2.markdown(original_bot_text, extras=["fenced-code-blocks", "tables", "break-on-newline"])
521
- except Exception:
522
- logger.error(f"Error during Markdown conversion for shared chat bot_text. Original: {original_bot_text[:200]}", exc_info=True)
523
- bot_text = original_bot_text
524
- else:
525
- # Fallback if agent misconfigured or not found in the specific shared flock
526
- is_error = True # Treat these as plain text
527
- if bot_agent and bot_agent not in getattr(flock_inst, "agents", {}):
528
- bot_text = f"Agent '{bot_agent}' (shared) not found in its Flock."
529
- elif not bot_agent:
530
- bot_text = f"No agent configured for this shared chat. Echoing: {message}"
531
- else: # Should not happen if frozen_chat_cfg and flock_inst were valid earlier
532
- bot_text = f"Shared Echo: {message}"
533
-
534
- duration_ms = int((datetime.now() - start_time).total_seconds() * 1000)
535
- history.append({
536
- "role": "bot",
537
- "text": bot_text,
538
- "timestamp": current_time,
539
- "agent": bot_agent or "shared-echo",
540
- "duration_ms": duration_ms,
541
- "raw_json": original_bot_text if 'original_bot_text' in locals() else bot_text,
542
- "flock_yaml": getattr(flock_inst, 'to_yaml', lambda: "")()
543
- })
544
-
545
- return templates.TemplateResponse(request, "partials/_chat_messages.html",
546
- {"request": request, "history": history, "now": datetime.now}
547
- )
548
-
549
- # ---------------- Feedback endpoints ----------------
550
- @router.post("/chat/htmx/feedback", response_class=HTMLResponse, include_in_schema=False)
551
- async def chat_feedback(request: Request,
552
- reason: str = Form(...),
553
- expected_response: str | None = Form(None),
554
- actual_response: str | None = Form(None),
555
- flock_definition: str | None = Form(None),
556
- agent_name: str | None = Form(None),
557
- store: SharedLinkStoreInterface = Depends(get_shared_link_store)):
558
- from uuid import uuid4
559
- rec = FeedbackRecord(
560
- feedback_id=uuid4().hex,
561
- share_id=None,
562
- context_type="chat",
563
- reason=reason,
564
- expected_response=expected_response,
565
- actual_response=actual_response,
566
- flock_definition=flock_definition,
567
- agent_name=agent_name,
568
- )
569
- await store.save_feedback(rec)
570
- toast_event = {
571
- "showGlobalToast": {
572
- "message": "Feedback received! Thanks",
573
- "type": "success"
574
- }
575
- }
576
- headers = {"HX-Trigger": json.dumps(toast_event)}
577
- return Response(status_code=204, headers=headers)
578
-
579
- @router.post("/chat/htmx/feedback-shared", response_class=HTMLResponse, include_in_schema=False)
580
- async def chat_feedback_shared(request: Request,
581
- share_id: str = Form(...),
582
- reason: str = Form(...),
583
- expected_response: str | None = Form(None),
584
- actual_response: str | None = Form(None),
585
- flock_definition: str | None = Form(None),
586
- agent_name: str | None = Form(None),
587
- store: SharedLinkStoreInterface = Depends(get_shared_link_store)):
588
- from uuid import uuid4
589
- rec = FeedbackRecord(
590
- feedback_id=uuid4().hex,
591
- share_id=share_id,
592
- context_type="chat",
593
- reason=reason,
594
- expected_response=expected_response,
595
- actual_response=actual_response,
596
- flock_definition=flock_definition,
597
- agent_name=agent_name,
598
- )
599
- await store.save_feedback(rec)
600
- toast_event = {
601
- "showGlobalToast": {
602
- "message": "Feedback received! Thanks",
603
- "type": "success"
604
- }
605
- }
606
- headers = {"HX-Trigger": json.dumps(toast_event)}
607
- return Response(status_code=204, headers=headers)
608
-
609
- @router.get("/chat/htmx/feedback-download/{format}", response_class=FileResponse, include_in_schema=False)
610
- async def chat_feedback_download_all(
611
- request: Request,
612
- format: Literal["csv", "xlsx"] = "csv",
613
- store: SharedLinkStoreInterface = Depends(get_shared_link_store)
614
- ):
615
- """Download all feedback records for all agents."""
616
- if format == "csv":
617
- return await create_csv_feedback_file(
618
- request=request,
619
- store=store,
620
- separator=","
621
- )
622
- elif format == "xlsx":
623
- return await create_xlsx_feedback_file(
624
- request=request,
625
- store=store,
626
- )
627
- else:
628
- from fastapi import HTTPException
629
-
630
- raise HTTPException(
631
- status_code=400,
632
- detail="Invalid file-format specified. Valid formats are: 'csv', 'xlsx'"
633
- )
634
-
635
- @router.get("/chat/htmx/feedback-download/{agent_name}/{format}", response_class=FileResponse, include_in_schema=False)
636
- async def chat_feedback_download(
637
- request: Request,
638
- agent_name: str,
639
- format: Literal["csv", "xlsx"] = "csv",
640
- store: SharedLinkStoreInterface = Depends(get_shared_link_store)
641
- ):
642
- """Download all feedback records for a specific agent as a File."""
643
- if format == "csv":
644
- return await create_csv_feedback_file_for_agent(
645
- request=request,
646
- store=store,
647
- separator=",",
648
- agent_name=agent_name,
649
- )
650
- elif format == "xlsx":
651
- return await create_xlsx_feedback_file_for_agent(
652
- request=request,
653
- store=store,
654
- agent_name=agent_name,
655
- )
656
- else:
657
- from fastapi import HTTPException
658
-
659
- raise HTTPException(
660
- status_code=400,
661
- detail="Invalid file-format specified. Valid formats are: 'csv', 'xlsx'"
662
- )