flock-core 0.5.0b27__py3-none-any.whl → 0.5.0b50__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 (357) 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.0b50.dist-info/METADATA +747 -0
  162. flock_core-0.5.0b50.dist-info/RECORD +398 -0
  163. flock_core-0.5.0b50.dist-info/entry_points.txt +2 -0
  164. {flock_core-0.5.0b27.dist-info → flock_core-0.5.0b50.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 -15
  197. flock/components/utility/memory_utility_component.py +0 -550
  198. flock/components/utility/metrics_utility_component.py +0 -700
  199. flock/config.py +0 -61
  200. flock/core/__init__.py +0 -110
  201. flock/core/agent/__init__.py +0 -16
  202. flock/core/agent/default_agent.py +0 -180
  203. flock/core/agent/flock_agent_components.py +0 -104
  204. flock/core/agent/flock_agent_execution.py +0 -101
  205. flock/core/agent/flock_agent_integration.py +0 -260
  206. flock/core/agent/flock_agent_lifecycle.py +0 -186
  207. flock/core/agent/flock_agent_serialization.py +0 -381
  208. flock/core/api/__init__.py +0 -10
  209. flock/core/api/custom_endpoint.py +0 -45
  210. flock/core/api/endpoints.py +0 -254
  211. flock/core/api/main.py +0 -162
  212. flock/core/api/models.py +0 -97
  213. flock/core/api/run_store.py +0 -224
  214. flock/core/api/runner.py +0 -44
  215. flock/core/api/service.py +0 -214
  216. flock/core/component/__init__.py +0 -15
  217. flock/core/component/agent_component_base.py +0 -309
  218. flock/core/component/evaluation_component.py +0 -62
  219. flock/core/component/routing_component.py +0 -74
  220. flock/core/component/utility_component.py +0 -69
  221. flock/core/config/flock_agent_config.py +0 -58
  222. flock/core/config/scheduled_agent_config.py +0 -40
  223. flock/core/context/context.py +0 -213
  224. flock/core/context/context_manager.py +0 -37
  225. flock/core/context/context_vars.py +0 -10
  226. flock/core/evaluation/utils.py +0 -396
  227. flock/core/execution/batch_executor.py +0 -369
  228. flock/core/execution/evaluation_executor.py +0 -438
  229. flock/core/execution/local_executor.py +0 -31
  230. flock/core/execution/opik_executor.py +0 -103
  231. flock/core/execution/temporal_executor.py +0 -164
  232. flock/core/flock.py +0 -634
  233. flock/core/flock_agent.py +0 -336
  234. flock/core/flock_factory.py +0 -551
  235. flock/core/flock_scheduler.py +0 -166
  236. flock/core/flock_server_manager.py +0 -136
  237. flock/core/interpreter/python_interpreter.py +0 -689
  238. flock/core/mcp/__init__.py +0 -1
  239. flock/core/mcp/flock_mcp_server.py +0 -680
  240. flock/core/mcp/mcp_client_manager.py +0 -201
  241. flock/core/mcp/types/__init__.py +0 -1
  242. flock/core/mixin/dspy_integration.py +0 -403
  243. flock/core/mixin/prompt_parser.py +0 -125
  244. flock/core/orchestration/__init__.py +0 -15
  245. flock/core/orchestration/flock_batch_processor.py +0 -94
  246. flock/core/orchestration/flock_evaluator.py +0 -113
  247. flock/core/orchestration/flock_execution.py +0 -295
  248. flock/core/orchestration/flock_initialization.py +0 -149
  249. flock/core/orchestration/flock_server_manager.py +0 -67
  250. flock/core/orchestration/flock_web_server.py +0 -117
  251. flock/core/registry/__init__.py +0 -45
  252. flock/core/registry/agent_registry.py +0 -69
  253. flock/core/registry/callable_registry.py +0 -139
  254. flock/core/registry/component_discovery.py +0 -142
  255. flock/core/registry/component_registry.py +0 -64
  256. flock/core/registry/config_mapping.py +0 -64
  257. flock/core/registry/decorators.py +0 -137
  258. flock/core/registry/registry_hub.py +0 -205
  259. flock/core/registry/server_registry.py +0 -57
  260. flock/core/registry/type_registry.py +0 -86
  261. flock/core/serialization/__init__.py +0 -13
  262. flock/core/serialization/callable_registry.py +0 -52
  263. flock/core/serialization/flock_serializer.py +0 -832
  264. flock/core/serialization/json_encoder.py +0 -41
  265. flock/core/serialization/secure_serializer.py +0 -175
  266. flock/core/serialization/serializable.py +0 -342
  267. flock/core/serialization/serialization_utils.py +0 -412
  268. flock/core/util/file_path_utils.py +0 -223
  269. flock/core/util/hydrator.py +0 -309
  270. flock/core/util/input_resolver.py +0 -164
  271. flock/core/util/loader.py +0 -59
  272. flock/core/util/splitter.py +0 -219
  273. flock/di.py +0 -27
  274. flock/platform/docker_tools.py +0 -49
  275. flock/platform/jaeger_install.py +0 -86
  276. flock/webapp/__init__.py +0 -1
  277. flock/webapp/app/__init__.py +0 -0
  278. flock/webapp/app/api/__init__.py +0 -0
  279. flock/webapp/app/api/agent_management.py +0 -241
  280. flock/webapp/app/api/execution.py +0 -709
  281. flock/webapp/app/api/flock_management.py +0 -129
  282. flock/webapp/app/api/registry_viewer.py +0 -30
  283. flock/webapp/app/chat.py +0 -665
  284. flock/webapp/app/config.py +0 -104
  285. flock/webapp/app/dependencies.py +0 -117
  286. flock/webapp/app/main.py +0 -1070
  287. flock/webapp/app/middleware.py +0 -113
  288. flock/webapp/app/models_ui.py +0 -7
  289. flock/webapp/app/services/__init__.py +0 -0
  290. flock/webapp/app/services/feedback_file_service.py +0 -363
  291. flock/webapp/app/services/flock_service.py +0 -337
  292. flock/webapp/app/services/sharing_models.py +0 -81
  293. flock/webapp/app/services/sharing_store.py +0 -598
  294. flock/webapp/app/templates/theme_mapper.html +0 -326
  295. flock/webapp/app/theme_mapper.py +0 -812
  296. flock/webapp/app/utils.py +0 -85
  297. flock/webapp/run.py +0 -215
  298. flock/webapp/static/css/chat.css +0 -301
  299. flock/webapp/static/css/components.css +0 -167
  300. flock/webapp/static/css/header.css +0 -39
  301. flock/webapp/static/css/layout.css +0 -46
  302. flock/webapp/static/css/sidebar.css +0 -127
  303. flock/webapp/static/css/two-pane.css +0 -48
  304. flock/webapp/templates/base.html +0 -200
  305. flock/webapp/templates/chat.html +0 -152
  306. flock/webapp/templates/chat_settings.html +0 -19
  307. flock/webapp/templates/flock_editor.html +0 -16
  308. flock/webapp/templates/index.html +0 -12
  309. flock/webapp/templates/partials/_agent_detail_form.html +0 -93
  310. flock/webapp/templates/partials/_agent_list.html +0 -18
  311. flock/webapp/templates/partials/_agent_manager_view.html +0 -51
  312. flock/webapp/templates/partials/_agent_tools_checklist.html +0 -14
  313. flock/webapp/templates/partials/_chat_container.html +0 -15
  314. flock/webapp/templates/partials/_chat_messages.html +0 -57
  315. flock/webapp/templates/partials/_chat_settings_form.html +0 -85
  316. flock/webapp/templates/partials/_create_flock_form.html +0 -50
  317. flock/webapp/templates/partials/_dashboard_flock_detail.html +0 -17
  318. flock/webapp/templates/partials/_dashboard_flock_file_list.html +0 -16
  319. flock/webapp/templates/partials/_dashboard_flock_properties_preview.html +0 -28
  320. flock/webapp/templates/partials/_dashboard_upload_flock_form.html +0 -16
  321. flock/webapp/templates/partials/_dynamic_input_form_content.html +0 -22
  322. flock/webapp/templates/partials/_env_vars_table.html +0 -23
  323. flock/webapp/templates/partials/_execution_form.html +0 -118
  324. flock/webapp/templates/partials/_execution_view_container.html +0 -28
  325. flock/webapp/templates/partials/_flock_file_list.html +0 -23
  326. flock/webapp/templates/partials/_flock_properties_form.html +0 -52
  327. flock/webapp/templates/partials/_flock_upload_form.html +0 -16
  328. flock/webapp/templates/partials/_header_flock_status.html +0 -5
  329. flock/webapp/templates/partials/_load_manager_view.html +0 -49
  330. flock/webapp/templates/partials/_registry_table.html +0 -25
  331. flock/webapp/templates/partials/_registry_viewer_content.html +0 -70
  332. flock/webapp/templates/partials/_results_display.html +0 -78
  333. flock/webapp/templates/partials/_settings_env_content.html +0 -9
  334. flock/webapp/templates/partials/_settings_theme_content.html +0 -14
  335. flock/webapp/templates/partials/_settings_view.html +0 -36
  336. flock/webapp/templates/partials/_share_chat_link_snippet.html +0 -11
  337. flock/webapp/templates/partials/_share_link_snippet.html +0 -35
  338. flock/webapp/templates/partials/_sidebar.html +0 -74
  339. flock/webapp/templates/partials/_streaming_results_container.html +0 -195
  340. flock/webapp/templates/partials/_structured_data_view.html +0 -40
  341. flock/webapp/templates/partials/_theme_preview.html +0 -36
  342. flock/webapp/templates/registry_viewer.html +0 -84
  343. flock/webapp/templates/shared_run_page.html +0 -140
  344. flock/workflow/__init__.py +0 -0
  345. flock/workflow/activities.py +0 -196
  346. flock/workflow/agent_activities.py +0 -24
  347. flock/workflow/agent_execution_activity.py +0 -202
  348. flock/workflow/flock_workflow.py +0 -214
  349. flock/workflow/temporal_config.py +0 -96
  350. flock/workflow/temporal_setup.py +0 -68
  351. flock_core-0.5.0b27.dist-info/METADATA +0 -274
  352. flock_core-0.5.0b27.dist-info/RECORD +0 -559
  353. flock_core-0.5.0b27.dist-info/entry_points.txt +0 -2
  354. /flock/{core/logging → logging}/formatters/themes.py +0 -0
  355. /flock/{core/logging → logging}/span_middleware/baggage_span_processor.py +0 -0
  356. /flock/{core/mcp → mcp}/util/__init__.py +0 -0
  357. {flock_core-0.5.0b27.dist-info → flock_core-0.5.0b50.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
- )