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
flock/workflow/__init__.py
DELETED
|
File without changes
|
flock/workflow/activities.py
DELETED
|
@@ -1,196 +0,0 @@
|
|
|
1
|
-
"""Defines Temporal activities for running a chain of agents with logging and tracing."""
|
|
2
|
-
|
|
3
|
-
from datetime import datetime
|
|
4
|
-
|
|
5
|
-
from opentelemetry import trace
|
|
6
|
-
from temporalio import activity
|
|
7
|
-
|
|
8
|
-
from flock.core.context.context import FlockContext
|
|
9
|
-
from flock.core.context.context_vars import FLOCK_CURRENT_AGENT, FLOCK_MODEL
|
|
10
|
-
|
|
11
|
-
# HandOffRequest removed - using agent.next_agent directly
|
|
12
|
-
from flock.core.logging.logging import get_logger
|
|
13
|
-
from flock.core.registry import get_registry
|
|
14
|
-
from flock.core.util.input_resolver import resolve_inputs
|
|
15
|
-
|
|
16
|
-
logger = get_logger("activities")
|
|
17
|
-
tracer = trace.get_tracer(__name__)
|
|
18
|
-
|
|
19
|
-
def apply_handoff_strategy(previous_agent_output:str, next_agent_input:str, previous_agent_handoff_strategy:str, previous_agent_handoff_map:dict[str, str]) -> str:
|
|
20
|
-
if previous_agent_handoff_strategy == "append":
|
|
21
|
-
return next_agent_input + previous_agent_output
|
|
22
|
-
elif previous_agent_handoff_strategy == "override":
|
|
23
|
-
return previous_agent_output
|
|
24
|
-
elif previous_agent_handoff_strategy == "static":
|
|
25
|
-
return next_agent_input
|
|
26
|
-
elif previous_agent_handoff_strategy == "map":
|
|
27
|
-
for key, value in previous_agent_handoff_map.items():
|
|
28
|
-
next_agent_input = next_agent_input.replace(key, value)
|
|
29
|
-
return next_agent_input
|
|
30
|
-
raise NotImplementedError
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
@activity.defn
|
|
34
|
-
async def run_agent(context: FlockContext) -> dict:
|
|
35
|
-
"""Runs a chain of agents using the provided context.
|
|
36
|
-
|
|
37
|
-
The context contains state, history, and agent definitions.
|
|
38
|
-
After each agent run, its output is merged into the context.
|
|
39
|
-
"""
|
|
40
|
-
# Start a top-level span for the entire run_agent activity.
|
|
41
|
-
with tracer.start_as_current_span("run_agent") as span:
|
|
42
|
-
registry = get_registry()
|
|
43
|
-
|
|
44
|
-
previous_agent_name = ""
|
|
45
|
-
previous_agent_output = ""
|
|
46
|
-
previous_agent_handoff_strategy = ""
|
|
47
|
-
previous_agent_handoff_map = {}
|
|
48
|
-
if isinstance(context, dict):
|
|
49
|
-
context = FlockContext.from_dict(context)
|
|
50
|
-
current_agent_name = context.get_variable(FLOCK_CURRENT_AGENT)
|
|
51
|
-
span.set_attribute("initial.agent", current_agent_name)
|
|
52
|
-
logger.info("Starting agent chain", initial_agent=current_agent_name)
|
|
53
|
-
|
|
54
|
-
agent = registry.get_agent(current_agent_name)
|
|
55
|
-
if agent.model is None or agent.evaluator.config.model is None:
|
|
56
|
-
agent.set_model(context.get_variable(FLOCK_MODEL))
|
|
57
|
-
|
|
58
|
-
if not agent:
|
|
59
|
-
logger.error("Agent not found", agent=current_agent_name)
|
|
60
|
-
span.record_exception(
|
|
61
|
-
Exception(f"Agent '{current_agent_name}' not found")
|
|
62
|
-
)
|
|
63
|
-
return {"error": f"Agent '{current_agent_name}' not found."}
|
|
64
|
-
|
|
65
|
-
# Loop over agents in the chain.
|
|
66
|
-
while agent:
|
|
67
|
-
# Create a nested span for this iteration.
|
|
68
|
-
with tracer.start_as_current_span("agent_iteration") as iter_span:
|
|
69
|
-
iter_span.set_attribute("agent.name", agent.name)
|
|
70
|
-
agent.context = context
|
|
71
|
-
# Resolve inputs for the agent.
|
|
72
|
-
# Gets values from context, previous agent output, and handoff strategy.
|
|
73
|
-
agent_inputs = resolve_inputs(
|
|
74
|
-
agent.input,
|
|
75
|
-
context,
|
|
76
|
-
previous_agent_name,
|
|
77
|
-
previous_agent_output,
|
|
78
|
-
previous_agent_handoff_strategy,
|
|
79
|
-
previous_agent_handoff_map
|
|
80
|
-
)
|
|
81
|
-
iter_span.add_event(
|
|
82
|
-
"resolved inputs", attributes={"inputs": str(agent_inputs)}
|
|
83
|
-
)
|
|
84
|
-
|
|
85
|
-
# Execute the agent with its own span.
|
|
86
|
-
with tracer.start_as_current_span("execute_agent") as exec_span:
|
|
87
|
-
logger.info("Executing agent", agent=agent.name)
|
|
88
|
-
try:
|
|
89
|
-
result = await agent.run_async(agent_inputs)
|
|
90
|
-
exec_span.set_attribute("result", str(result))
|
|
91
|
-
logger.debug(
|
|
92
|
-
"Agent execution completed", agent=agent.name
|
|
93
|
-
)
|
|
94
|
-
context.record(
|
|
95
|
-
agent.name,
|
|
96
|
-
result,
|
|
97
|
-
timestamp=datetime.now().isoformat(),
|
|
98
|
-
hand_off=None,
|
|
99
|
-
called_from=previous_agent_name,
|
|
100
|
-
)
|
|
101
|
-
except Exception as e:
|
|
102
|
-
logger.error(
|
|
103
|
-
"Agent execution failed",
|
|
104
|
-
agent=agent.name,
|
|
105
|
-
error=str(e),
|
|
106
|
-
)
|
|
107
|
-
exec_span.record_exception(e)
|
|
108
|
-
raise
|
|
109
|
-
|
|
110
|
-
# Determine the next agent using routing component if available
|
|
111
|
-
next_agent_name = None
|
|
112
|
-
|
|
113
|
-
if agent.router:
|
|
114
|
-
logger.info(
|
|
115
|
-
f"Using router: {agent.router.__class__.__name__}",
|
|
116
|
-
agent=agent.name,
|
|
117
|
-
)
|
|
118
|
-
try:
|
|
119
|
-
# Route to the next agent using new routing component
|
|
120
|
-
next_agent_name = await agent.router.determine_next_step(
|
|
121
|
-
agent, result, context
|
|
122
|
-
)
|
|
123
|
-
|
|
124
|
-
# Set next_agent on the agent instance
|
|
125
|
-
agent.next_agent = next_agent_name
|
|
126
|
-
|
|
127
|
-
except Exception as e:
|
|
128
|
-
logger.error(
|
|
129
|
-
f"Router error: {e}",
|
|
130
|
-
agent=agent.name,
|
|
131
|
-
error=str(e),
|
|
132
|
-
)
|
|
133
|
-
iter_span.record_exception(e)
|
|
134
|
-
return {"error": f"Router error: {e}"}
|
|
135
|
-
else:
|
|
136
|
-
# Check if next_agent was set directly by user
|
|
137
|
-
next_agent_name = agent.next_agent
|
|
138
|
-
if callable(next_agent_name):
|
|
139
|
-
try:
|
|
140
|
-
next_agent_name = next_agent_name(context, result)
|
|
141
|
-
except Exception as e:
|
|
142
|
-
logger.error(f"next_agent callable error: {e}")
|
|
143
|
-
return {"error": f"next_agent callable error: {e}"}
|
|
144
|
-
|
|
145
|
-
if not next_agent_name:
|
|
146
|
-
logger.info(
|
|
147
|
-
"No next agent found, completing chain",
|
|
148
|
-
agent=agent.name,
|
|
149
|
-
)
|
|
150
|
-
iter_span.add_event("chain completed")
|
|
151
|
-
return result
|
|
152
|
-
|
|
153
|
-
# Record the agent run in the context.
|
|
154
|
-
context.record(
|
|
155
|
-
agent.name,
|
|
156
|
-
result,
|
|
157
|
-
timestamp=datetime.now().isoformat(),
|
|
158
|
-
hand_off={"next_agent": next_agent_name} if next_agent_name else None,
|
|
159
|
-
called_from=previous_agent_name,
|
|
160
|
-
)
|
|
161
|
-
# Remember the current agent's details for the next iteration.
|
|
162
|
-
previous_agent_name = agent.name
|
|
163
|
-
previous_agent_output = agent.output
|
|
164
|
-
previous_agent_handoff_strategy = agent.config.handoff_strategy
|
|
165
|
-
previous_agent_handoff_map = agent.config.handoff_map
|
|
166
|
-
|
|
167
|
-
# Activate the next agent.
|
|
168
|
-
try:
|
|
169
|
-
agent = registry.get_agent(next_agent_name)
|
|
170
|
-
if not agent:
|
|
171
|
-
logger.error(
|
|
172
|
-
"Next agent not found",
|
|
173
|
-
agent=next_agent_name,
|
|
174
|
-
)
|
|
175
|
-
iter_span.record_exception(
|
|
176
|
-
Exception(
|
|
177
|
-
f"Next agent '{next_agent_name}' not found"
|
|
178
|
-
)
|
|
179
|
-
)
|
|
180
|
-
return {
|
|
181
|
-
"error": f"Next agent '{next_agent_name}' not found."
|
|
182
|
-
}
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
186
|
-
context.set_variable(FLOCK_CURRENT_AGENT, agent.name)
|
|
187
|
-
|
|
188
|
-
logger.info("Handing off to next agent", next=agent.name)
|
|
189
|
-
iter_span.set_attribute("next.agent", agent.name)
|
|
190
|
-
except Exception as e:
|
|
191
|
-
logger.error("Error during handoff", error=str(e))
|
|
192
|
-
iter_span.record_exception(e)
|
|
193
|
-
return {"error": f"Error during handoff: {e}"}
|
|
194
|
-
|
|
195
|
-
# If the loop exits unexpectedly, return the initial input.
|
|
196
|
-
return context.get_variable("init_input")
|
|
@@ -1,24 +0,0 @@
|
|
|
1
|
-
from temporalio import activity
|
|
2
|
-
|
|
3
|
-
from flock.core.context.context import FlockContext
|
|
4
|
-
from flock.core.flock_agent import FlockAgent
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
@activity.defn
|
|
8
|
-
async def run_declarative_agent_activity(params: dict) -> dict:
|
|
9
|
-
"""Temporal activity to run a declarative (or batch) agent.
|
|
10
|
-
|
|
11
|
-
Expects a dictionary with:
|
|
12
|
-
- "agent_data": a dict representation of the agent (as produced by .dict()),
|
|
13
|
-
- "context_data": a dict containing the FlockContext state and optionally other fields.
|
|
14
|
-
|
|
15
|
-
The activity reconstructs the agent and a FlockContext, then calls the agent’s _evaluate() method.
|
|
16
|
-
"""
|
|
17
|
-
agent_data = params.get("agent_data")
|
|
18
|
-
context_data = params.get("context_data", {})
|
|
19
|
-
# Reconstruct the agent from its serialized representation.
|
|
20
|
-
agent = FlockAgent.from_dict(agent_data)
|
|
21
|
-
# Reconstruct the FlockContext from the state.
|
|
22
|
-
context = FlockContext.from_dict(context_data)
|
|
23
|
-
result = await agent.evaluate(context)
|
|
24
|
-
return result
|
|
@@ -1,202 +0,0 @@
|
|
|
1
|
-
"""Defines granular Temporal activities for executing a single agent
|
|
2
|
-
and determining the next agent in a Flock workflow.
|
|
3
|
-
"""
|
|
4
|
-
|
|
5
|
-
from collections.abc import Callable
|
|
6
|
-
|
|
7
|
-
from opentelemetry import trace
|
|
8
|
-
from temporalio import activity
|
|
9
|
-
|
|
10
|
-
# Third-party imports only within activity functions if needed, or pass context
|
|
11
|
-
# For core flock types, import directly
|
|
12
|
-
from flock.core.context.context import FlockContext
|
|
13
|
-
from flock.core.context.context_vars import FLOCK_MODEL
|
|
14
|
-
from flock.core.flock_agent import FlockAgent # Import concrete class if needed
|
|
15
|
-
from flock.core.registry import get_registry
|
|
16
|
-
# HandOffRequest removed - using agent.next_agent directly
|
|
17
|
-
from flock.core.logging.logging import get_logger
|
|
18
|
-
from flock.core.util.input_resolver import resolve_inputs
|
|
19
|
-
|
|
20
|
-
logger = get_logger("agent_activity") # Using a distinct logger category
|
|
21
|
-
tracer = trace.get_tracer(__name__)
|
|
22
|
-
registry = get_registry() # Get registry instance once
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
@activity.defn
|
|
26
|
-
async def execute_single_agent(agent_name: str, context_dict: dict) -> dict:
|
|
27
|
-
"""Executes a single specified agent and returns its result.
|
|
28
|
-
|
|
29
|
-
Args:
|
|
30
|
-
agent_name: The name of the agent to execute.
|
|
31
|
-
context: The current FlockContext (passed from the workflow).
|
|
32
|
-
|
|
33
|
-
Returns:
|
|
34
|
-
The raw result dictionary from the agent's execution.
|
|
35
|
-
|
|
36
|
-
Raises:
|
|
37
|
-
ValueError: If the agent is not found in the registry.
|
|
38
|
-
Exception: Propagates exceptions from agent execution for Temporal retries.
|
|
39
|
-
"""
|
|
40
|
-
with tracer.start_as_current_span("execute_single_agent") as span:
|
|
41
|
-
span.set_attribute("agent.name", agent_name)
|
|
42
|
-
logger.info("Executing single agent", agent=agent_name)
|
|
43
|
-
|
|
44
|
-
agent = registry.get_agent(agent_name)
|
|
45
|
-
if not agent:
|
|
46
|
-
logger.error("Agent not found in registry", agent=agent_name)
|
|
47
|
-
# Raise error for Temporal to potentially retry/fail the activity
|
|
48
|
-
raise ValueError(f"Agent '{agent_name}' not found in registry.")
|
|
49
|
-
|
|
50
|
-
# Rehydrate context from dict and set on agent (transient for this execution)
|
|
51
|
-
context = FlockContext.from_dict(context_dict)
|
|
52
|
-
agent.context = context
|
|
53
|
-
|
|
54
|
-
# Ensure model is set (using context value if needed)
|
|
55
|
-
# Consider if this should be done once when agent is added or workflow starts
|
|
56
|
-
if agent.model is None:
|
|
57
|
-
agent_model = context.get_variable(FLOCK_MODEL)
|
|
58
|
-
if agent_model:
|
|
59
|
-
agent.set_model(agent_model)
|
|
60
|
-
logger.debug(
|
|
61
|
-
f"Set model for agent '{agent_name}' from context: {agent_model}"
|
|
62
|
-
)
|
|
63
|
-
|
|
64
|
-
# Resolve agent-specific callables if necessary
|
|
65
|
-
# This might be better handled in the workflow before the loop starts
|
|
66
|
-
# or when agents are initially loaded. Assuming it's handled elsewhere for now.
|
|
67
|
-
# agent.resolve_callables(context=context)
|
|
68
|
-
|
|
69
|
-
# Resolve inputs for this specific agent run
|
|
70
|
-
previous_agent_name = context.get_last_agent_name() # May be None on first agent
|
|
71
|
-
prev_def = (
|
|
72
|
-
context.get_agent_definition(previous_agent_name)
|
|
73
|
-
if previous_agent_name
|
|
74
|
-
else None
|
|
75
|
-
)
|
|
76
|
-
prev_out_spec = (
|
|
77
|
-
(prev_def.agent_data.get("output_spec") if isinstance(prev_def, type(prev_def)) else None)
|
|
78
|
-
if prev_def and isinstance(prev_def.agent_data, dict)
|
|
79
|
-
else None
|
|
80
|
-
)
|
|
81
|
-
prev_cfg = (
|
|
82
|
-
prev_def.agent_data.get("config")
|
|
83
|
-
if prev_def and isinstance(prev_def.agent_data, dict)
|
|
84
|
-
else {}
|
|
85
|
-
)
|
|
86
|
-
prev_strategy = (
|
|
87
|
-
prev_cfg.get("handoff_strategy") if isinstance(prev_cfg, dict) else None
|
|
88
|
-
) or "static"
|
|
89
|
-
prev_map = (
|
|
90
|
-
prev_cfg.get("handoff_map") if isinstance(prev_cfg, dict) else None
|
|
91
|
-
) or {}
|
|
92
|
-
logger.debug(
|
|
93
|
-
f"Resolving inputs for {agent_name} with previous agent {previous_agent_name}"
|
|
94
|
-
)
|
|
95
|
-
agent_inputs = resolve_inputs(
|
|
96
|
-
agent.input,
|
|
97
|
-
context,
|
|
98
|
-
previous_agent_name or "",
|
|
99
|
-
prev_out_spec or "",
|
|
100
|
-
prev_strategy,
|
|
101
|
-
prev_map,
|
|
102
|
-
)
|
|
103
|
-
span.add_event(
|
|
104
|
-
"resolved inputs", attributes={"inputs": str(agent_inputs)}
|
|
105
|
-
)
|
|
106
|
-
|
|
107
|
-
try:
|
|
108
|
-
# Execute just this agent
|
|
109
|
-
result = await agent.run_async(agent_inputs)
|
|
110
|
-
# Avoid logging potentially large results directly to span attributes
|
|
111
|
-
result_str = str(result)
|
|
112
|
-
span.set_attribute("result.type", type(result).__name__)
|
|
113
|
-
span.set_attribute(
|
|
114
|
-
"result.preview",
|
|
115
|
-
result_str[:500] + ("..." if len(result_str) > 500 else ""),
|
|
116
|
-
)
|
|
117
|
-
logger.info("Single agent execution completed", agent=agent_name)
|
|
118
|
-
return result
|
|
119
|
-
except Exception as e:
|
|
120
|
-
logger.error(
|
|
121
|
-
"Single agent execution failed",
|
|
122
|
-
agent=agent_name,
|
|
123
|
-
error=str(e),
|
|
124
|
-
exc_info=True,
|
|
125
|
-
)
|
|
126
|
-
# Debug aid: ensure exception prints even if logger is muted in environment
|
|
127
|
-
print(f"[agent_activity] Single agent execution failed for {agent_name}: {e!r}")
|
|
128
|
-
span.record_exception(e)
|
|
129
|
-
# Re-raise the exception for Temporal to handle based on retry policy
|
|
130
|
-
raise
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
@activity.defn
|
|
134
|
-
async def determine_next_agent(
|
|
135
|
-
current_agent_name: str, result: dict, context_dict: dict
|
|
136
|
-
) -> str | None:
|
|
137
|
-
"""Determine the next agent using the agent's routing component.
|
|
138
|
-
|
|
139
|
-
Returns the next agent's name or None if the workflow should terminate.
|
|
140
|
-
"""
|
|
141
|
-
with tracer.start_as_current_span("determine_next_agent") as span:
|
|
142
|
-
span.set_attribute("agent.name", current_agent_name)
|
|
143
|
-
logger.info("Determining next agent after", agent=current_agent_name)
|
|
144
|
-
|
|
145
|
-
agent = registry.get_agent(current_agent_name)
|
|
146
|
-
if not agent:
|
|
147
|
-
logger.error(
|
|
148
|
-
"Agent not found for routing", agent=current_agent_name
|
|
149
|
-
)
|
|
150
|
-
raise ValueError(
|
|
151
|
-
f"Agent '{current_agent_name}' not found for routing."
|
|
152
|
-
)
|
|
153
|
-
|
|
154
|
-
if not agent.router:
|
|
155
|
-
logger.info(
|
|
156
|
-
"No router defined for agent", agent=current_agent_name
|
|
157
|
-
)
|
|
158
|
-
span.add_event("no_router")
|
|
159
|
-
return None # Indicate no handoff
|
|
160
|
-
|
|
161
|
-
logger.debug(
|
|
162
|
-
f"Using router {agent.router.__class__.__name__}",
|
|
163
|
-
agent=agent.name,
|
|
164
|
-
)
|
|
165
|
-
try:
|
|
166
|
-
# Execute routing logic on the router component (unified architecture)
|
|
167
|
-
context = FlockContext.from_dict(context_dict)
|
|
168
|
-
next_val = await agent.router.determine_next_step(agent, result, context)
|
|
169
|
-
|
|
170
|
-
# Convert to a simple agent name if needed
|
|
171
|
-
if isinstance(next_val, FlockAgent):
|
|
172
|
-
next_name = next_val.name
|
|
173
|
-
elif isinstance(next_val, str):
|
|
174
|
-
next_name = next_val
|
|
175
|
-
else:
|
|
176
|
-
next_name = None
|
|
177
|
-
|
|
178
|
-
if not next_name:
|
|
179
|
-
logger.info("Router determined no next agent", agent=agent.name)
|
|
180
|
-
span.add_event("no_next_agent_from_router")
|
|
181
|
-
return None
|
|
182
|
-
|
|
183
|
-
logger.info(
|
|
184
|
-
"Next agent determined",
|
|
185
|
-
next_agent=next_name,
|
|
186
|
-
agent=agent.name,
|
|
187
|
-
)
|
|
188
|
-
span.set_attribute("next_agent", next_name)
|
|
189
|
-
return next_name
|
|
190
|
-
|
|
191
|
-
except Exception as e:
|
|
192
|
-
# Catch potential errors during routing execution
|
|
193
|
-
logger.error(
|
|
194
|
-
"Router execution failed",
|
|
195
|
-
agent=agent.name,
|
|
196
|
-
error=str(e),
|
|
197
|
-
exc_info=True,
|
|
198
|
-
)
|
|
199
|
-
print(f"[agent_activity] Router execution failed for {agent.name}: {e!r}")
|
|
200
|
-
span.record_exception(e)
|
|
201
|
-
# Let Temporal handle the activity failure based on retry policy
|
|
202
|
-
raise
|
flock/workflow/flock_workflow.py
DELETED
|
@@ -1,214 +0,0 @@
|
|
|
1
|
-
from datetime import timedelta
|
|
2
|
-
from typing import Any
|
|
3
|
-
|
|
4
|
-
from temporalio import workflow
|
|
5
|
-
|
|
6
|
-
# Import activities from the new file
|
|
7
|
-
with workflow.unsafe.imports_passed_through():
|
|
8
|
-
from flock.core.context.context import AgentDefinition, FlockContext
|
|
9
|
-
from flock.core.context.context_vars import FLOCK_CURRENT_AGENT
|
|
10
|
-
# HandOffRequest removed - using agent.next_agent directly
|
|
11
|
-
from flock.core.logging.logging import get_logger
|
|
12
|
-
from flock.workflow.agent_execution_activity import (
|
|
13
|
-
determine_next_agent,
|
|
14
|
-
execute_single_agent,
|
|
15
|
-
)
|
|
16
|
-
from flock.workflow.temporal_config import (
|
|
17
|
-
TemporalActivityConfig,
|
|
18
|
-
TemporalRetryPolicyConfig,
|
|
19
|
-
)
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
logger = get_logger("workflow")
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
@workflow.defn
|
|
26
|
-
class FlockWorkflow:
|
|
27
|
-
# No need for __init__ storing context anymore if passed to run
|
|
28
|
-
|
|
29
|
-
@workflow.run
|
|
30
|
-
async def run(self, workflow_args: dict[str, Any]) -> dict:
|
|
31
|
-
# --- Workflow Initialization ---
|
|
32
|
-
# Arguments are packed into a single dictionary
|
|
33
|
-
context_dict = workflow_args["context_dict"]
|
|
34
|
-
default_retry_config_dict = workflow_args["default_retry_config_dict"]
|
|
35
|
-
|
|
36
|
-
# Deserialize context and default retry config
|
|
37
|
-
context = FlockContext.from_dict(context_dict)
|
|
38
|
-
default_retry_config = TemporalRetryPolicyConfig.model_validate(
|
|
39
|
-
default_retry_config_dict
|
|
40
|
-
)
|
|
41
|
-
|
|
42
|
-
context.workflow_id = workflow.info().workflow_id
|
|
43
|
-
context.workflow_timestamp = workflow.info().start_time.strftime(
|
|
44
|
-
"%Y-%m-%d %H:%M:%S"
|
|
45
|
-
)
|
|
46
|
-
|
|
47
|
-
current_agent_name = context.get_variable(FLOCK_CURRENT_AGENT)
|
|
48
|
-
final_result = None
|
|
49
|
-
previous_agent_name = (
|
|
50
|
-
None # Keep track of the agent that called the current one
|
|
51
|
-
)
|
|
52
|
-
|
|
53
|
-
logger.info(
|
|
54
|
-
"Starting workflow execution",
|
|
55
|
-
workflow_id=context.workflow_id,
|
|
56
|
-
start_time=context.workflow_timestamp,
|
|
57
|
-
initial_agent=current_agent_name,
|
|
58
|
-
)
|
|
59
|
-
|
|
60
|
-
try:
|
|
61
|
-
while current_agent_name:
|
|
62
|
-
logger.info(
|
|
63
|
-
"Executing agent activity", agent=current_agent_name
|
|
64
|
-
)
|
|
65
|
-
|
|
66
|
-
# --- Determine Activity Settings ---
|
|
67
|
-
agent_def: AgentDefinition | None = (
|
|
68
|
-
context.get_agent_definition(current_agent_name)
|
|
69
|
-
)
|
|
70
|
-
agent_activity_config: TemporalActivityConfig | None = None
|
|
71
|
-
final_retry_config = (
|
|
72
|
-
default_retry_config # Start with the workflow default
|
|
73
|
-
)
|
|
74
|
-
|
|
75
|
-
if agent_def and agent_def.agent_data.get(
|
|
76
|
-
"temporal_activity_config"
|
|
77
|
-
):
|
|
78
|
-
try:
|
|
79
|
-
agent_activity_config = (
|
|
80
|
-
TemporalActivityConfig.model_validate(
|
|
81
|
-
agent_def.agent_data["temporal_activity_config"]
|
|
82
|
-
)
|
|
83
|
-
)
|
|
84
|
-
logger.debug(
|
|
85
|
-
f"Loaded agent-specific temporal config for {current_agent_name}"
|
|
86
|
-
)
|
|
87
|
-
except Exception as e:
|
|
88
|
-
logger.warn(
|
|
89
|
-
f"Failed to validate agent temporal config for {current_agent_name}: {e}. Using defaults."
|
|
90
|
-
)
|
|
91
|
-
|
|
92
|
-
# Layering logic: Agent config overrides workflow default config
|
|
93
|
-
activity_task_queue = (
|
|
94
|
-
workflow.info().task_queue
|
|
95
|
-
) # Default to workflow task queue
|
|
96
|
-
activity_timeout = timedelta(
|
|
97
|
-
minutes=5
|
|
98
|
-
) # Fallback default timeout
|
|
99
|
-
|
|
100
|
-
if agent_activity_config:
|
|
101
|
-
activity_task_queue = (
|
|
102
|
-
agent_activity_config.task_queue or activity_task_queue
|
|
103
|
-
)
|
|
104
|
-
activity_timeout = (
|
|
105
|
-
agent_activity_config.start_to_close_timeout
|
|
106
|
-
or activity_timeout
|
|
107
|
-
)
|
|
108
|
-
if agent_activity_config.retry_policy:
|
|
109
|
-
final_retry_config = agent_activity_config.retry_policy
|
|
110
|
-
|
|
111
|
-
# Convert config to actual Temporal object
|
|
112
|
-
final_retry_policy = final_retry_config.to_temporalio_policy()
|
|
113
|
-
|
|
114
|
-
logger.debug(
|
|
115
|
-
f"Final activity settings for {current_agent_name}: "
|
|
116
|
-
f"queue='{activity_task_queue}', timeout={activity_timeout}, "
|
|
117
|
-
f"retries={final_retry_policy.maximum_attempts}"
|
|
118
|
-
)
|
|
119
|
-
|
|
120
|
-
# --- Execute the current agent activity ---
|
|
121
|
-
try:
|
|
122
|
-
agent_result = await workflow.execute_activity(
|
|
123
|
-
execute_single_agent,
|
|
124
|
-
args=[current_agent_name, context.model_dump(mode="json")],
|
|
125
|
-
task_queue=activity_task_queue,
|
|
126
|
-
start_to_close_timeout=activity_timeout,
|
|
127
|
-
retry_policy=final_retry_policy,
|
|
128
|
-
)
|
|
129
|
-
except Exception as e:
|
|
130
|
-
logger.error("execute_single_agent activity failed", agent=current_agent_name, error=str(e))
|
|
131
|
-
raise
|
|
132
|
-
|
|
133
|
-
# Record the execution in the context history
|
|
134
|
-
# Note: The 'called_from' is the agent *before* this one
|
|
135
|
-
context.record(
|
|
136
|
-
agent_name=current_agent_name,
|
|
137
|
-
data=agent_result,
|
|
138
|
-
timestamp=workflow.now().isoformat(), # Use deterministic workflow time
|
|
139
|
-
hand_off=None, # Will be updated if handoff occurs
|
|
140
|
-
called_from=previous_agent_name, # Pass the correct previous agent
|
|
141
|
-
)
|
|
142
|
-
|
|
143
|
-
final_result = agent_result # Store the result of the last successful agent
|
|
144
|
-
|
|
145
|
-
logger.info(
|
|
146
|
-
"Determining next agent activity",
|
|
147
|
-
current_agent=current_agent_name,
|
|
148
|
-
)
|
|
149
|
-
# --- Determine the next agent (using routing component) ---
|
|
150
|
-
try:
|
|
151
|
-
next_agent_name = await workflow.execute_activity(
|
|
152
|
-
determine_next_agent,
|
|
153
|
-
args=[current_agent_name, agent_result, context.model_dump(mode="json")],
|
|
154
|
-
start_to_close_timeout=timedelta(minutes=1),
|
|
155
|
-
retry_policy=default_retry_config.to_temporalio_policy(),
|
|
156
|
-
)
|
|
157
|
-
except Exception as e:
|
|
158
|
-
logger.error("determine_next_agent activity failed", agent=current_agent_name, error=str(e))
|
|
159
|
-
raise
|
|
160
|
-
|
|
161
|
-
# Update previous agent name for the next loop iteration
|
|
162
|
-
previous_agent_name = current_agent_name
|
|
163
|
-
|
|
164
|
-
if next_agent_name:
|
|
165
|
-
logger.debug("Next agent received", data=next_agent_name)
|
|
166
|
-
# Update the last record's handoff information for observability
|
|
167
|
-
if context.history:
|
|
168
|
-
context.history[-1].hand_off = {"next_agent": next_agent_name}
|
|
169
|
-
|
|
170
|
-
# Set the next agent
|
|
171
|
-
current_agent_name = next_agent_name
|
|
172
|
-
context.set_variable(FLOCK_CURRENT_AGENT, current_agent_name)
|
|
173
|
-
logger.info("Next agent set", agent=current_agent_name)
|
|
174
|
-
else:
|
|
175
|
-
logger.info("No next agent, workflow terminating.")
|
|
176
|
-
current_agent_name = None
|
|
177
|
-
|
|
178
|
-
# --- Workflow Completion ---
|
|
179
|
-
logger.success(
|
|
180
|
-
"Workflow completed successfully",
|
|
181
|
-
final_agent=previous_agent_name,
|
|
182
|
-
)
|
|
183
|
-
context.set_variable(
|
|
184
|
-
"flock.result",
|
|
185
|
-
{
|
|
186
|
-
"result": final_result, # Return the last agent's result
|
|
187
|
-
"success": True,
|
|
188
|
-
},
|
|
189
|
-
)
|
|
190
|
-
return final_result # Return the actual result of the last agent
|
|
191
|
-
|
|
192
|
-
except Exception as e:
|
|
193
|
-
# Catch exceptions from activities (e.g., after retries fail) or workflow logic errors
|
|
194
|
-
logger.exception(f"Workflow execution failed: {e!r}")
|
|
195
|
-
try:
|
|
196
|
-
from temporalio.exceptions import ActivityError
|
|
197
|
-
if isinstance(e, ActivityError) and getattr(e, "cause", None) is not None:
|
|
198
|
-
logger.error(f"ActivityError cause: {e.cause!r}")
|
|
199
|
-
except Exception:
|
|
200
|
-
pass
|
|
201
|
-
context.set_variable(
|
|
202
|
-
"flock.result",
|
|
203
|
-
{
|
|
204
|
-
"result": f"Workflow failed: {e}",
|
|
205
|
-
"success": False,
|
|
206
|
-
},
|
|
207
|
-
)
|
|
208
|
-
# It's often better to let Temporal record the failure status
|
|
209
|
-
# by re-raising the exception rather than returning a custom error dict.
|
|
210
|
-
# However, returning the context might be useful for debugging.
|
|
211
|
-
# Consider re-raising: raise
|
|
212
|
-
return context.model_dump(
|
|
213
|
-
mode="json"
|
|
214
|
-
) # Return context state on failure
|