2025-04-16 05:42:56 +08:00
|
|
|
|
import os
|
2025-03-30 14:48:57 +08:00
|
|
|
|
import json
|
2025-04-18 18:50:39 +08:00
|
|
|
|
import re
|
2025-04-10 18:52:56 +08:00
|
|
|
|
from uuid import uuid4
|
2025-04-17 01:23:28 +08:00
|
|
|
|
from typing import Optional
|
2025-04-10 18:52:56 +08:00
|
|
|
|
|
2025-04-17 06:13:04 +08:00
|
|
|
|
# from agent.tools.message_tool import MessageTool
|
2025-04-18 06:17:48 +08:00
|
|
|
|
from agent.tools.message_tool import MessageTool
|
2025-04-13 05:40:01 +08:00
|
|
|
|
from agent.tools.sb_deploy_tool import SandboxDeployTool
|
2025-04-21 22:01:51 +08:00
|
|
|
|
from agent.tools.sb_expose_tool import SandboxExposeTool
|
2025-05-10 11:46:48 +08:00
|
|
|
|
from agent.tools.web_search_tool import SandboxWebSearchTool
|
2025-04-10 18:52:56 +08:00
|
|
|
|
from dotenv import load_dotenv
|
2025-04-24 08:45:58 +08:00
|
|
|
|
from utils.config import config
|
2025-04-10 18:52:56 +08:00
|
|
|
|
|
2025-04-06 17:10:18 +08:00
|
|
|
|
from agentpress.thread_manager import ThreadManager
|
2025-04-10 18:52:56 +08:00
|
|
|
|
from agentpress.response_processor import ProcessorConfig
|
2025-04-09 07:37:06 +08:00
|
|
|
|
from agent.tools.sb_shell_tool import SandboxShellTool
|
2025-04-09 20:54:25 +08:00
|
|
|
|
from agent.tools.sb_files_tool import SandboxFilesTool
|
2025-04-14 21:06:02 +08:00
|
|
|
|
from agent.tools.sb_browser_tool import SandboxBrowserTool
|
2025-04-17 01:23:28 +08:00
|
|
|
|
from agent.tools.data_providers_tool import DataProvidersTool
|
2025-04-07 00:45:02 +08:00
|
|
|
|
from agent.prompt import get_system_prompt
|
2025-05-10 12:37:07 +08:00
|
|
|
|
from utils.logger import logger
|
2025-04-27 07:47:06 +08:00
|
|
|
|
from utils.auth_utils import get_account_id_from_thread
|
|
|
|
|
from services.billing import check_billing_status
|
2025-04-28 08:28:21 +08:00
|
|
|
|
from agent.tools.sb_vision_tool import SandboxVisionTool
|
2025-05-21 08:39:28 +08:00
|
|
|
|
from services.langfuse import langfuse
|
|
|
|
|
from langfuse.client import StatefulTraceClient
|
2025-05-22 00:19:46 +08:00
|
|
|
|
from agent.gemini_prompt import get_gemini_system_prompt
|
2025-04-11 00:02:21 +08:00
|
|
|
|
|
2025-04-04 23:06:49 +08:00
|
|
|
|
load_dotenv()
|
2025-03-30 14:48:57 +08:00
|
|
|
|
|
2025-04-18 12:49:41 +08:00
|
|
|
|
async def run_agent(
|
|
|
|
|
thread_id: str,
|
|
|
|
|
project_id: str,
|
|
|
|
|
stream: bool,
|
|
|
|
|
thread_manager: Optional[ThreadManager] = None,
|
|
|
|
|
native_max_auto_continues: int = 25,
|
2025-05-10 09:58:57 +08:00
|
|
|
|
max_iterations: int = 100,
|
2025-05-05 08:53:24 +08:00
|
|
|
|
model_name: str = "anthropic/claude-3-7-sonnet-latest",
|
2025-04-18 12:49:41 +08:00
|
|
|
|
enable_thinking: Optional[bool] = False,
|
2025-04-18 14:13:44 +08:00
|
|
|
|
reasoning_effort: Optional[str] = 'low',
|
2025-05-24 15:08:41 +08:00
|
|
|
|
enable_context_manager: bool = True,
|
2025-05-24 16:15:44 +08:00
|
|
|
|
agent_config: Optional[dict] = None,
|
2025-05-21 08:39:28 +08:00
|
|
|
|
trace: Optional[StatefulTraceClient] = None
|
2025-04-18 12:49:41 +08:00
|
|
|
|
):
|
2025-03-30 14:48:57 +08:00
|
|
|
|
"""Run the development agent with specified configuration."""
|
2025-05-10 12:37:07 +08:00
|
|
|
|
logger.info(f"🚀 Starting agent with model: {model_name}")
|
2025-05-24 15:08:41 +08:00
|
|
|
|
if agent_config:
|
|
|
|
|
logger.info(f"Using custom agent: {agent_config.get('name', 'Unknown')}")
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 21:34:18 +08:00
|
|
|
|
thread_manager = ThreadManager()
|
2025-04-23 21:43:57 +08:00
|
|
|
|
|
2025-04-10 18:52:56 +08:00
|
|
|
|
client = await thread_manager.db.client
|
2025-04-14 08:32:08 +08:00
|
|
|
|
|
|
|
|
|
# Get account ID from thread for billing checks
|
|
|
|
|
account_id = await get_account_id_from_thread(client, thread_id)
|
|
|
|
|
if not account_id:
|
|
|
|
|
raise ValueError("Could not determine account ID for thread")
|
|
|
|
|
|
2025-04-23 17:34:00 +08:00
|
|
|
|
# Get sandbox info from project
|
|
|
|
|
project = await client.table('projects').select('*').eq('project_id', project_id).execute()
|
|
|
|
|
if not project.data or len(project.data) == 0:
|
|
|
|
|
raise ValueError(f"Project {project_id} not found")
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-05-21 08:39:28 +08:00
|
|
|
|
if not trace:
|
|
|
|
|
logger.warning("No trace provided, creating a new one")
|
|
|
|
|
trace = langfuse.trace(name="agent_run", id=thread_id, session_id=thread_id, metadata={"project_id": project_id})
|
|
|
|
|
|
2025-04-23 17:34:00 +08:00
|
|
|
|
project_data = project.data[0]
|
|
|
|
|
sandbox_info = project_data.get('sandbox', {})
|
|
|
|
|
if not sandbox_info.get('id'):
|
|
|
|
|
raise ValueError(f"No sandbox found for project {project_id}")
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 18:00:25 +08:00
|
|
|
|
# Initialize tools with project_id instead of sandbox object
|
|
|
|
|
# This ensures each tool independently verifies it's operating on the correct project
|
2025-05-24 15:08:41 +08:00
|
|
|
|
|
|
|
|
|
# Get enabled tools from agent config, or use defaults
|
|
|
|
|
enabled_tools = None
|
|
|
|
|
if agent_config and 'agentpress_tools' in agent_config:
|
|
|
|
|
enabled_tools = agent_config['agentpress_tools']
|
|
|
|
|
logger.info(f"Using custom tool configuration from agent")
|
|
|
|
|
|
|
|
|
|
# Register tools based on configuration
|
|
|
|
|
# If no agent config (enabled_tools is None), register ALL tools for full Suna capabilities
|
|
|
|
|
# If agent config exists, only register explicitly enabled tools
|
|
|
|
|
|
|
|
|
|
if enabled_tools is None:
|
|
|
|
|
# No agent specified - register ALL tools for full Suna experience
|
|
|
|
|
logger.info("No agent specified - registering all tools for full Suna capabilities")
|
|
|
|
|
thread_manager.add_tool(SandboxShellTool, project_id=project_id, thread_manager=thread_manager)
|
|
|
|
|
thread_manager.add_tool(SandboxFilesTool, project_id=project_id, thread_manager=thread_manager)
|
|
|
|
|
thread_manager.add_tool(SandboxBrowserTool, project_id=project_id, thread_id=thread_id, thread_manager=thread_manager)
|
|
|
|
|
thread_manager.add_tool(SandboxDeployTool, project_id=project_id, thread_manager=thread_manager)
|
|
|
|
|
thread_manager.add_tool(SandboxExposeTool, project_id=project_id, thread_manager=thread_manager)
|
|
|
|
|
thread_manager.add_tool(MessageTool)
|
|
|
|
|
thread_manager.add_tool(SandboxWebSearchTool, project_id=project_id, thread_manager=thread_manager)
|
|
|
|
|
thread_manager.add_tool(SandboxVisionTool, project_id=project_id, thread_id=thread_id, thread_manager=thread_manager)
|
|
|
|
|
if config.RAPID_API_KEY:
|
|
|
|
|
thread_manager.add_tool(DataProvidersTool)
|
|
|
|
|
else:
|
|
|
|
|
# Agent specified - only register explicitly enabled tools
|
|
|
|
|
logger.info("Custom agent specified - registering only enabled tools")
|
|
|
|
|
if enabled_tools.get('sb_shell_tool', {}).get('enabled', False):
|
|
|
|
|
thread_manager.add_tool(SandboxShellTool, project_id=project_id, thread_manager=thread_manager)
|
|
|
|
|
if enabled_tools.get('sb_files_tool', {}).get('enabled', False):
|
|
|
|
|
thread_manager.add_tool(SandboxFilesTool, project_id=project_id, thread_manager=thread_manager)
|
|
|
|
|
if enabled_tools.get('sb_browser_tool', {}).get('enabled', False):
|
|
|
|
|
thread_manager.add_tool(SandboxBrowserTool, project_id=project_id, thread_id=thread_id, thread_manager=thread_manager)
|
|
|
|
|
if enabled_tools.get('sb_deploy_tool', {}).get('enabled', False):
|
|
|
|
|
thread_manager.add_tool(SandboxDeployTool, project_id=project_id, thread_manager=thread_manager)
|
|
|
|
|
if enabled_tools.get('sb_expose_tool', {}).get('enabled', False):
|
|
|
|
|
thread_manager.add_tool(SandboxExposeTool, project_id=project_id, thread_manager=thread_manager)
|
|
|
|
|
if enabled_tools.get('message_tool', {}).get('enabled', False):
|
|
|
|
|
thread_manager.add_tool(MessageTool)
|
|
|
|
|
if enabled_tools.get('web_search_tool', {}).get('enabled', False):
|
|
|
|
|
thread_manager.add_tool(SandboxWebSearchTool, project_id=project_id, thread_manager=thread_manager)
|
|
|
|
|
if enabled_tools.get('sb_vision_tool', {}).get('enabled', False):
|
|
|
|
|
thread_manager.add_tool(SandboxVisionTool, project_id=project_id, thread_id=thread_id, thread_manager=thread_manager)
|
|
|
|
|
if config.RAPID_API_KEY and enabled_tools.get('data_providers_tool', {}).get('enabled', False):
|
|
|
|
|
thread_manager.add_tool(DataProvidersTool)
|
2025-05-07 07:56:52 +08:00
|
|
|
|
|
2025-05-24 15:08:41 +08:00
|
|
|
|
# Prepare system prompt
|
|
|
|
|
# First, get the default system prompt
|
2025-05-22 18:35:27 +08:00
|
|
|
|
if "gemini-2.5-flash" in model_name.lower():
|
2025-05-24 15:08:41 +08:00
|
|
|
|
default_system_content = get_gemini_system_prompt()
|
|
|
|
|
else:
|
|
|
|
|
# Use the original prompt - the LLM can only use tools that are registered
|
|
|
|
|
default_system_content = get_system_prompt()
|
|
|
|
|
|
|
|
|
|
# Add sample response for non-anthropic models
|
|
|
|
|
if "anthropic" not in model_name.lower():
|
2025-05-07 07:56:52 +08:00
|
|
|
|
sample_response_path = os.path.join(os.path.dirname(__file__), 'sample_responses/1.txt')
|
|
|
|
|
with open(sample_response_path, 'r') as file:
|
|
|
|
|
sample_response = file.read()
|
2025-05-24 15:08:41 +08:00
|
|
|
|
default_system_content = default_system_content + "\n\n <sample_assistant_response>" + sample_response + "</sample_assistant_response>"
|
|
|
|
|
|
|
|
|
|
# Handle custom agent system prompt
|
|
|
|
|
if agent_config and agent_config.get('system_prompt'):
|
|
|
|
|
custom_system_prompt = agent_config['system_prompt'].strip()
|
2025-05-07 07:56:52 +08:00
|
|
|
|
|
2025-05-24 15:08:41 +08:00
|
|
|
|
# Completely replace the default system prompt with the custom one
|
|
|
|
|
# This prevents confusion and tool hallucination
|
|
|
|
|
system_content = custom_system_prompt
|
|
|
|
|
logger.info(f"Using ONLY custom agent system prompt for: {agent_config.get('name', 'Unknown')}")
|
2025-05-07 07:56:52 +08:00
|
|
|
|
else:
|
2025-05-24 15:08:41 +08:00
|
|
|
|
# Use just the default system prompt
|
|
|
|
|
system_content = default_system_content
|
|
|
|
|
logger.info("Using default system prompt only")
|
|
|
|
|
|
|
|
|
|
system_message = { "role": "system", "content": system_content }
|
2025-03-30 14:48:57 +08:00
|
|
|
|
|
2025-04-10 21:13:32 +08:00
|
|
|
|
iteration_count = 0
|
|
|
|
|
continue_execution = True
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-10 21:13:32 +08:00
|
|
|
|
while continue_execution and iteration_count < max_iterations:
|
|
|
|
|
iteration_count += 1
|
2025-05-10 12:37:07 +08:00
|
|
|
|
logger.info(f"🔄 Running iteration {iteration_count} of {max_iterations}...")
|
2025-04-14 08:32:08 +08:00
|
|
|
|
|
2025-04-18 06:17:48 +08:00
|
|
|
|
# Billing check on each iteration - still needed within the iterations
|
2025-04-14 08:32:08 +08:00
|
|
|
|
can_run, message, subscription = await check_billing_status(client, account_id)
|
|
|
|
|
if not can_run:
|
|
|
|
|
error_msg = f"Billing limit reached: {message}"
|
|
|
|
|
# Yield a special message to indicate billing limit reached
|
|
|
|
|
yield {
|
|
|
|
|
"type": "status",
|
|
|
|
|
"status": "stopped",
|
|
|
|
|
"message": error_msg
|
|
|
|
|
}
|
|
|
|
|
break
|
2025-04-11 07:53:01 +08:00
|
|
|
|
# Check if last message is from assistant using direct Supabase query
|
2025-04-26 19:53:09 +08:00
|
|
|
|
latest_message = await client.table('messages').select('*').eq('thread_id', thread_id).in_('type', ['assistant', 'tool', 'user']).order('created_at', desc=True).limit(1).execute()
|
2025-04-11 07:53:01 +08:00
|
|
|
|
if latest_message.data and len(latest_message.data) > 0:
|
|
|
|
|
message_type = latest_message.data[0].get('type')
|
|
|
|
|
if message_type == 'assistant':
|
2025-05-10 12:37:07 +08:00
|
|
|
|
logger.info(f"Last message was from assistant, stopping execution")
|
2025-04-11 07:53:01 +08:00
|
|
|
|
continue_execution = False
|
|
|
|
|
break
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-28 08:28:21 +08:00
|
|
|
|
# ---- Temporary Message Handling (Browser State & Image Context) ----
|
2025-04-16 00:36:01 +08:00
|
|
|
|
temporary_message = None
|
2025-04-28 08:28:21 +08:00
|
|
|
|
temp_message_content_list = [] # List to hold text/image blocks
|
|
|
|
|
|
|
|
|
|
# Get the latest browser_state message
|
|
|
|
|
latest_browser_state_msg = await client.table('messages').select('*').eq('thread_id', thread_id).eq('type', 'browser_state').order('created_at', desc=True).limit(1).execute()
|
|
|
|
|
if latest_browser_state_msg.data and len(latest_browser_state_msg.data) > 0:
|
2025-04-16 00:36:01 +08:00
|
|
|
|
try:
|
2025-04-28 08:28:21 +08:00
|
|
|
|
browser_content = json.loads(latest_browser_state_msg.data[0]["content"])
|
|
|
|
|
screenshot_base64 = browser_content.get("screenshot_base64")
|
2025-05-14 13:28:11 +08:00
|
|
|
|
screenshot_url = browser_content.get("screenshot_url")
|
|
|
|
|
|
|
|
|
|
# Create a copy of the browser state without screenshot data
|
2025-04-28 08:28:21 +08:00
|
|
|
|
browser_state_text = browser_content.copy()
|
|
|
|
|
browser_state_text.pop('screenshot_base64', None)
|
|
|
|
|
browser_state_text.pop('screenshot_url', None)
|
|
|
|
|
|
|
|
|
|
if browser_state_text:
|
|
|
|
|
temp_message_content_list.append({
|
2025-04-16 00:36:01 +08:00
|
|
|
|
"type": "text",
|
2025-04-28 08:28:21 +08:00
|
|
|
|
"text": f"The following is the current state of the browser:\n{json.dumps(browser_state_text, indent=2)}"
|
2025-04-16 00:36:01 +08:00
|
|
|
|
})
|
2025-05-14 13:28:11 +08:00
|
|
|
|
|
|
|
|
|
# Prioritize screenshot_url if available
|
|
|
|
|
if screenshot_url:
|
|
|
|
|
temp_message_content_list.append({
|
|
|
|
|
"type": "image_url",
|
|
|
|
|
"image_url": {
|
|
|
|
|
"url": screenshot_url,
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
elif screenshot_base64:
|
|
|
|
|
# Fallback to base64 if URL not available
|
2025-04-28 08:28:21 +08:00
|
|
|
|
temp_message_content_list.append({
|
2025-04-16 00:36:01 +08:00
|
|
|
|
"type": "image_url",
|
2025-04-28 08:28:21 +08:00
|
|
|
|
"image_url": {
|
|
|
|
|
"url": f"data:image/jpeg;base64,{screenshot_base64}",
|
|
|
|
|
}
|
2025-04-16 00:36:01 +08:00
|
|
|
|
})
|
|
|
|
|
else:
|
2025-05-14 13:28:11 +08:00
|
|
|
|
logger.warning("Browser state found but no screenshot data.")
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-16 00:36:01 +08:00
|
|
|
|
except Exception as e:
|
2025-04-28 08:28:21 +08:00
|
|
|
|
logger.error(f"Error parsing browser state: {e}")
|
|
|
|
|
|
|
|
|
|
# Get the latest image_context message (NEW)
|
|
|
|
|
latest_image_context_msg = await client.table('messages').select('*').eq('thread_id', thread_id).eq('type', 'image_context').order('created_at', desc=True).limit(1).execute()
|
|
|
|
|
if latest_image_context_msg.data and len(latest_image_context_msg.data) > 0:
|
|
|
|
|
try:
|
|
|
|
|
image_context_content = json.loads(latest_image_context_msg.data[0]["content"])
|
|
|
|
|
base64_image = image_context_content.get("base64")
|
|
|
|
|
mime_type = image_context_content.get("mime_type")
|
|
|
|
|
file_path = image_context_content.get("file_path", "unknown file")
|
|
|
|
|
|
|
|
|
|
if base64_image and mime_type:
|
|
|
|
|
temp_message_content_list.append({
|
|
|
|
|
"type": "text",
|
|
|
|
|
"text": f"Here is the image you requested to see: '{file_path}'"
|
|
|
|
|
})
|
|
|
|
|
temp_message_content_list.append({
|
|
|
|
|
"type": "image_url",
|
|
|
|
|
"image_url": {
|
|
|
|
|
"url": f"data:{mime_type};base64,{base64_image}",
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
else:
|
|
|
|
|
logger.warning(f"Image context found for '{file_path}' but missing base64 or mime_type.")
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-28 08:28:21 +08:00
|
|
|
|
await client.table('messages').delete().eq('message_id', latest_image_context_msg.data[0]["message_id"]).execute()
|
|
|
|
|
except Exception as e:
|
|
|
|
|
logger.error(f"Error parsing image context: {e}")
|
|
|
|
|
|
|
|
|
|
# If we have any content, construct the temporary_message
|
|
|
|
|
if temp_message_content_list:
|
|
|
|
|
temporary_message = {"role": "user", "content": temp_message_content_list}
|
|
|
|
|
# logger.debug(f"Constructed temporary message with {len(temp_message_content_list)} content blocks.")
|
|
|
|
|
# ---- End Temporary Message Handling ----
|
|
|
|
|
|
2025-04-26 19:53:09 +08:00
|
|
|
|
# Set max_tokens based on model
|
|
|
|
|
max_tokens = None
|
|
|
|
|
if "sonnet" in model_name.lower():
|
|
|
|
|
max_tokens = 64000
|
|
|
|
|
elif "gpt-4" in model_name.lower():
|
|
|
|
|
max_tokens = 4096
|
2025-05-10 09:58:57 +08:00
|
|
|
|
|
2025-05-21 08:39:28 +08:00
|
|
|
|
generation = trace.generation(name="thread_manager.run_thread")
|
2025-05-10 09:58:57 +08:00
|
|
|
|
try:
|
|
|
|
|
# Make the LLM call and process the response
|
|
|
|
|
response = await thread_manager.run_thread(
|
|
|
|
|
thread_id=thread_id,
|
|
|
|
|
system_prompt=system_message,
|
|
|
|
|
stream=stream,
|
|
|
|
|
llm_model=model_name,
|
|
|
|
|
llm_temperature=0,
|
|
|
|
|
llm_max_tokens=max_tokens,
|
|
|
|
|
tool_choice="auto",
|
|
|
|
|
max_xml_tool_calls=1,
|
|
|
|
|
temporary_message=temporary_message,
|
|
|
|
|
processor_config=ProcessorConfig(
|
|
|
|
|
xml_tool_calling=True,
|
|
|
|
|
native_tool_calling=False,
|
|
|
|
|
execute_tools=True,
|
|
|
|
|
execute_on_stream=True,
|
|
|
|
|
tool_execution_strategy="parallel",
|
|
|
|
|
xml_adding_strategy="user_message"
|
|
|
|
|
),
|
|
|
|
|
native_max_auto_continues=native_max_auto_continues,
|
|
|
|
|
include_xml_examples=True,
|
|
|
|
|
enable_thinking=enable_thinking,
|
|
|
|
|
reasoning_effort=reasoning_effort,
|
2025-05-21 08:39:28 +08:00
|
|
|
|
enable_context_manager=enable_context_manager,
|
|
|
|
|
generation=generation,
|
|
|
|
|
trace=trace
|
2025-05-10 09:58:57 +08:00
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
if isinstance(response, dict) and "status" in response and response["status"] == "error":
|
2025-05-10 12:37:07 +08:00
|
|
|
|
logger.error(f"Error response from run_thread: {response.get('message', 'Unknown error')}")
|
2025-05-10 09:58:57 +08:00
|
|
|
|
yield response
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
# Track if we see ask, complete, or web-browser-takeover tool calls
|
|
|
|
|
last_tool_call = None
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-05-10 09:58:57 +08:00
|
|
|
|
# Process the response
|
|
|
|
|
error_detected = False
|
|
|
|
|
try:
|
2025-05-21 08:39:28 +08:00
|
|
|
|
full_response = ""
|
2025-05-10 09:58:57 +08:00
|
|
|
|
async for chunk in response:
|
|
|
|
|
# If we receive an error chunk, we should stop after this iteration
|
|
|
|
|
if isinstance(chunk, dict) and chunk.get('type') == 'status' and chunk.get('status') == 'error':
|
2025-05-10 12:37:07 +08:00
|
|
|
|
logger.error(f"Error chunk detected: {chunk.get('message', 'Unknown error')}")
|
2025-05-10 09:58:57 +08:00
|
|
|
|
error_detected = True
|
|
|
|
|
yield chunk # Forward the error chunk
|
|
|
|
|
continue # Continue processing other chunks but don't break yet
|
|
|
|
|
|
|
|
|
|
# Check for XML versions like <ask>, <complete>, or <web-browser-takeover> in assistant content chunks
|
|
|
|
|
if chunk.get('type') == 'assistant' and 'content' in chunk:
|
|
|
|
|
try:
|
|
|
|
|
# The content field might be a JSON string or object
|
|
|
|
|
content = chunk.get('content', '{}')
|
|
|
|
|
if isinstance(content, str):
|
|
|
|
|
assistant_content_json = json.loads(content)
|
|
|
|
|
else:
|
|
|
|
|
assistant_content_json = content
|
|
|
|
|
|
|
|
|
|
# The actual text content is nested within
|
|
|
|
|
assistant_text = assistant_content_json.get('content', '')
|
2025-05-21 08:39:28 +08:00
|
|
|
|
full_response += assistant_text
|
2025-05-10 09:58:57 +08:00
|
|
|
|
if isinstance(assistant_text, str): # Ensure it's a string
|
|
|
|
|
# Check for the closing tags as they signal the end of the tool usage
|
|
|
|
|
if '</ask>' in assistant_text or '</complete>' in assistant_text or '</web-browser-takeover>' in assistant_text:
|
|
|
|
|
if '</ask>' in assistant_text:
|
|
|
|
|
xml_tool = 'ask'
|
|
|
|
|
elif '</complete>' in assistant_text:
|
|
|
|
|
xml_tool = 'complete'
|
|
|
|
|
elif '</web-browser-takeover>' in assistant_text:
|
|
|
|
|
xml_tool = 'web-browser-takeover'
|
|
|
|
|
|
|
|
|
|
last_tool_call = xml_tool
|
2025-05-10 12:37:07 +08:00
|
|
|
|
logger.info(f"Agent used XML tool: {xml_tool}")
|
2025-05-10 09:58:57 +08:00
|
|
|
|
except json.JSONDecodeError:
|
|
|
|
|
# Handle cases where content might not be valid JSON
|
2025-05-10 12:37:07 +08:00
|
|
|
|
logger.warning(f"Warning: Could not parse assistant content JSON: {chunk.get('content')}")
|
2025-05-10 09:58:57 +08:00
|
|
|
|
except Exception as e:
|
2025-05-10 12:37:07 +08:00
|
|
|
|
logger.error(f"Error processing assistant chunk: {e}")
|
2025-05-10 09:58:57 +08:00
|
|
|
|
|
|
|
|
|
yield chunk
|
|
|
|
|
|
|
|
|
|
# Check if we should stop based on the last tool call or error
|
|
|
|
|
if error_detected:
|
2025-05-10 12:37:07 +08:00
|
|
|
|
logger.info(f"Stopping due to error detected in response")
|
2025-05-22 16:36:58 +08:00
|
|
|
|
generation.end(output=full_response, status_message="error_detected", level="ERROR")
|
2025-05-10 09:58:57 +08:00
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
if last_tool_call in ['ask', 'complete', 'web-browser-takeover']:
|
2025-05-10 12:37:07 +08:00
|
|
|
|
logger.info(f"Agent decided to stop with tool: {last_tool_call}")
|
2025-05-21 08:39:28 +08:00
|
|
|
|
generation.end(output=full_response, status_message="agent_stopped")
|
2025-05-10 09:58:57 +08:00
|
|
|
|
continue_execution = False
|
2025-05-21 08:39:28 +08:00
|
|
|
|
|
2025-05-10 09:58:57 +08:00
|
|
|
|
except Exception as e:
|
|
|
|
|
# Just log the error and re-raise to stop all iterations
|
|
|
|
|
error_msg = f"Error during response streaming: {str(e)}"
|
2025-05-10 12:37:07 +08:00
|
|
|
|
logger.error(f"Error: {error_msg}")
|
2025-05-22 16:36:58 +08:00
|
|
|
|
generation.end(output=full_response, status_message=error_msg, level="ERROR")
|
2025-05-10 09:58:57 +08:00
|
|
|
|
yield {
|
|
|
|
|
"type": "status",
|
|
|
|
|
"status": "error",
|
|
|
|
|
"message": error_msg
|
|
|
|
|
}
|
|
|
|
|
# Stop execution immediately on any error
|
|
|
|
|
break
|
|
|
|
|
|
|
|
|
|
except Exception as e:
|
|
|
|
|
# Just log the error and re-raise to stop all iterations
|
|
|
|
|
error_msg = f"Error running thread: {str(e)}"
|
2025-05-10 12:37:07 +08:00
|
|
|
|
logger.error(f"Error: {error_msg}")
|
2025-05-10 09:58:57 +08:00
|
|
|
|
yield {
|
|
|
|
|
"type": "status",
|
|
|
|
|
"status": "error",
|
|
|
|
|
"message": error_msg
|
|
|
|
|
}
|
|
|
|
|
# Stop execution immediately on any error
|
|
|
|
|
break
|
2025-05-21 08:39:28 +08:00
|
|
|
|
generation.end(output=full_response)
|
|
|
|
|
|
|
|
|
|
langfuse.flush() # Flush Langfuse events at the end of the run
|
|
|
|
|
|
2025-04-10 21:13:32 +08:00
|
|
|
|
|
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # TESTING
|
2025-04-04 23:06:49 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# async def test_agent():
|
|
|
|
|
# """Test function to run the agent with a sample query"""
|
|
|
|
|
# from agentpress.thread_manager import ThreadManager
|
|
|
|
|
# from services.supabase import DBConnection
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Initialize ThreadManager
|
|
|
|
|
# thread_manager = ThreadManager()
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Create a test thread directly with Postgres function
|
|
|
|
|
# client = await DBConnection().client
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# try:
|
|
|
|
|
# # Get user's personal account
|
|
|
|
|
# account_result = await client.rpc('get_personal_account').execute()
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # if not account_result.data:
|
|
|
|
|
# # print("Error: No personal account found")
|
|
|
|
|
# # return
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# account_id = "a5fe9cb6-4812-407e-a61c-fe95b7320c59"
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# if not account_id:
|
|
|
|
|
# print("Error: Could not get account ID")
|
|
|
|
|
# return
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Find or create a test project in the user's account
|
|
|
|
|
# project_result = await client.table('projects').select('*').eq('name', 'test11').eq('account_id', account_id).execute()
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# if project_result.data and len(project_result.data) > 0:
|
|
|
|
|
# # Use existing test project
|
|
|
|
|
# project_id = project_result.data[0]['project_id']
|
|
|
|
|
# print(f"\n🔄 Using existing test project: {project_id}")
|
|
|
|
|
# else:
|
|
|
|
|
# # Create new test project if none exists
|
|
|
|
|
# project_result = await client.table('projects').insert({
|
2025-04-26 19:53:09 +08:00
|
|
|
|
# "name": "test11",
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# "account_id": account_id
|
|
|
|
|
# }).execute()
|
|
|
|
|
# project_id = project_result.data[0]['project_id']
|
|
|
|
|
# print(f"\n✨ Created new test project: {project_id}")
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Create a thread for this project
|
|
|
|
|
# thread_result = await client.table('threads').insert({
|
|
|
|
|
# 'project_id': project_id,
|
|
|
|
|
# 'account_id': account_id
|
|
|
|
|
# }).execute()
|
|
|
|
|
# thread_data = thread_result.data[0] if thread_result.data else None
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# if not thread_data:
|
|
|
|
|
# print("Error: No thread data returned")
|
|
|
|
|
# return
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# thread_id = thread_data['thread_id']
|
|
|
|
|
# except Exception as e:
|
|
|
|
|
# print(f"Error setting up thread: {str(e)}")
|
|
|
|
|
# return
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# print(f"\n🤖 Agent Thread Created: {thread_id}\n")
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Interactive message input loop
|
|
|
|
|
# while True:
|
|
|
|
|
# # Get user input
|
|
|
|
|
# user_message = input("\n💬 Enter your message (or 'exit' to quit): ")
|
|
|
|
|
# if user_message.lower() == 'exit':
|
|
|
|
|
# break
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# if not user_message.strip():
|
|
|
|
|
# print("\n🔄 Running agent...\n")
|
|
|
|
|
# await process_agent_response(thread_id, project_id, thread_manager)
|
|
|
|
|
# continue
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Add the user message to the thread
|
|
|
|
|
# await thread_manager.add_message(
|
|
|
|
|
# thread_id=thread_id,
|
|
|
|
|
# type="user",
|
|
|
|
|
# content={
|
|
|
|
|
# "role": "user",
|
|
|
|
|
# "content": user_message
|
|
|
|
|
# },
|
|
|
|
|
# is_llm_message=True
|
|
|
|
|
# )
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# print("\n🔄 Running agent...\n")
|
|
|
|
|
# await process_agent_response(thread_id, project_id, thread_manager)
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# print("\n👋 Test completed. Goodbye!")
|
2025-04-09 07:24:15 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# async def process_agent_response(
|
|
|
|
|
# thread_id: str,
|
|
|
|
|
# project_id: str,
|
|
|
|
|
# thread_manager: ThreadManager,
|
|
|
|
|
# stream: bool = True,
|
|
|
|
|
# model_name: str = "anthropic/claude-3-7-sonnet-latest",
|
|
|
|
|
# enable_thinking: Optional[bool] = False,
|
|
|
|
|
# reasoning_effort: Optional[str] = 'low',
|
|
|
|
|
# enable_context_manager: bool = True
|
|
|
|
|
# ):
|
|
|
|
|
# """Process the streaming response from the agent."""
|
|
|
|
|
# chunk_counter = 0
|
|
|
|
|
# current_response = ""
|
|
|
|
|
# tool_usage_counter = 0 # Renamed from tool_call_counter as we track usage via status
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Create a test sandbox for processing with a unique test prefix to avoid conflicts with production sandboxes
|
|
|
|
|
# sandbox_pass = str(uuid4())
|
|
|
|
|
# sandbox = create_sandbox(sandbox_pass)
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Store the original ID so we can refer to it
|
|
|
|
|
# original_sandbox_id = sandbox.id
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Generate a clear test identifier
|
|
|
|
|
# test_prefix = f"test_{uuid4().hex[:8]}_"
|
|
|
|
|
# logger.info(f"Created test sandbox with ID {original_sandbox_id} and test prefix {test_prefix}")
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Log the sandbox URL for debugging
|
|
|
|
|
# print(f"\033[91mTest sandbox created: {str(sandbox.get_preview_link(6080))}/vnc_lite.html?password={sandbox_pass}\033[0m")
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# async for chunk in run_agent(
|
|
|
|
|
# thread_id=thread_id,
|
|
|
|
|
# project_id=project_id,
|
|
|
|
|
# sandbox=sandbox,
|
|
|
|
|
# stream=stream,
|
|
|
|
|
# thread_manager=thread_manager,
|
|
|
|
|
# native_max_auto_continues=25,
|
|
|
|
|
# model_name=model_name,
|
|
|
|
|
# enable_thinking=enable_thinking,
|
|
|
|
|
# reasoning_effort=reasoning_effort,
|
|
|
|
|
# enable_context_manager=enable_context_manager
|
|
|
|
|
# ):
|
|
|
|
|
# chunk_counter += 1
|
|
|
|
|
# # print(f"CHUNK: {chunk}") # Uncomment for debugging
|
2025-04-18 18:50:39 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# if chunk.get('type') == 'assistant':
|
|
|
|
|
# # Try parsing the content JSON
|
|
|
|
|
# try:
|
|
|
|
|
# # Handle content as string or object
|
|
|
|
|
# content = chunk.get('content', '{}')
|
|
|
|
|
# if isinstance(content, str):
|
|
|
|
|
# content_json = json.loads(content)
|
|
|
|
|
# else:
|
|
|
|
|
# content_json = content
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# actual_content = content_json.get('content', '')
|
|
|
|
|
# # Print the actual assistant text content as it comes
|
|
|
|
|
# if actual_content:
|
|
|
|
|
# # Check if it contains XML tool tags, if so, print the whole tag for context
|
|
|
|
|
# if '<' in actual_content and '>' in actual_content:
|
|
|
|
|
# # Avoid printing potentially huge raw content if it's not just text
|
|
|
|
|
# if len(actual_content) < 500: # Heuristic limit
|
|
|
|
|
# print(actual_content, end='', flush=True)
|
|
|
|
|
# else:
|
|
|
|
|
# # Maybe just print a summary if it's too long or contains complex XML
|
|
|
|
|
# if '</ask>' in actual_content: print("<ask>...</ask>", end='', flush=True)
|
|
|
|
|
# elif '</complete>' in actual_content: print("<complete>...</complete>", end='', flush=True)
|
|
|
|
|
# else: print("<tool_call>...</tool_call>", end='', flush=True) # Generic case
|
|
|
|
|
# else:
|
|
|
|
|
# # Regular text content
|
|
|
|
|
# print(actual_content, end='', flush=True)
|
|
|
|
|
# current_response += actual_content # Accumulate only text part
|
|
|
|
|
# except json.JSONDecodeError:
|
|
|
|
|
# # If content is not JSON (e.g., just a string chunk), print directly
|
|
|
|
|
# raw_content = chunk.get('content', '')
|
|
|
|
|
# print(raw_content, end='', flush=True)
|
|
|
|
|
# current_response += raw_content
|
|
|
|
|
# except Exception as e:
|
|
|
|
|
# print(f"\nError processing assistant chunk: {e}\n")
|
2025-04-18 18:50:39 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# elif chunk.get('type') == 'tool': # Updated from 'tool_result'
|
|
|
|
|
# # Add timestamp and format tool result nicely
|
|
|
|
|
# tool_name = "UnknownTool" # Try to get from metadata if available
|
|
|
|
|
# result_content = "No content"
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Parse metadata - handle both string and dict formats
|
|
|
|
|
# metadata = chunk.get('metadata', {})
|
|
|
|
|
# if isinstance(metadata, str):
|
|
|
|
|
# try:
|
|
|
|
|
# metadata = json.loads(metadata)
|
|
|
|
|
# except json.JSONDecodeError:
|
|
|
|
|
# metadata = {}
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# linked_assistant_msg_id = metadata.get('assistant_message_id')
|
|
|
|
|
# parsing_details = metadata.get('parsing_details')
|
|
|
|
|
# if parsing_details:
|
|
|
|
|
# tool_name = parsing_details.get('xml_tag_name', 'UnknownTool') # Get name from parsing details
|
2025-04-18 18:50:39 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# try:
|
|
|
|
|
# # Content is a JSON string or object
|
2025-04-26 19:53:09 +08:00
|
|
|
|
# content = chunk.get('content', '{}')
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# if isinstance(content, str):
|
|
|
|
|
# content_json = json.loads(content)
|
|
|
|
|
# else:
|
|
|
|
|
# content_json = content
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # The actual tool result is nested inside content.content
|
|
|
|
|
# tool_result_str = content_json.get('content', '')
|
|
|
|
|
# # Extract the actual tool result string (remove outer <tool_result> tag if present)
|
|
|
|
|
# match = re.search(rf'<{tool_name}>(.*?)</{tool_name}>', tool_result_str, re.DOTALL)
|
|
|
|
|
# if match:
|
|
|
|
|
# result_content = match.group(1).strip()
|
|
|
|
|
# # Try to parse the result string itself as JSON for pretty printing
|
|
|
|
|
# try:
|
|
|
|
|
# result_obj = json.loads(result_content)
|
|
|
|
|
# result_content = json.dumps(result_obj, indent=2)
|
|
|
|
|
# except json.JSONDecodeError:
|
|
|
|
|
# # Keep as string if not JSON
|
|
|
|
|
# pass
|
|
|
|
|
# else:
|
|
|
|
|
# # Fallback if tag extraction fails
|
|
|
|
|
# result_content = tool_result_str
|
2025-04-18 18:50:39 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# except json.JSONDecodeError:
|
|
|
|
|
# result_content = chunk.get('content', 'Error parsing tool content')
|
|
|
|
|
# except Exception as e:
|
|
|
|
|
# result_content = f"Error processing tool chunk: {e}"
|
2025-04-18 18:50:39 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# print(f"\n\n🛠️ TOOL RESULT [{tool_name}] → {result_content}")
|
2025-04-18 18:50:39 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# elif chunk.get('type') == 'status':
|
|
|
|
|
# # Log tool status changes
|
|
|
|
|
# try:
|
|
|
|
|
# # Handle content as string or object
|
|
|
|
|
# status_content = chunk.get('content', '{}')
|
|
|
|
|
# if isinstance(status_content, str):
|
|
|
|
|
# status_content = json.loads(status_content)
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# status_type = status_content.get('status_type')
|
|
|
|
|
# function_name = status_content.get('function_name', '')
|
|
|
|
|
# xml_tag_name = status_content.get('xml_tag_name', '') # Get XML tag if available
|
|
|
|
|
# tool_name = xml_tag_name or function_name # Prefer XML tag name
|
2025-04-18 18:50:39 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# if status_type == 'tool_started' and tool_name:
|
|
|
|
|
# tool_usage_counter += 1
|
|
|
|
|
# print(f"\n⏳ TOOL STARTING #{tool_usage_counter} [{tool_name}]")
|
|
|
|
|
# print(" " + "-" * 40)
|
|
|
|
|
# # Return to the current content display
|
|
|
|
|
# if current_response:
|
|
|
|
|
# print("\nContinuing response:", flush=True)
|
|
|
|
|
# print(current_response, end='', flush=True)
|
|
|
|
|
# elif status_type == 'tool_completed' and tool_name:
|
|
|
|
|
# status_emoji = "✅"
|
|
|
|
|
# print(f"\n{status_emoji} TOOL COMPLETED: {tool_name}")
|
|
|
|
|
# elif status_type == 'finish':
|
|
|
|
|
# finish_reason = status_content.get('finish_reason', '')
|
|
|
|
|
# if finish_reason:
|
|
|
|
|
# print(f"\n📌 Finished: {finish_reason}")
|
|
|
|
|
# # else: # Print other status types if needed for debugging
|
|
|
|
|
# # print(f"\nℹ️ STATUS: {chunk.get('content')}")
|
2025-04-18 18:50:39 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# except json.JSONDecodeError:
|
|
|
|
|
# print(f"\nWarning: Could not parse status content JSON: {chunk.get('content')}")
|
|
|
|
|
# except Exception as e:
|
|
|
|
|
# print(f"\nError processing status chunk: {e}")
|
2025-04-18 18:50:39 +08:00
|
|
|
|
|
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Removed elif chunk.get('type') == 'tool_call': block
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Update final message
|
|
|
|
|
# print(f"\n\n✅ Agent run completed with {tool_usage_counter} tool executions")
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Try to clean up the test sandbox if possible
|
|
|
|
|
# try:
|
|
|
|
|
# # Attempt to delete/archive the sandbox to clean up resources
|
|
|
|
|
# # Note: Actual deletion may depend on the Daytona SDK's capabilities
|
|
|
|
|
# logger.info(f"Attempting to clean up test sandbox {original_sandbox_id}")
|
|
|
|
|
# # If there's a method to archive/delete the sandbox, call it here
|
|
|
|
|
# # Example: daytona.archive_sandbox(sandbox.id)
|
|
|
|
|
# except Exception as e:
|
|
|
|
|
# logger.warning(f"Failed to clean up test sandbox {original_sandbox_id}: {str(e)}")
|
2025-04-04 23:06:49 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# if __name__ == "__main__":
|
|
|
|
|
# import asyncio
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Configure any environment variables or setup needed for testing
|
|
|
|
|
# load_dotenv() # Ensure environment variables are loaded
|
2025-04-26 19:53:09 +08:00
|
|
|
|
|
2025-04-23 16:13:07 +08:00
|
|
|
|
# # Run the test function
|
|
|
|
|
# asyncio.run(test_agent())
|