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.
- flock/__init__.py +12 -217
- flock/agent.py +678 -0
- flock/api/themes.py +71 -0
- flock/artifacts.py +79 -0
- flock/cli.py +75 -0
- flock/components.py +173 -0
- flock/dashboard/__init__.py +28 -0
- flock/dashboard/collector.py +283 -0
- flock/dashboard/events.py +182 -0
- flock/dashboard/launcher.py +230 -0
- flock/dashboard/service.py +537 -0
- flock/dashboard/websocket.py +235 -0
- flock/engines/__init__.py +6 -0
- flock/engines/dspy_engine.py +856 -0
- flock/examples.py +128 -0
- flock/{core/util → helper}/cli_helper.py +4 -3
- flock/{core/logging → logging}/__init__.py +2 -3
- flock/{core/logging → logging}/formatters/enum_builder.py +3 -4
- flock/{core/logging → logging}/formatters/theme_builder.py +19 -44
- flock/{core/logging → logging}/formatters/themed_formatter.py +69 -115
- flock/{core/logging → logging}/logging.py +77 -61
- flock/{core/logging → logging}/telemetry.py +20 -26
- flock/{core/logging → logging}/telemetry_exporter/base_exporter.py +2 -2
- flock/{core/logging → logging}/telemetry_exporter/file_exporter.py +6 -9
- flock/{core/logging → logging}/telemetry_exporter/sqlite_exporter.py +2 -3
- flock/{core/logging → logging}/trace_and_logged.py +20 -24
- flock/mcp/__init__.py +91 -0
- flock/{core/mcp/mcp_client.py → mcp/client.py} +103 -154
- flock/{core/mcp/mcp_config.py → mcp/config.py} +62 -117
- flock/mcp/manager.py +255 -0
- flock/mcp/servers/sse/__init__.py +1 -1
- flock/mcp/servers/sse/flock_sse_server.py +11 -53
- flock/mcp/servers/stdio/__init__.py +1 -1
- flock/mcp/servers/stdio/flock_stdio_server.py +8 -48
- flock/mcp/servers/streamable_http/flock_streamable_http_server.py +17 -62
- flock/mcp/servers/websockets/flock_websocket_server.py +7 -40
- flock/{core/mcp/flock_mcp_tool.py → mcp/tool.py} +16 -26
- flock/mcp/types/__init__.py +42 -0
- flock/{core/mcp → mcp}/types/callbacks.py +9 -15
- flock/{core/mcp → mcp}/types/factories.py +7 -6
- flock/{core/mcp → mcp}/types/handlers.py +13 -18
- flock/{core/mcp → mcp}/types/types.py +70 -74
- flock/{core/mcp → mcp}/util/helpers.py +1 -1
- flock/orchestrator.py +645 -0
- flock/registry.py +148 -0
- flock/runtime.py +262 -0
- flock/service.py +140 -0
- flock/store.py +69 -0
- flock/subscription.py +111 -0
- flock/themes/andromeda.toml +1 -1
- flock/themes/apple-system-colors.toml +1 -1
- flock/themes/arcoiris.toml +1 -1
- flock/themes/atomonelight.toml +1 -1
- flock/themes/ayu copy.toml +1 -1
- flock/themes/ayu-light.toml +1 -1
- flock/themes/belafonte-day.toml +1 -1
- flock/themes/belafonte-night.toml +1 -1
- flock/themes/blulocodark.toml +1 -1
- flock/themes/breeze.toml +1 -1
- flock/themes/broadcast.toml +1 -1
- flock/themes/brogrammer.toml +1 -1
- flock/themes/builtin-dark.toml +1 -1
- flock/themes/builtin-pastel-dark.toml +1 -1
- flock/themes/catppuccin-latte.toml +1 -1
- flock/themes/catppuccin-macchiato.toml +1 -1
- flock/themes/catppuccin-mocha.toml +1 -1
- flock/themes/cga.toml +1 -1
- flock/themes/chalk.toml +1 -1
- flock/themes/ciapre.toml +1 -1
- flock/themes/coffee-theme.toml +1 -1
- flock/themes/cyberpunkscarletprotocol.toml +1 -1
- flock/themes/dark+.toml +1 -1
- flock/themes/darkermatrix.toml +1 -1
- flock/themes/darkside.toml +1 -1
- flock/themes/desert.toml +1 -1
- flock/themes/django.toml +1 -1
- flock/themes/djangosmooth.toml +1 -1
- flock/themes/doomone.toml +1 -1
- flock/themes/dotgov.toml +1 -1
- flock/themes/dracula+.toml +1 -1
- flock/themes/duckbones.toml +1 -1
- flock/themes/encom.toml +1 -1
- flock/themes/espresso.toml +1 -1
- flock/themes/everblush.toml +1 -1
- flock/themes/fairyfloss.toml +1 -1
- flock/themes/fideloper.toml +1 -1
- flock/themes/fishtank.toml +1 -1
- flock/themes/flexoki-light.toml +1 -1
- flock/themes/floraverse.toml +1 -1
- flock/themes/framer.toml +1 -1
- flock/themes/galizur.toml +1 -1
- flock/themes/github.toml +1 -1
- flock/themes/grass.toml +1 -1
- flock/themes/grey-green.toml +1 -1
- flock/themes/gruvboxlight.toml +1 -1
- flock/themes/guezwhoz.toml +1 -1
- flock/themes/harper.toml +1 -1
- flock/themes/hax0r-blue.toml +1 -1
- flock/themes/hopscotch.256.toml +1 -1
- flock/themes/ic-green-ppl.toml +1 -1
- flock/themes/iceberg-dark.toml +1 -1
- flock/themes/japanesque.toml +1 -1
- flock/themes/jubi.toml +1 -1
- flock/themes/kibble.toml +1 -1
- flock/themes/kolorit.toml +1 -1
- flock/themes/kurokula.toml +1 -1
- flock/themes/materialdesigncolors.toml +1 -1
- flock/themes/matrix.toml +1 -1
- flock/themes/mellifluous.toml +1 -1
- flock/themes/midnight-in-mojave.toml +1 -1
- flock/themes/monokai-remastered.toml +1 -1
- flock/themes/monokai-soda.toml +1 -1
- flock/themes/neon.toml +1 -1
- flock/themes/neopolitan.toml +1 -1
- flock/themes/nord-light.toml +1 -1
- flock/themes/ocean.toml +1 -1
- flock/themes/onehalfdark.toml +1 -1
- flock/themes/onehalflight.toml +1 -1
- flock/themes/palenighthc.toml +1 -1
- flock/themes/paulmillr.toml +1 -1
- flock/themes/pencildark.toml +1 -1
- flock/themes/pnevma.toml +1 -1
- flock/themes/purple-rain.toml +1 -1
- flock/themes/purplepeter.toml +1 -1
- flock/themes/raycast-dark.toml +1 -1
- flock/themes/red-sands.toml +1 -1
- flock/themes/relaxed.toml +1 -1
- flock/themes/retro.toml +1 -1
- flock/themes/rose-pine.toml +1 -1
- flock/themes/royal.toml +1 -1
- flock/themes/ryuuko.toml +1 -1
- flock/themes/sakura.toml +1 -1
- flock/themes/scarlet-protocol.toml +1 -1
- flock/themes/seoulbones-dark.toml +1 -1
- flock/themes/shades-of-purple.toml +1 -1
- flock/themes/smyck.toml +1 -1
- flock/themes/softserver.toml +1 -1
- flock/themes/solarized-darcula.toml +1 -1
- flock/themes/square.toml +1 -1
- flock/themes/sugarplum.toml +1 -1
- flock/themes/thayer-bright.toml +1 -1
- flock/themes/tokyonight.toml +1 -1
- flock/themes/tomorrow.toml +1 -1
- flock/themes/ubuntu.toml +1 -1
- flock/themes/ultradark.toml +1 -1
- flock/themes/ultraviolent.toml +1 -1
- flock/themes/unikitty.toml +1 -1
- flock/themes/urple.toml +1 -1
- flock/themes/vesper.toml +1 -1
- flock/themes/vimbones.toml +1 -1
- flock/themes/wildcherry.toml +1 -1
- flock/themes/wilmersdorf.toml +1 -1
- flock/themes/wryan.toml +1 -1
- flock/themes/xcodedarkhc.toml +1 -1
- flock/themes/xcodelight.toml +1 -1
- flock/themes/zenbones-light.toml +1 -1
- flock/themes/zenwritten-dark.toml +1 -1
- flock/utilities.py +301 -0
- flock/{components/utility → utility}/output_utility_component.py +68 -53
- flock/visibility.py +107 -0
- flock_core-0.5.56b0.dist-info/METADATA +747 -0
- flock_core-0.5.56b0.dist-info/RECORD +398 -0
- flock_core-0.5.56b0.dist-info/entry_points.txt +2 -0
- {flock_core-0.5.0b28.dist-info → flock_core-0.5.56b0.dist-info}/licenses/LICENSE +1 -1
- flock/adapter/__init__.py +0 -14
- flock/adapter/azure_adapter.py +0 -68
- flock/adapter/chroma_adapter.py +0 -73
- flock/adapter/faiss_adapter.py +0 -97
- flock/adapter/pinecone_adapter.py +0 -51
- flock/adapter/vector_base.py +0 -47
- flock/cli/assets/release_notes.md +0 -140
- flock/cli/config.py +0 -8
- flock/cli/constants.py +0 -36
- flock/cli/create_agent.py +0 -1
- flock/cli/create_flock.py +0 -280
- flock/cli/execute_flock.py +0 -620
- flock/cli/load_agent.py +0 -1
- flock/cli/load_examples.py +0 -1
- flock/cli/load_flock.py +0 -192
- flock/cli/load_release_notes.py +0 -20
- flock/cli/loaded_flock_cli.py +0 -254
- flock/cli/manage_agents.py +0 -459
- flock/cli/registry_management.py +0 -889
- flock/cli/runner.py +0 -41
- flock/cli/settings.py +0 -857
- flock/cli/utils.py +0 -135
- flock/cli/view_results.py +0 -29
- flock/cli/yaml_editor.py +0 -396
- flock/components/__init__.py +0 -30
- flock/components/evaluation/__init__.py +0 -9
- flock/components/evaluation/declarative_evaluation_component.py +0 -606
- flock/components/routing/__init__.py +0 -15
- flock/components/routing/conditional_routing_component.py +0 -494
- flock/components/routing/default_routing_component.py +0 -103
- flock/components/routing/llm_routing_component.py +0 -206
- flock/components/utility/__init__.py +0 -22
- flock/components/utility/example_utility_component.py +0 -250
- flock/components/utility/feedback_utility_component.py +0 -206
- flock/components/utility/memory_utility_component.py +0 -550
- flock/components/utility/metrics_utility_component.py +0 -700
- flock/config.py +0 -61
- flock/core/__init__.py +0 -110
- flock/core/agent/__init__.py +0 -16
- flock/core/agent/default_agent.py +0 -216
- flock/core/agent/flock_agent_components.py +0 -104
- flock/core/agent/flock_agent_execution.py +0 -101
- flock/core/agent/flock_agent_integration.py +0 -260
- flock/core/agent/flock_agent_lifecycle.py +0 -186
- flock/core/agent/flock_agent_serialization.py +0 -381
- flock/core/api/__init__.py +0 -10
- flock/core/api/custom_endpoint.py +0 -45
- flock/core/api/endpoints.py +0 -254
- flock/core/api/main.py +0 -162
- flock/core/api/models.py +0 -97
- flock/core/api/run_store.py +0 -224
- flock/core/api/runner.py +0 -44
- flock/core/api/service.py +0 -214
- flock/core/component/__init__.py +0 -15
- flock/core/component/agent_component_base.py +0 -309
- flock/core/component/evaluation_component.py +0 -62
- flock/core/component/routing_component.py +0 -74
- flock/core/component/utility_component.py +0 -69
- flock/core/config/flock_agent_config.py +0 -58
- flock/core/config/scheduled_agent_config.py +0 -40
- flock/core/context/context.py +0 -213
- flock/core/context/context_manager.py +0 -37
- flock/core/context/context_vars.py +0 -10
- flock/core/evaluation/utils.py +0 -396
- flock/core/execution/batch_executor.py +0 -369
- flock/core/execution/evaluation_executor.py +0 -438
- flock/core/execution/local_executor.py +0 -31
- flock/core/execution/opik_executor.py +0 -103
- flock/core/execution/temporal_executor.py +0 -164
- flock/core/flock.py +0 -634
- flock/core/flock_agent.py +0 -336
- flock/core/flock_factory.py +0 -613
- flock/core/flock_scheduler.py +0 -166
- flock/core/flock_server_manager.py +0 -136
- flock/core/interpreter/python_interpreter.py +0 -689
- flock/core/mcp/__init__.py +0 -1
- flock/core/mcp/flock_mcp_server.py +0 -680
- flock/core/mcp/mcp_client_manager.py +0 -201
- flock/core/mcp/types/__init__.py +0 -1
- flock/core/mixin/dspy_integration.py +0 -403
- flock/core/mixin/prompt_parser.py +0 -125
- flock/core/orchestration/__init__.py +0 -15
- flock/core/orchestration/flock_batch_processor.py +0 -94
- flock/core/orchestration/flock_evaluator.py +0 -113
- flock/core/orchestration/flock_execution.py +0 -295
- flock/core/orchestration/flock_initialization.py +0 -149
- flock/core/orchestration/flock_server_manager.py +0 -67
- flock/core/orchestration/flock_web_server.py +0 -117
- flock/core/registry/__init__.py +0 -45
- flock/core/registry/agent_registry.py +0 -69
- flock/core/registry/callable_registry.py +0 -139
- flock/core/registry/component_discovery.py +0 -142
- flock/core/registry/component_registry.py +0 -64
- flock/core/registry/config_mapping.py +0 -64
- flock/core/registry/decorators.py +0 -137
- flock/core/registry/registry_hub.py +0 -205
- flock/core/registry/server_registry.py +0 -57
- flock/core/registry/type_registry.py +0 -86
- flock/core/serialization/__init__.py +0 -13
- flock/core/serialization/callable_registry.py +0 -52
- flock/core/serialization/flock_serializer.py +0 -832
- flock/core/serialization/json_encoder.py +0 -41
- flock/core/serialization/secure_serializer.py +0 -175
- flock/core/serialization/serializable.py +0 -342
- flock/core/serialization/serialization_utils.py +0 -412
- flock/core/util/file_path_utils.py +0 -223
- flock/core/util/hydrator.py +0 -309
- flock/core/util/input_resolver.py +0 -164
- flock/core/util/loader.py +0 -59
- flock/core/util/splitter.py +0 -219
- flock/di.py +0 -27
- flock/platform/docker_tools.py +0 -49
- flock/platform/jaeger_install.py +0 -86
- flock/webapp/__init__.py +0 -1
- flock/webapp/app/__init__.py +0 -0
- flock/webapp/app/api/__init__.py +0 -0
- flock/webapp/app/api/agent_management.py +0 -241
- flock/webapp/app/api/execution.py +0 -709
- flock/webapp/app/api/flock_management.py +0 -129
- flock/webapp/app/api/registry_viewer.py +0 -30
- flock/webapp/app/chat.py +0 -665
- flock/webapp/app/config.py +0 -104
- flock/webapp/app/dependencies.py +0 -117
- flock/webapp/app/main.py +0 -1070
- flock/webapp/app/middleware.py +0 -113
- flock/webapp/app/models_ui.py +0 -7
- flock/webapp/app/services/__init__.py +0 -0
- flock/webapp/app/services/feedback_file_service.py +0 -363
- flock/webapp/app/services/flock_service.py +0 -337
- flock/webapp/app/services/sharing_models.py +0 -81
- flock/webapp/app/services/sharing_store.py +0 -762
- flock/webapp/app/templates/theme_mapper.html +0 -326
- flock/webapp/app/theme_mapper.py +0 -812
- flock/webapp/app/utils.py +0 -85
- flock/webapp/run.py +0 -215
- flock/webapp/static/css/chat.css +0 -301
- flock/webapp/static/css/components.css +0 -167
- flock/webapp/static/css/header.css +0 -39
- flock/webapp/static/css/layout.css +0 -46
- flock/webapp/static/css/sidebar.css +0 -127
- flock/webapp/static/css/two-pane.css +0 -48
- flock/webapp/templates/base.html +0 -200
- flock/webapp/templates/chat.html +0 -152
- flock/webapp/templates/chat_settings.html +0 -19
- flock/webapp/templates/flock_editor.html +0 -16
- flock/webapp/templates/index.html +0 -12
- flock/webapp/templates/partials/_agent_detail_form.html +0 -93
- flock/webapp/templates/partials/_agent_list.html +0 -18
- flock/webapp/templates/partials/_agent_manager_view.html +0 -51
- flock/webapp/templates/partials/_agent_tools_checklist.html +0 -14
- flock/webapp/templates/partials/_chat_container.html +0 -15
- flock/webapp/templates/partials/_chat_messages.html +0 -57
- flock/webapp/templates/partials/_chat_settings_form.html +0 -85
- flock/webapp/templates/partials/_create_flock_form.html +0 -50
- flock/webapp/templates/partials/_dashboard_flock_detail.html +0 -17
- flock/webapp/templates/partials/_dashboard_flock_file_list.html +0 -16
- flock/webapp/templates/partials/_dashboard_flock_properties_preview.html +0 -28
- flock/webapp/templates/partials/_dashboard_upload_flock_form.html +0 -16
- flock/webapp/templates/partials/_dynamic_input_form_content.html +0 -22
- flock/webapp/templates/partials/_env_vars_table.html +0 -23
- flock/webapp/templates/partials/_execution_form.html +0 -118
- flock/webapp/templates/partials/_execution_view_container.html +0 -28
- flock/webapp/templates/partials/_flock_file_list.html +0 -23
- flock/webapp/templates/partials/_flock_properties_form.html +0 -52
- flock/webapp/templates/partials/_flock_upload_form.html +0 -16
- flock/webapp/templates/partials/_header_flock_status.html +0 -5
- flock/webapp/templates/partials/_load_manager_view.html +0 -49
- flock/webapp/templates/partials/_registry_table.html +0 -25
- flock/webapp/templates/partials/_registry_viewer_content.html +0 -70
- flock/webapp/templates/partials/_results_display.html +0 -78
- flock/webapp/templates/partials/_settings_env_content.html +0 -9
- flock/webapp/templates/partials/_settings_theme_content.html +0 -14
- flock/webapp/templates/partials/_settings_view.html +0 -36
- flock/webapp/templates/partials/_share_chat_link_snippet.html +0 -11
- flock/webapp/templates/partials/_share_link_snippet.html +0 -35
- flock/webapp/templates/partials/_sidebar.html +0 -74
- flock/webapp/templates/partials/_streaming_results_container.html +0 -195
- flock/webapp/templates/partials/_structured_data_view.html +0 -40
- flock/webapp/templates/partials/_theme_preview.html +0 -36
- flock/webapp/templates/registry_viewer.html +0 -84
- flock/webapp/templates/shared_run_page.html +0 -140
- flock/workflow/__init__.py +0 -0
- flock/workflow/activities.py +0 -196
- flock/workflow/agent_activities.py +0 -24
- flock/workflow/agent_execution_activity.py +0 -202
- flock/workflow/flock_workflow.py +0 -214
- flock/workflow/temporal_config.py +0 -96
- flock/workflow/temporal_setup.py +0 -68
- flock_core-0.5.0b28.dist-info/METADATA +0 -274
- flock_core-0.5.0b28.dist-info/RECORD +0 -561
- flock_core-0.5.0b28.dist-info/entry_points.txt +0 -2
- /flock/{core/logging → logging}/formatters/themes.py +0 -0
- /flock/{core/logging → logging}/span_middleware/baggage_span_processor.py +0 -0
- /flock/{core/mcp → mcp}/util/__init__.py +0 -0
- {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
|
-
)
|