mirror of
				https://github.com/microsoft/autogen.git
				synced 2025-10-25 14:59:31 +00:00 
			
		
		
		
	 32d2a18bf1
			
		
	
	
		32d2a18bf1
		
			
		
	
	
	
	
		
			
			<!-- Thank you for your contribution! Please review https://microsoft.github.io/autogen/docs/Contribute before opening a pull request. --> <!-- Please add a reviewer to the assignee section when you create a PR. If you don't have the access to it, we will shortly find a reviewer and assign them to your PR. --> ## Why are these changes needed? https://github.com/user-attachments/assets/e160f16d-f42d-49e2-a6c6-687e4e6786f4 Enable file upload/paste as a task in AGS. Enables tasks like - Can you research and fact check the ideas in this screenshot? - Summarize this file Only text and images supported for now Underneath, it constructs TextMessage and Multimodal messages as the task. <!-- Please give a short summary of the change and the problem this solves. --> ## Related issue number <!-- For example: "Closes #1234" --> Closes #5773 ## Checks - [ ] I've included any doc changes needed for <https://microsoft.github.io/autogen/>. See <https://github.com/microsoft/autogen/blob/main/CONTRIBUTING.md> to build and test documentation locally. - [ ] I've added tests (if relevant) corresponding to the changes introduced in this PR. - [ ] I've made sure all auto checks have passed. --------- Co-authored-by: Jack Gerrits <jackgerrits@users.noreply.github.com>
		
			
				
	
	
		
			171 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			171 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import asyncio
 | |
| import json
 | |
| import logging
 | |
| import os
 | |
| import time
 | |
| from pathlib import Path
 | |
| from typing import AsyncGenerator, Callable, List, Optional, Sequence, Union
 | |
| 
 | |
| import aiofiles
 | |
| import yaml
 | |
| from autogen_agentchat.agents import UserProxyAgent
 | |
| from autogen_agentchat.base import TaskResult
 | |
| from autogen_agentchat.messages import BaseAgentEvent, BaseChatMessage
 | |
| from autogen_agentchat.teams import BaseGroupChat
 | |
| from autogen_core import EVENT_LOGGER_NAME, CancellationToken, ComponentModel
 | |
| from autogen_core.logging import LLMCallEvent
 | |
| 
 | |
| from ..datamodel.types import EnvironmentVariable, LLMCallEventMessage, TeamResult
 | |
| from ..web.managers.run_context import RunContext
 | |
| 
 | |
| logger = logging.getLogger(__name__)
 | |
| 
 | |
| 
 | |
| class RunEventLogger(logging.Handler):
 | |
|     """Event logger that queues LLMCallEvents for streaming"""
 | |
| 
 | |
|     def __init__(self):
 | |
|         super().__init__()
 | |
|         self.events = asyncio.Queue()
 | |
| 
 | |
|     def emit(self, record: logging.LogRecord):
 | |
|         if isinstance(record.msg, LLMCallEvent):
 | |
|             self.events.put_nowait(LLMCallEventMessage(content=str(record.msg)))
 | |
| 
 | |
| 
 | |
| class TeamManager:
 | |
|     """Manages team operations including loading configs and running teams"""
 | |
| 
 | |
|     def __init__(self):
 | |
|         self._team: Optional[BaseGroupChat] = None
 | |
|         self._run_context = RunContext()
 | |
| 
 | |
|     @staticmethod
 | |
|     async def load_from_file(path: Union[str, Path]) -> dict:
 | |
|         """Load team configuration from JSON/YAML file"""
 | |
|         path = Path(path)
 | |
|         if not path.exists():
 | |
|             raise FileNotFoundError(f"Config file not found: {path}")
 | |
| 
 | |
|         async with aiofiles.open(path) as f:
 | |
|             content = await f.read()
 | |
|             if path.suffix == ".json":
 | |
|                 return json.loads(content)
 | |
|             elif path.suffix in (".yml", ".yaml"):
 | |
|                 return yaml.safe_load(content)
 | |
|             raise ValueError(f"Unsupported file format: {path.suffix}")
 | |
| 
 | |
|     @staticmethod
 | |
|     async def load_from_directory(directory: Union[str, Path]) -> List[dict]:
 | |
|         """Load all team configurations from a directory"""
 | |
|         directory = Path(directory)
 | |
|         configs = []
 | |
|         valid_extensions = {".json", ".yaml", ".yml"}
 | |
| 
 | |
|         for path in directory.iterdir():
 | |
|             if path.is_file() and path.suffix.lower() in valid_extensions:
 | |
|                 try:
 | |
|                     config = await TeamManager.load_from_file(path)
 | |
|                     configs.append(config)
 | |
|                 except Exception as e:
 | |
|                     logger.error(f"Failed to load {path}: {e}")
 | |
| 
 | |
|         return configs
 | |
| 
 | |
|     async def _create_team(
 | |
|         self,
 | |
|         team_config: Union[str, Path, dict, ComponentModel],
 | |
|         input_func: Optional[Callable] = None,
 | |
|         env_vars: Optional[List[EnvironmentVariable]] = None,
 | |
|     ) -> BaseGroupChat:
 | |
|         """Create team instance from config"""
 | |
|         if isinstance(team_config, (str, Path)):
 | |
|             config = await self.load_from_file(team_config)
 | |
|         elif isinstance(team_config, dict):
 | |
|             config = team_config
 | |
|         else:
 | |
|             config = team_config.model_dump()
 | |
| 
 | |
|         # Load env vars into environment if provided
 | |
|         if env_vars:
 | |
|             logger.info("Loading environment variables")
 | |
|             for var in env_vars:
 | |
|                 os.environ[var.name] = var.value
 | |
| 
 | |
|         self._team = BaseGroupChat.load_component(config)
 | |
| 
 | |
|         for agent in self._team._participants:
 | |
|             if hasattr(agent, "input_func") and isinstance(agent, UserProxyAgent) and input_func:
 | |
|                 agent.input_func = input_func
 | |
| 
 | |
|         return self._team
 | |
| 
 | |
|     async def run_stream(
 | |
|         self,
 | |
|         task: str | BaseChatMessage | Sequence[BaseChatMessage] | None,
 | |
|         team_config: Union[str, Path, dict, ComponentModel],
 | |
|         input_func: Optional[Callable] = None,
 | |
|         cancellation_token: Optional[CancellationToken] = None,
 | |
|         env_vars: Optional[List[EnvironmentVariable]] = None,
 | |
|     ) -> AsyncGenerator[Union[BaseAgentEvent | BaseChatMessage | LLMCallEvent, BaseChatMessage, TeamResult], None]:
 | |
|         """Stream team execution results"""
 | |
|         start_time = time.time()
 | |
|         team = None
 | |
| 
 | |
|         # Setup logger correctly
 | |
|         logger = logging.getLogger(EVENT_LOGGER_NAME)
 | |
|         logger.setLevel(logging.INFO)
 | |
|         llm_event_logger = RunEventLogger()
 | |
|         logger.handlers = [llm_event_logger]  # Replace all handlers
 | |
| 
 | |
|         try:
 | |
|             team = await self._create_team(team_config, input_func, env_vars)
 | |
| 
 | |
|             async for message in team.run_stream(task=task, cancellation_token=cancellation_token):
 | |
|                 if cancellation_token and cancellation_token.is_cancelled():
 | |
|                     break
 | |
| 
 | |
|                 if isinstance(message, TaskResult):
 | |
|                     yield TeamResult(task_result=message, usage="", duration=time.time() - start_time)
 | |
|                 else:
 | |
|                     yield message
 | |
| 
 | |
|                 # Check for any LLM events
 | |
|                 while not llm_event_logger.events.empty():
 | |
|                     event = await llm_event_logger.events.get()
 | |
|                     yield event
 | |
|         finally:
 | |
|             # Cleanup - remove our handler
 | |
|             if llm_event_logger in logger.handlers:
 | |
|                 logger.handlers.remove(llm_event_logger)
 | |
| 
 | |
|             # Ensure cleanup happens
 | |
|             if team and hasattr(team, "_participants"):
 | |
|                 for agent in team._participants:
 | |
|                     if hasattr(agent, "close"):
 | |
|                         await agent.close()
 | |
| 
 | |
|     async def run(
 | |
|         self,
 | |
|         task: str | BaseChatMessage | Sequence[BaseChatMessage] | None,
 | |
|         team_config: Union[str, Path, dict, ComponentModel],
 | |
|         input_func: Optional[Callable] = None,
 | |
|         cancellation_token: Optional[CancellationToken] = None,
 | |
|         env_vars: Optional[List[EnvironmentVariable]] = None,
 | |
|     ) -> TeamResult:
 | |
|         """Run team synchronously"""
 | |
|         start_time = time.time()
 | |
|         team = None
 | |
| 
 | |
|         try:
 | |
|             team = await self._create_team(team_config, input_func, env_vars)
 | |
|             result = await team.run(task=task, cancellation_token=cancellation_token)
 | |
| 
 | |
|             return TeamResult(task_result=result, usage="", duration=time.time() - start_time)
 | |
| 
 | |
|         finally:
 | |
|             if team and hasattr(team, "_participants"):
 | |
|                 for agent in team._participants:
 | |
|                     if hasattr(agent, "close"):
 | |
|                         await agent.close()
 |