flock-core 0.5.0b28__py3-none-any.whl → 0.5.56b0__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 (359) 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/{core/util → helper}/cli_helper.py +4 -3
  17. flock/{core/logging → logging}/__init__.py +2 -3
  18. flock/{core/logging → logging}/formatters/enum_builder.py +3 -4
  19. flock/{core/logging → logging}/formatters/theme_builder.py +19 -44
  20. flock/{core/logging → logging}/formatters/themed_formatter.py +69 -115
  21. flock/{core/logging → logging}/logging.py +77 -61
  22. flock/{core/logging → logging}/telemetry.py +20 -26
  23. flock/{core/logging → logging}/telemetry_exporter/base_exporter.py +2 -2
  24. flock/{core/logging → logging}/telemetry_exporter/file_exporter.py +6 -9
  25. flock/{core/logging → logging}/telemetry_exporter/sqlite_exporter.py +2 -3
  26. flock/{core/logging → logging}/trace_and_logged.py +20 -24
  27. flock/mcp/__init__.py +91 -0
  28. flock/{core/mcp/mcp_client.py → mcp/client.py} +103 -154
  29. flock/{core/mcp/mcp_config.py → mcp/config.py} +62 -117
  30. flock/mcp/manager.py +255 -0
  31. flock/mcp/servers/sse/__init__.py +1 -1
  32. flock/mcp/servers/sse/flock_sse_server.py +11 -53
  33. flock/mcp/servers/stdio/__init__.py +1 -1
  34. flock/mcp/servers/stdio/flock_stdio_server.py +8 -48
  35. flock/mcp/servers/streamable_http/flock_streamable_http_server.py +17 -62
  36. flock/mcp/servers/websockets/flock_websocket_server.py +7 -40
  37. flock/{core/mcp/flock_mcp_tool.py → mcp/tool.py} +16 -26
  38. flock/mcp/types/__init__.py +42 -0
  39. flock/{core/mcp → mcp}/types/callbacks.py +9 -15
  40. flock/{core/mcp → mcp}/types/factories.py +7 -6
  41. flock/{core/mcp → mcp}/types/handlers.py +13 -18
  42. flock/{core/mcp → mcp}/types/types.py +70 -74
  43. flock/{core/mcp → mcp}/util/helpers.py +1 -1
  44. flock/orchestrator.py +645 -0
  45. flock/registry.py +148 -0
  46. flock/runtime.py +262 -0
  47. flock/service.py +140 -0
  48. flock/store.py +69 -0
  49. flock/subscription.py +111 -0
  50. flock/themes/andromeda.toml +1 -1
  51. flock/themes/apple-system-colors.toml +1 -1
  52. flock/themes/arcoiris.toml +1 -1
  53. flock/themes/atomonelight.toml +1 -1
  54. flock/themes/ayu copy.toml +1 -1
  55. flock/themes/ayu-light.toml +1 -1
  56. flock/themes/belafonte-day.toml +1 -1
  57. flock/themes/belafonte-night.toml +1 -1
  58. flock/themes/blulocodark.toml +1 -1
  59. flock/themes/breeze.toml +1 -1
  60. flock/themes/broadcast.toml +1 -1
  61. flock/themes/brogrammer.toml +1 -1
  62. flock/themes/builtin-dark.toml +1 -1
  63. flock/themes/builtin-pastel-dark.toml +1 -1
  64. flock/themes/catppuccin-latte.toml +1 -1
  65. flock/themes/catppuccin-macchiato.toml +1 -1
  66. flock/themes/catppuccin-mocha.toml +1 -1
  67. flock/themes/cga.toml +1 -1
  68. flock/themes/chalk.toml +1 -1
  69. flock/themes/ciapre.toml +1 -1
  70. flock/themes/coffee-theme.toml +1 -1
  71. flock/themes/cyberpunkscarletprotocol.toml +1 -1
  72. flock/themes/dark+.toml +1 -1
  73. flock/themes/darkermatrix.toml +1 -1
  74. flock/themes/darkside.toml +1 -1
  75. flock/themes/desert.toml +1 -1
  76. flock/themes/django.toml +1 -1
  77. flock/themes/djangosmooth.toml +1 -1
  78. flock/themes/doomone.toml +1 -1
  79. flock/themes/dotgov.toml +1 -1
  80. flock/themes/dracula+.toml +1 -1
  81. flock/themes/duckbones.toml +1 -1
  82. flock/themes/encom.toml +1 -1
  83. flock/themes/espresso.toml +1 -1
  84. flock/themes/everblush.toml +1 -1
  85. flock/themes/fairyfloss.toml +1 -1
  86. flock/themes/fideloper.toml +1 -1
  87. flock/themes/fishtank.toml +1 -1
  88. flock/themes/flexoki-light.toml +1 -1
  89. flock/themes/floraverse.toml +1 -1
  90. flock/themes/framer.toml +1 -1
  91. flock/themes/galizur.toml +1 -1
  92. flock/themes/github.toml +1 -1
  93. flock/themes/grass.toml +1 -1
  94. flock/themes/grey-green.toml +1 -1
  95. flock/themes/gruvboxlight.toml +1 -1
  96. flock/themes/guezwhoz.toml +1 -1
  97. flock/themes/harper.toml +1 -1
  98. flock/themes/hax0r-blue.toml +1 -1
  99. flock/themes/hopscotch.256.toml +1 -1
  100. flock/themes/ic-green-ppl.toml +1 -1
  101. flock/themes/iceberg-dark.toml +1 -1
  102. flock/themes/japanesque.toml +1 -1
  103. flock/themes/jubi.toml +1 -1
  104. flock/themes/kibble.toml +1 -1
  105. flock/themes/kolorit.toml +1 -1
  106. flock/themes/kurokula.toml +1 -1
  107. flock/themes/materialdesigncolors.toml +1 -1
  108. flock/themes/matrix.toml +1 -1
  109. flock/themes/mellifluous.toml +1 -1
  110. flock/themes/midnight-in-mojave.toml +1 -1
  111. flock/themes/monokai-remastered.toml +1 -1
  112. flock/themes/monokai-soda.toml +1 -1
  113. flock/themes/neon.toml +1 -1
  114. flock/themes/neopolitan.toml +1 -1
  115. flock/themes/nord-light.toml +1 -1
  116. flock/themes/ocean.toml +1 -1
  117. flock/themes/onehalfdark.toml +1 -1
  118. flock/themes/onehalflight.toml +1 -1
  119. flock/themes/palenighthc.toml +1 -1
  120. flock/themes/paulmillr.toml +1 -1
  121. flock/themes/pencildark.toml +1 -1
  122. flock/themes/pnevma.toml +1 -1
  123. flock/themes/purple-rain.toml +1 -1
  124. flock/themes/purplepeter.toml +1 -1
  125. flock/themes/raycast-dark.toml +1 -1
  126. flock/themes/red-sands.toml +1 -1
  127. flock/themes/relaxed.toml +1 -1
  128. flock/themes/retro.toml +1 -1
  129. flock/themes/rose-pine.toml +1 -1
  130. flock/themes/royal.toml +1 -1
  131. flock/themes/ryuuko.toml +1 -1
  132. flock/themes/sakura.toml +1 -1
  133. flock/themes/scarlet-protocol.toml +1 -1
  134. flock/themes/seoulbones-dark.toml +1 -1
  135. flock/themes/shades-of-purple.toml +1 -1
  136. flock/themes/smyck.toml +1 -1
  137. flock/themes/softserver.toml +1 -1
  138. flock/themes/solarized-darcula.toml +1 -1
  139. flock/themes/square.toml +1 -1
  140. flock/themes/sugarplum.toml +1 -1
  141. flock/themes/thayer-bright.toml +1 -1
  142. flock/themes/tokyonight.toml +1 -1
  143. flock/themes/tomorrow.toml +1 -1
  144. flock/themes/ubuntu.toml +1 -1
  145. flock/themes/ultradark.toml +1 -1
  146. flock/themes/ultraviolent.toml +1 -1
  147. flock/themes/unikitty.toml +1 -1
  148. flock/themes/urple.toml +1 -1
  149. flock/themes/vesper.toml +1 -1
  150. flock/themes/vimbones.toml +1 -1
  151. flock/themes/wildcherry.toml +1 -1
  152. flock/themes/wilmersdorf.toml +1 -1
  153. flock/themes/wryan.toml +1 -1
  154. flock/themes/xcodedarkhc.toml +1 -1
  155. flock/themes/xcodelight.toml +1 -1
  156. flock/themes/zenbones-light.toml +1 -1
  157. flock/themes/zenwritten-dark.toml +1 -1
  158. flock/utilities.py +301 -0
  159. flock/{components/utility → utility}/output_utility_component.py +68 -53
  160. flock/visibility.py +107 -0
  161. flock_core-0.5.56b0.dist-info/METADATA +747 -0
  162. flock_core-0.5.56b0.dist-info/RECORD +398 -0
  163. flock_core-0.5.56b0.dist-info/entry_points.txt +2 -0
  164. {flock_core-0.5.0b28.dist-info → flock_core-0.5.56b0.dist-info}/licenses/LICENSE +1 -1
  165. flock/adapter/__init__.py +0 -14
  166. flock/adapter/azure_adapter.py +0 -68
  167. flock/adapter/chroma_adapter.py +0 -73
  168. flock/adapter/faiss_adapter.py +0 -97
  169. flock/adapter/pinecone_adapter.py +0 -51
  170. flock/adapter/vector_base.py +0 -47
  171. flock/cli/assets/release_notes.md +0 -140
  172. flock/cli/config.py +0 -8
  173. flock/cli/constants.py +0 -36
  174. flock/cli/create_agent.py +0 -1
  175. flock/cli/create_flock.py +0 -280
  176. flock/cli/execute_flock.py +0 -620
  177. flock/cli/load_agent.py +0 -1
  178. flock/cli/load_examples.py +0 -1
  179. flock/cli/load_flock.py +0 -192
  180. flock/cli/load_release_notes.py +0 -20
  181. flock/cli/loaded_flock_cli.py +0 -254
  182. flock/cli/manage_agents.py +0 -459
  183. flock/cli/registry_management.py +0 -889
  184. flock/cli/runner.py +0 -41
  185. flock/cli/settings.py +0 -857
  186. flock/cli/utils.py +0 -135
  187. flock/cli/view_results.py +0 -29
  188. flock/cli/yaml_editor.py +0 -396
  189. flock/components/__init__.py +0 -30
  190. flock/components/evaluation/__init__.py +0 -9
  191. flock/components/evaluation/declarative_evaluation_component.py +0 -606
  192. flock/components/routing/__init__.py +0 -15
  193. flock/components/routing/conditional_routing_component.py +0 -494
  194. flock/components/routing/default_routing_component.py +0 -103
  195. flock/components/routing/llm_routing_component.py +0 -206
  196. flock/components/utility/__init__.py +0 -22
  197. flock/components/utility/example_utility_component.py +0 -250
  198. flock/components/utility/feedback_utility_component.py +0 -206
  199. flock/components/utility/memory_utility_component.py +0 -550
  200. flock/components/utility/metrics_utility_component.py +0 -700
  201. flock/config.py +0 -61
  202. flock/core/__init__.py +0 -110
  203. flock/core/agent/__init__.py +0 -16
  204. flock/core/agent/default_agent.py +0 -216
  205. flock/core/agent/flock_agent_components.py +0 -104
  206. flock/core/agent/flock_agent_execution.py +0 -101
  207. flock/core/agent/flock_agent_integration.py +0 -260
  208. flock/core/agent/flock_agent_lifecycle.py +0 -186
  209. flock/core/agent/flock_agent_serialization.py +0 -381
  210. flock/core/api/__init__.py +0 -10
  211. flock/core/api/custom_endpoint.py +0 -45
  212. flock/core/api/endpoints.py +0 -254
  213. flock/core/api/main.py +0 -162
  214. flock/core/api/models.py +0 -97
  215. flock/core/api/run_store.py +0 -224
  216. flock/core/api/runner.py +0 -44
  217. flock/core/api/service.py +0 -214
  218. flock/core/component/__init__.py +0 -15
  219. flock/core/component/agent_component_base.py +0 -309
  220. flock/core/component/evaluation_component.py +0 -62
  221. flock/core/component/routing_component.py +0 -74
  222. flock/core/component/utility_component.py +0 -69
  223. flock/core/config/flock_agent_config.py +0 -58
  224. flock/core/config/scheduled_agent_config.py +0 -40
  225. flock/core/context/context.py +0 -213
  226. flock/core/context/context_manager.py +0 -37
  227. flock/core/context/context_vars.py +0 -10
  228. flock/core/evaluation/utils.py +0 -396
  229. flock/core/execution/batch_executor.py +0 -369
  230. flock/core/execution/evaluation_executor.py +0 -438
  231. flock/core/execution/local_executor.py +0 -31
  232. flock/core/execution/opik_executor.py +0 -103
  233. flock/core/execution/temporal_executor.py +0 -164
  234. flock/core/flock.py +0 -634
  235. flock/core/flock_agent.py +0 -336
  236. flock/core/flock_factory.py +0 -613
  237. flock/core/flock_scheduler.py +0 -166
  238. flock/core/flock_server_manager.py +0 -136
  239. flock/core/interpreter/python_interpreter.py +0 -689
  240. flock/core/mcp/__init__.py +0 -1
  241. flock/core/mcp/flock_mcp_server.py +0 -680
  242. flock/core/mcp/mcp_client_manager.py +0 -201
  243. flock/core/mcp/types/__init__.py +0 -1
  244. flock/core/mixin/dspy_integration.py +0 -403
  245. flock/core/mixin/prompt_parser.py +0 -125
  246. flock/core/orchestration/__init__.py +0 -15
  247. flock/core/orchestration/flock_batch_processor.py +0 -94
  248. flock/core/orchestration/flock_evaluator.py +0 -113
  249. flock/core/orchestration/flock_execution.py +0 -295
  250. flock/core/orchestration/flock_initialization.py +0 -149
  251. flock/core/orchestration/flock_server_manager.py +0 -67
  252. flock/core/orchestration/flock_web_server.py +0 -117
  253. flock/core/registry/__init__.py +0 -45
  254. flock/core/registry/agent_registry.py +0 -69
  255. flock/core/registry/callable_registry.py +0 -139
  256. flock/core/registry/component_discovery.py +0 -142
  257. flock/core/registry/component_registry.py +0 -64
  258. flock/core/registry/config_mapping.py +0 -64
  259. flock/core/registry/decorators.py +0 -137
  260. flock/core/registry/registry_hub.py +0 -205
  261. flock/core/registry/server_registry.py +0 -57
  262. flock/core/registry/type_registry.py +0 -86
  263. flock/core/serialization/__init__.py +0 -13
  264. flock/core/serialization/callable_registry.py +0 -52
  265. flock/core/serialization/flock_serializer.py +0 -832
  266. flock/core/serialization/json_encoder.py +0 -41
  267. flock/core/serialization/secure_serializer.py +0 -175
  268. flock/core/serialization/serializable.py +0 -342
  269. flock/core/serialization/serialization_utils.py +0 -412
  270. flock/core/util/file_path_utils.py +0 -223
  271. flock/core/util/hydrator.py +0 -309
  272. flock/core/util/input_resolver.py +0 -164
  273. flock/core/util/loader.py +0 -59
  274. flock/core/util/splitter.py +0 -219
  275. flock/di.py +0 -27
  276. flock/platform/docker_tools.py +0 -49
  277. flock/platform/jaeger_install.py +0 -86
  278. flock/webapp/__init__.py +0 -1
  279. flock/webapp/app/__init__.py +0 -0
  280. flock/webapp/app/api/__init__.py +0 -0
  281. flock/webapp/app/api/agent_management.py +0 -241
  282. flock/webapp/app/api/execution.py +0 -709
  283. flock/webapp/app/api/flock_management.py +0 -129
  284. flock/webapp/app/api/registry_viewer.py +0 -30
  285. flock/webapp/app/chat.py +0 -665
  286. flock/webapp/app/config.py +0 -104
  287. flock/webapp/app/dependencies.py +0 -117
  288. flock/webapp/app/main.py +0 -1070
  289. flock/webapp/app/middleware.py +0 -113
  290. flock/webapp/app/models_ui.py +0 -7
  291. flock/webapp/app/services/__init__.py +0 -0
  292. flock/webapp/app/services/feedback_file_service.py +0 -363
  293. flock/webapp/app/services/flock_service.py +0 -337
  294. flock/webapp/app/services/sharing_models.py +0 -81
  295. flock/webapp/app/services/sharing_store.py +0 -762
  296. flock/webapp/app/templates/theme_mapper.html +0 -326
  297. flock/webapp/app/theme_mapper.py +0 -812
  298. flock/webapp/app/utils.py +0 -85
  299. flock/webapp/run.py +0 -215
  300. flock/webapp/static/css/chat.css +0 -301
  301. flock/webapp/static/css/components.css +0 -167
  302. flock/webapp/static/css/header.css +0 -39
  303. flock/webapp/static/css/layout.css +0 -46
  304. flock/webapp/static/css/sidebar.css +0 -127
  305. flock/webapp/static/css/two-pane.css +0 -48
  306. flock/webapp/templates/base.html +0 -200
  307. flock/webapp/templates/chat.html +0 -152
  308. flock/webapp/templates/chat_settings.html +0 -19
  309. flock/webapp/templates/flock_editor.html +0 -16
  310. flock/webapp/templates/index.html +0 -12
  311. flock/webapp/templates/partials/_agent_detail_form.html +0 -93
  312. flock/webapp/templates/partials/_agent_list.html +0 -18
  313. flock/webapp/templates/partials/_agent_manager_view.html +0 -51
  314. flock/webapp/templates/partials/_agent_tools_checklist.html +0 -14
  315. flock/webapp/templates/partials/_chat_container.html +0 -15
  316. flock/webapp/templates/partials/_chat_messages.html +0 -57
  317. flock/webapp/templates/partials/_chat_settings_form.html +0 -85
  318. flock/webapp/templates/partials/_create_flock_form.html +0 -50
  319. flock/webapp/templates/partials/_dashboard_flock_detail.html +0 -17
  320. flock/webapp/templates/partials/_dashboard_flock_file_list.html +0 -16
  321. flock/webapp/templates/partials/_dashboard_flock_properties_preview.html +0 -28
  322. flock/webapp/templates/partials/_dashboard_upload_flock_form.html +0 -16
  323. flock/webapp/templates/partials/_dynamic_input_form_content.html +0 -22
  324. flock/webapp/templates/partials/_env_vars_table.html +0 -23
  325. flock/webapp/templates/partials/_execution_form.html +0 -118
  326. flock/webapp/templates/partials/_execution_view_container.html +0 -28
  327. flock/webapp/templates/partials/_flock_file_list.html +0 -23
  328. flock/webapp/templates/partials/_flock_properties_form.html +0 -52
  329. flock/webapp/templates/partials/_flock_upload_form.html +0 -16
  330. flock/webapp/templates/partials/_header_flock_status.html +0 -5
  331. flock/webapp/templates/partials/_load_manager_view.html +0 -49
  332. flock/webapp/templates/partials/_registry_table.html +0 -25
  333. flock/webapp/templates/partials/_registry_viewer_content.html +0 -70
  334. flock/webapp/templates/partials/_results_display.html +0 -78
  335. flock/webapp/templates/partials/_settings_env_content.html +0 -9
  336. flock/webapp/templates/partials/_settings_theme_content.html +0 -14
  337. flock/webapp/templates/partials/_settings_view.html +0 -36
  338. flock/webapp/templates/partials/_share_chat_link_snippet.html +0 -11
  339. flock/webapp/templates/partials/_share_link_snippet.html +0 -35
  340. flock/webapp/templates/partials/_sidebar.html +0 -74
  341. flock/webapp/templates/partials/_streaming_results_container.html +0 -195
  342. flock/webapp/templates/partials/_structured_data_view.html +0 -40
  343. flock/webapp/templates/partials/_theme_preview.html +0 -36
  344. flock/webapp/templates/registry_viewer.html +0 -84
  345. flock/webapp/templates/shared_run_page.html +0 -140
  346. flock/workflow/__init__.py +0 -0
  347. flock/workflow/activities.py +0 -196
  348. flock/workflow/agent_activities.py +0 -24
  349. flock/workflow/agent_execution_activity.py +0 -202
  350. flock/workflow/flock_workflow.py +0 -214
  351. flock/workflow/temporal_config.py +0 -96
  352. flock/workflow/temporal_setup.py +0 -68
  353. flock_core-0.5.0b28.dist-info/METADATA +0 -274
  354. flock_core-0.5.0b28.dist-info/RECORD +0 -561
  355. flock_core-0.5.0b28.dist-info/entry_points.txt +0 -2
  356. /flock/{core/logging → logging}/formatters/themes.py +0 -0
  357. /flock/{core/logging → logging}/span_middleware/baggage_span_processor.py +0 -0
  358. /flock/{core/mcp → mcp}/util/__init__.py +0 -0
  359. {flock_core-0.5.0b28.dist-info → flock_core-0.5.56b0.dist-info}/WHEEL +0 -0
@@ -1,337 +0,0 @@
1
- # src/flock/webapp/app/services/flock_service.py
2
- from typing import TYPE_CHECKING, Any, Optional
3
-
4
- import yaml
5
-
6
- # Conditional import for Flock and FlockFactory for type hinting
7
- if TYPE_CHECKING:
8
- from flock.core.flock import Flock
9
- from flock.core.flock_factory import FlockFactory
10
- else:
11
- Flock = "flock.core.flock.Flock"
12
- FlockFactory = "flock.core.flock_factory.FlockFactory"
13
-
14
- from flock.core.api.run_store import RunStore
15
- from flock.core.logging.logging import get_logger
16
- from flock.core.registry import get_registry
17
- from flock.webapp.app.config import FLOCK_FILES_DIR
18
- from flock.webapp.app.dependencies import set_global_flock_services
19
-
20
- logger = get_logger("webapp.service")
21
-
22
-
23
- def get_available_flock_files() -> list[str]:
24
- """Returns a sorted list of available .yaml, .yml, or .flock files."""
25
- if not FLOCK_FILES_DIR.exists():
26
- return []
27
- return sorted(
28
- [
29
- f.name
30
- for f in FLOCK_FILES_DIR.iterdir()
31
- if f.is_file() and (f.suffix.lower() in [".yaml", ".yml", ".flock"])
32
- ]
33
- )
34
-
35
- def _ensure_run_store_in_app_state(app_state: object) -> RunStore: # app_state is Starlette's State
36
- """Ensures a RunStore instance exists in app_state, creating if necessary."""
37
- run_store = getattr(app_state, 'run_store', None)
38
- if not isinstance(run_store, RunStore):
39
- logger.info("RunStore not found or invalid in app_state, creating a new one for this session.")
40
- run_store = RunStore()
41
- setattr(app_state, 'run_store', run_store)
42
- return run_store
43
-
44
-
45
- def load_flock_from_file_service(filename: str, app_state: object) -> Optional["Flock"]:
46
- """Loads a Flock instance from a file.
47
- Updates app_state with the loaded flock/filename and updates DI services.
48
- """
49
- from flock.core.flock import Flock as ConcreteFlock
50
-
51
- file_path = FLOCK_FILES_DIR / filename
52
- if not file_path.exists():
53
- logger.error(f"Flock file not found: {file_path}")
54
- clear_current_flock_service(app_state)
55
- return None
56
- try:
57
- logger.info(f"Loading flock from: {file_path}")
58
- loaded_flock = ConcreteFlock.load_from_file(str(file_path))
59
-
60
- setattr(app_state, 'flock_instance', loaded_flock)
61
- setattr(app_state, 'flock_filename', filename)
62
- run_store = _ensure_run_store_in_app_state(app_state)
63
- set_global_flock_services(loaded_flock, run_store)
64
-
65
- logger.info(f"Service: Successfully loaded flock '{loaded_flock.name}' from '{filename}'. DI updated.")
66
- return loaded_flock
67
- except Exception as e:
68
- logger.error(f"Service: Error loading flock from {file_path}: {e}", exc_info=True)
69
- clear_current_flock_service(app_state)
70
- return None
71
-
72
-
73
- def create_new_flock_service(
74
- name: str, model: str | None, description: str | None, app_state: object
75
- ) -> "Flock":
76
- """Creates a new Flock instance.
77
- Updates app_state and DI services.
78
- """
79
- from flock.core.flock import Flock as ConcreteFlock
80
-
81
- effective_model = model.strip() if model and model.strip() else None
82
- new_flock = ConcreteFlock(
83
- name=name,
84
- model=effective_model,
85
- description=description,
86
- show_flock_banner=False,
87
- )
88
- default_filename = f"{name.replace(' ', '_').lower()}.flock.yaml"
89
-
90
- setattr(app_state, 'flock_instance', new_flock)
91
- setattr(app_state, 'flock_filename', default_filename)
92
- run_store = _ensure_run_store_in_app_state(app_state)
93
- set_global_flock_services(new_flock, run_store)
94
-
95
- logger.info(f"Service: Created new flock '{name}'. DI updated. Default filename: '{default_filename}'.")
96
- return new_flock
97
-
98
-
99
- def clear_current_flock_service(app_state: object):
100
- """Clears the current Flock from app_state and updates DI services.
101
- """
102
- if hasattr(app_state, 'flock_instance'):
103
- delattr(app_state, 'flock_instance')
104
- if hasattr(app_state, 'flock_filename'):
105
- delattr(app_state, 'flock_filename')
106
-
107
- run_store = _ensure_run_store_in_app_state(app_state)
108
- set_global_flock_services(None, run_store)
109
- logger.info("Service: Current flock cleared from app_state. DI updated (Flock is None).")
110
-
111
-
112
- def save_current_flock_to_file_service(new_filename: str, app_state: object) -> tuple[bool, str]:
113
- """Saves the Flock from app_state to a file. Updates app_state's current filename on success.
114
- """
115
- current_flock: Flock | None = getattr(app_state, 'flock_instance', None)
116
- if not current_flock:
117
- return False, "No flock loaded to save."
118
- if not new_filename.strip():
119
- return False, "Filename cannot be empty."
120
-
121
- save_path = FLOCK_FILES_DIR / new_filename
122
- try:
123
- current_flock.to_yaml_file(str(save_path))
124
- setattr(app_state, 'flock_filename', new_filename) # Update filename in app_state
125
- logger.info(f"Service: Flock '{current_flock.name}' saved to '{new_filename}'.")
126
- return True, f"Flock saved to '{new_filename}'."
127
- except Exception as e:
128
- logger.error(f"Service: Error saving flock '{current_flock.name}' to {save_path}: {e}", exc_info=True)
129
- return False, f"Error saving flock: {e}"
130
-
131
-
132
- def update_flock_properties_service(
133
- name: str, model: str | None, description: str | None, app_state: object
134
- ) -> bool:
135
- """Updates properties of the Flock in app_state. Updates app_state's filename if name changes.
136
- """
137
- current_flock: Flock | None = getattr(app_state, 'flock_instance', None)
138
- current_filename: str | None = getattr(app_state, 'flock_filename', None)
139
-
140
- if not current_flock:
141
- logger.warning("Service: Attempted to update properties, but no flock loaded.")
142
- return False
143
-
144
- old_name = current_flock.name
145
- old_name_default_filename = f"{old_name.replace(' ', '_').lower()}.flock.yaml"
146
-
147
- current_flock.name = name
148
- current_flock.model = model.strip() if model and model.strip() else None
149
- current_flock.description = description
150
-
151
- if current_filename == old_name_default_filename and old_name != name:
152
- new_default_filename = f"{name.replace(' ', '_').lower()}.flock.yaml"
153
- setattr(app_state, 'flock_filename', new_default_filename)
154
- logger.info(f"Service: Default filename updated to '{new_default_filename}' due to flock name change.")
155
-
156
- logger.info(f"Service: Flock properties updated for '{name}'.")
157
- return True
158
-
159
-
160
- def add_agent_to_current_flock_service(agent_config: dict, app_state: object) -> bool:
161
- """Adds an agent to the Flock in app_state."""
162
- from flock.core.flock_factory import FlockFactory as ConcreteFlockFactory
163
-
164
- current_flock: Flock | None = getattr(app_state, 'flock_instance', None)
165
- if not current_flock:
166
- logger.warning("Service: Cannot add agent, no flock loaded.")
167
- return False
168
-
169
- registry = get_registry()
170
- tools_instances = []
171
- if agent_config.get("tools_names"):
172
- for tool_name in agent_config["tools_names"]:
173
- try: tools_instances.append(registry.get_callable(tool_name))
174
- except KeyError: logger.warning(f"Service: Tool '{tool_name}' not found for agent '{agent_config['name']}'.")
175
- try:
176
- agent = ConcreteFlockFactory.create_default_agent(
177
- name=agent_config["name"],
178
- description=agent_config.get("description"),
179
- model=agent_config.get("model"),
180
- input=agent_config["input"],
181
- output=agent_config["output"],
182
- tools=tools_instances or None,
183
- )
184
- current_flock.add_agent(agent)
185
- logger.info(f"Service: Agent '{agent.name}' added to flock '{current_flock.name}'.")
186
- return True
187
- except Exception as e:
188
- logger.error(f"Service: Error adding agent to flock '{current_flock.name}': {e}", exc_info=True)
189
- return False
190
-
191
-
192
- def update_agent_in_current_flock_service(
193
- original_agent_name: str, agent_config: dict, app_state: object
194
- ) -> bool:
195
- """Updates an agent in the Flock in app_state."""
196
- current_flock: Flock | None = getattr(app_state, 'flock_instance', None)
197
- if not current_flock:
198
- logger.warning("Service: Cannot update agent, no flock loaded.")
199
- return False
200
-
201
- agent_to_update = current_flock.agents.get(original_agent_name)
202
- if not agent_to_update:
203
- logger.warning(f"Service: Agent '{original_agent_name}' not found in flock '{current_flock.name}' for update.")
204
- return False
205
-
206
- registry = get_registry()
207
- tools_instances = []
208
- if agent_config.get("tools_names"):
209
- for tool_name in agent_config["tools_names"]:
210
- try: tools_instances.append(registry.get_callable(tool_name))
211
- except KeyError: logger.warning(f"Service: Tool '{tool_name}' not found during agent update.")
212
- try:
213
- new_name = agent_config["name"]
214
- agent_to_update.description = agent_config.get("description")
215
- current_agent_model = agent_config.get("model")
216
- agent_to_update.model = current_agent_model if current_agent_model and current_agent_model.strip() else None
217
- agent_to_update.input = agent_config["input"]
218
- agent_to_update.output = agent_config["output"]
219
- agent_to_update.tools = tools_instances or []
220
-
221
- if original_agent_name != new_name:
222
- current_flock._agents.pop(original_agent_name)
223
- agent_to_update.name = new_name
224
- current_flock.add_agent(agent_to_update)
225
- logger.info(f"Service: Agent '{original_agent_name}' renamed to '{new_name}' and updated in flock '{current_flock.name}'.")
226
- else:
227
- logger.info(f"Service: Agent '{original_agent_name}' updated in flock '{current_flock.name}'.")
228
- return True
229
- except Exception as e:
230
- logger.error(f"Service: Error updating agent '{original_agent_name}' in flock '{current_flock.name}': {e}", exc_info=True)
231
- return False
232
-
233
-
234
- def remove_agent_from_current_flock_service(agent_name: str, app_state: object) -> bool:
235
- """Removes an agent from the Flock in app_state."""
236
- current_flock: Flock | None = getattr(app_state, 'flock_instance', None)
237
- if not current_flock or agent_name not in current_flock.agents:
238
- logger.warning(f"Service: Cannot remove agent '{agent_name}', no flock loaded or agent not found.")
239
- return False
240
- try:
241
- del current_flock._agents[agent_name]
242
- logger.info(f"Service: Agent '{agent_name}' removed from flock '{current_flock.name}'.")
243
- return True
244
- except Exception as e:
245
- logger.error(f"Service: Error removing agent '{agent_name}' from flock '{current_flock.name}': {e}", exc_info=True)
246
- return False
247
-
248
-
249
- async def run_current_flock_service(
250
- start_agent_name: str,
251
- inputs: dict[str, Any],
252
- app_state: Any,
253
- ) -> dict[str, Any]:
254
- """Runs the specified agent from the current flock instance in app_state."""
255
- logger.info(f"Attempting to run agent: {start_agent_name} using flock from app_state.")
256
-
257
- current_flock: Flock | None = getattr(app_state, "flock_instance", None)
258
- run_store: RunStore | None = getattr(app_state, "run_store", None)
259
-
260
- if not current_flock:
261
- logger.error("Run service: No Flock instance available in app_state.")
262
- return {"error": "No Flock loaded in the application."}
263
- if not run_store:
264
- logger.error("Run service: No RunStore instance available in app_state.")
265
- # Attempt to initialize a default run_store if missing and this service is critical
266
- # This might indicate an issue in the application lifecycle setup
267
- logger.warning("Run service: Initializing a default RunStore as none was found in app_state.")
268
- run_store = RunStore()
269
- setattr(app_state, "run_store", run_store)
270
- # Also update global DI if this is how it's managed elsewhere for consistency,
271
- # though ideally DI setup handles this more centrally.
272
- # from flock.webapp.app.dependencies import set_global_flock_services
273
- # set_global_flock_services(current_flock, run_store)
274
-
275
-
276
- if start_agent_name not in current_flock.agents:
277
- logger.error(f"Run service: Agent '{start_agent_name}' not found in current flock '{current_flock.name}'.")
278
- return {"error": f"Agent '{start_agent_name}' not found."}
279
-
280
- try:
281
- logger.info(f"Executing agent '{start_agent_name}' from flock '{current_flock.name}' using app_state.")
282
- # Direct execution using the flock from app_state
283
- result = await current_flock.run_async(
284
- agent=start_agent_name, input=inputs, box_result=False
285
- )
286
- # Store run details using the run_store from app_state
287
- if hasattr(run_store, "add_run_details"): # Check if RunStore has this method
288
- run_id = result.get("run_id", "unknown_run_id") # Assuming run_async result might contain run_id
289
- run_store.add_run_details(run_id=run_id, agent_name=start_agent_name, inputs=inputs, outputs=result)
290
- return result
291
- except Exception as e:
292
- logger.error(f"Run service: Error during agent execution: {e}", exc_info=True)
293
- return {"error": f"An error occurred: {e}"}
294
-
295
-
296
- def get_registered_items_service(item_type: str) -> list[dict]:
297
- """Retrieves items of a specific type from the global FlockRegistry."""
298
- registry = get_registry()
299
- items_dict: dict | None = None
300
- if item_type == "type": items_dict = registry._types
301
- elif item_type == "tool": items_dict = registry._callables
302
- elif item_type == "component": items_dict = registry._components
303
- else: return []
304
-
305
- if items_dict is None: return []
306
-
307
- items = []
308
- for name, item_obj in items_dict.items():
309
- module_path = "N/A"
310
- try: module_path = item_obj.__module__
311
- except AttributeError: pass
312
- items.append({"name": name, "module": module_path})
313
- return sorted(items, key=lambda x: x["name"])
314
-
315
-
316
- def get_flock_preview_service(filename: str) -> dict | None:
317
- """Loads basic properties of a flock file for preview."""
318
- file_path = FLOCK_FILES_DIR / filename
319
- if not file_path.exists():
320
- logger.warning(f"Service: Preview failed, file not found {file_path}")
321
- return None
322
- try:
323
- with file_path.open("r", encoding="utf-8") as f:
324
- data = yaml.safe_load(f)
325
- if isinstance(data, dict):
326
- return {
327
- "name": data.get("name", filename),
328
- "model": data.get("model"),
329
- "description": data.get("description"),
330
- "agents_count": len(data.get("agents", {})),
331
- "enable_temporal": data.get("enable_temporal", False)
332
- }
333
- logger.warning(f"Service: Preview failed, '{filename}' is not a valid Flock YAML (not a dict).")
334
- return {"name": filename, "error": "Not a valid Flock YAML structure"}
335
- except Exception as e:
336
- logger.error(f"Service: Error getting flock preview for {filename}: {e}", exc_info=True)
337
- return {"name": filename, "error": str(e)}
@@ -1,81 +0,0 @@
1
- from datetime import datetime
2
-
3
- from pydantic import BaseModel, Field
4
-
5
-
6
- class SharedLinkConfig(BaseModel):
7
- """Configuration for a shared Flock agent execution link or chat session."""
8
-
9
- share_id: str = Field(..., description="Unique identifier for the shared link.")
10
- agent_name: str = Field(..., description="The name of the agent being shared (for run) or the chat agent (for chat).")
11
- flock_definition: str = Field(..., description="The YAML/JSON string definition of the Flock the agent belongs to.")
12
- created_at: datetime = Field(
13
- default_factory=datetime.utcnow, description="Timestamp of when the link was created."
14
- )
15
- share_type: str = Field(default="agent_run", description="Type of share: 'agent_run' or 'chat'")
16
-
17
- # Chat-specific settings (only relevant if share_type is 'chat')
18
- chat_message_key: str | None = Field(None, description="Message key for chat input mapping.")
19
- chat_history_key: str | None = Field(None, description="History key for chat input mapping.")
20
- chat_response_key: str | None = Field(None, description="Response key for chat output mapping.")
21
-
22
- # Placeholder for future enhancement: pre-filled input values
23
- # input_values: Optional[Dict[str, Any]] = Field(
24
- # None, description="Optional pre-filled input values for the agent."
25
- # )
26
-
27
- model_config = {
28
- "from_attributes": True,
29
- "json_schema_extra": {
30
- "examples": [
31
- {
32
- "share_id": "abcdef123456",
33
- "agent_name": "MyChatAgent",
34
- "flock_definition": "name: MySharedFlock\nagents:\n MyChatAgent:\n input: 'message: str'\n output: 'response: str'\n # ... rest of flock YAML ...",
35
- "created_at": "2023-10-26T10:00:00Z",
36
- "share_type": "chat",
37
- "chat_message_key": "user_input",
38
- "chat_history_key": "conversation_history",
39
- "chat_response_key": "agent_output"
40
- }
41
- ]
42
- }
43
- }
44
-
45
- # -----------------------------------------------------------
46
- # Feedback model (user ratings / corrections)
47
- # -----------------------------------------------------------
48
-
49
-
50
- class FeedbackRecord(BaseModel):
51
- """A user-submitted piece of feedback for an agent run or chat turn."""
52
-
53
- feedback_id: str = Field(..., description="Unique identifier for this feedback entry.")
54
- share_id: str | None = Field(None, description="If the feedback refers to a shared link, its ID; otherwise None.")
55
- context_type: str = Field(
56
- default="agent_run",
57
- description="Where the feedback originates (agent_run | chat | other)",
58
- )
59
- reason: str = Field(..., description="User-supplied reason / comment.")
60
- expected_response: str | None = Field(
61
- None,
62
- description="Desired or corrected response in JSON or plain text.",
63
- )
64
- actual_response: str | None = Field(
65
- None,
66
- description="Original response shown to the user (optional, for context).",
67
- )
68
- created_at: datetime = Field(default_factory=datetime.utcnow)
69
- # When share_id is None store explicit target identifiers
70
- flock_name: str | None = Field(
71
- None,
72
- description="Name of the flock that produced the result when no share_id is used.",
73
- )
74
- agent_name: str | None = Field(
75
- None,
76
- description="Name of the agent or chat involved in the feedback.",
77
- )
78
- flock_definition: str | None = Field(
79
- None,
80
- description="Full YAML definition of the flock when feedback was submitted.",
81
- )