mirror of
				https://github.com/microsoft/autogen.git
				synced 2025-10-26 07:19:33 +00:00 
			
		
		
		
	
		
			
	
	
		
			202 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
		
		
			
		
	
	
			202 lines
		
	
	
		
			6.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
|   | # api/deps.py | ||
|  | from typing import Optional | ||
|  | from fastapi import Depends, HTTPException, status | ||
|  | import logging | ||
|  | from contextlib import contextmanager | ||
|  | 
 | ||
|  | from ..database import DatabaseManager | ||
|  | from .managers.connection import WebSocketManager | ||
|  | from ..teammanager import TeamManager | ||
|  | from .config import settings | ||
|  | from ..database import ConfigurationManager | ||
|  | 
 | ||
|  | logger = logging.getLogger(__name__) | ||
|  | 
 | ||
|  | # Global manager instances | ||
|  | _db_manager: Optional[DatabaseManager] = None | ||
|  | _websocket_manager: Optional[WebSocketManager] = None | ||
|  | _team_manager: Optional[TeamManager] = None | ||
|  | 
 | ||
|  | # Context manager for database sessions | ||
|  | 
 | ||
|  | 
 | ||
|  | @contextmanager | ||
|  | def get_db_context(): | ||
|  |     """Provide a transactional scope around a series of operations.""" | ||
|  |     if not _db_manager: | ||
|  |         raise HTTPException( | ||
|  |             status_code=status.HTTP_500_INTERNAL_SERVER_ERROR, | ||
|  |             detail="Database manager not initialized" | ||
|  |         ) | ||
|  |     try: | ||
|  |         yield _db_manager | ||
|  |     except Exception as e: | ||
|  |         logger.error(f"Database operation failed: {str(e)}") | ||
|  |         raise HTTPException( | ||
|  |             status_code=status.HTTP_500_INTERNAL_SERVER_ERROR, | ||
|  |             detail="Database operation failed" | ||
|  |         ) | ||
|  | 
 | ||
|  | # Dependency providers | ||
|  | 
 | ||
|  | 
 | ||
|  | async def get_db() -> DatabaseManager: | ||
|  |     """Dependency provider for database manager""" | ||
|  |     if not _db_manager: | ||
|  |         raise HTTPException( | ||
|  |             status_code=status.HTTP_500_INTERNAL_SERVER_ERROR, | ||
|  |             detail="Database manager not initialized" | ||
|  |         ) | ||
|  |     return _db_manager | ||
|  | 
 | ||
|  | 
 | ||
|  | async def get_websocket_manager() -> WebSocketManager: | ||
|  |     """Dependency provider for connection manager""" | ||
|  |     if not _websocket_manager: | ||
|  |         raise HTTPException( | ||
|  |             status_code=status.HTTP_500_INTERNAL_SERVER_ERROR, | ||
|  |             detail="Connection manager not initialized" | ||
|  |         ) | ||
|  |     return _websocket_manager | ||
|  | 
 | ||
|  | 
 | ||
|  | async def get_team_manager() -> TeamManager: | ||
|  |     """Dependency provider for team manager""" | ||
|  |     if not _team_manager: | ||
|  |         raise HTTPException( | ||
|  |             status_code=status.HTTP_500_INTERNAL_SERVER_ERROR, | ||
|  |             detail="Team manager not initialized" | ||
|  |         ) | ||
|  |     return _team_manager | ||
|  | 
 | ||
|  | # Authentication dependency | ||
|  | 
 | ||
|  | 
 | ||
|  | async def get_current_user( | ||
|  |     # Add your authentication logic here | ||
|  |     # For example: token: str = Depends(oauth2_scheme) | ||
|  | ) -> str: | ||
|  |     """
 | ||
|  |     Dependency for getting the current authenticated user. | ||
|  |     Replace with your actual authentication logic. | ||
|  |     """
 | ||
|  |     # Implement your user authentication here | ||
|  |     return "user_id"  # Replace with actual user identification | ||
|  | 
 | ||
|  | # Manager initialization and cleanup | ||
|  | 
 | ||
|  | 
 | ||
|  | async def init_managers(database_uri: str, config_dir: str) -> None: | ||
|  |     """Initialize all manager instances""" | ||
|  |     global _db_manager, _websocket_manager, _team_manager | ||
|  | 
 | ||
|  |     logger.info("Initializing managers...") | ||
|  | 
 | ||
|  |     try: | ||
|  |         # Initialize database manager | ||
|  |         _db_manager = DatabaseManager( | ||
|  |             engine_uri=database_uri, auto_upgrade=settings.UPGRADE_DATABASE) | ||
|  |         _db_manager.create_db_and_tables() | ||
|  | 
 | ||
|  |         # init default team config | ||
|  | 
 | ||
|  |         _team_config_manager = ConfigurationManager(db_manager=_db_manager) | ||
|  |         import_result = await _team_config_manager.import_directory( | ||
|  |             config_dir, settings.DEFAULT_USER_ID, check_exists=True) | ||
|  | 
 | ||
|  |         # Initialize connection manager | ||
|  |         _websocket_manager = WebSocketManager( | ||
|  |             db_manager=_db_manager | ||
|  |         ) | ||
|  |         logger.info("Connection manager initialized") | ||
|  | 
 | ||
|  |         # Initialize team manager | ||
|  |         _team_manager = TeamManager() | ||
|  |         logger.info("Team manager initialized") | ||
|  | 
 | ||
|  |     except Exception as e: | ||
|  |         logger.error(f"Failed to initialize managers: {str(e)}") | ||
|  |         await cleanup_managers()  # Cleanup any partially initialized managers | ||
|  |         raise | ||
|  | 
 | ||
|  | 
 | ||
|  | async def cleanup_managers() -> None: | ||
|  |     """Cleanup and shutdown all manager instances""" | ||
|  |     global _db_manager, _websocket_manager, _team_manager | ||
|  | 
 | ||
|  |     logger.info("Cleaning up managers...") | ||
|  | 
 | ||
|  |     # Cleanup connection manager first to ensure all active connections are closed | ||
|  |     if _websocket_manager: | ||
|  |         try: | ||
|  |             await _websocket_manager.cleanup() | ||
|  |         except Exception as e: | ||
|  |             logger.error(f"Error cleaning up connection manager: {str(e)}") | ||
|  |         finally: | ||
|  |             _websocket_manager = None | ||
|  | 
 | ||
|  |     # TeamManager doesn't need explicit cleanup since WebSocketManager handles it | ||
|  |     _team_manager = None | ||
|  | 
 | ||
|  |     # Cleanup database manager last | ||
|  |     if _db_manager: | ||
|  |         try: | ||
|  |             await _db_manager.close() | ||
|  |         except Exception as e: | ||
|  |             logger.error(f"Error cleaning up database manager: {str(e)}") | ||
|  |         finally: | ||
|  |             _db_manager = None | ||
|  | 
 | ||
|  |     logger.info("All managers cleaned up") | ||
|  | 
 | ||
|  | # Utility functions for dependency management | ||
|  | 
 | ||
|  | 
 | ||
|  | def get_manager_status() -> dict: | ||
|  |     """Get the initialization status of all managers""" | ||
|  |     return { | ||
|  |         "database_manager": _db_manager is not None, | ||
|  |         "websocket_manager": _websocket_manager is not None, | ||
|  |         "team_manager": _team_manager is not None | ||
|  |     } | ||
|  | 
 | ||
|  | # Combined dependencies | ||
|  | 
 | ||
|  | 
 | ||
|  | async def get_managers(): | ||
|  |     """Get all managers in one dependency""" | ||
|  |     return { | ||
|  |         "db": await get_db(), | ||
|  |         "connection": await get_websocket_manager(), | ||
|  |         "team": await get_team_manager() | ||
|  |     } | ||
|  | 
 | ||
|  | # Error handling for manager operations | ||
|  | 
 | ||
|  | 
 | ||
|  | class ManagerOperationError(Exception): | ||
|  |     """Custom exception for manager operation errors""" | ||
|  | 
 | ||
|  |     def __init__(self, manager_name: str, operation: str, detail: str): | ||
|  |         self.manager_name = manager_name | ||
|  |         self.operation = operation | ||
|  |         self.detail = detail | ||
|  |         super().__init__(f"{manager_name} failed during {operation}: {detail}") | ||
|  | 
 | ||
|  | # Dependency for requiring specific managers | ||
|  | 
 | ||
|  | 
 | ||
|  | def require_managers(*manager_names: str): | ||
|  |     """Decorator to require specific managers for a route""" | ||
|  |     async def dependency(): | ||
|  |         status = get_manager_status() | ||
|  |         missing = [name for name in manager_names if not status.get( | ||
|  |             f"{name}_manager")] | ||
|  |         if missing: | ||
|  |             raise HTTPException( | ||
|  |                 status_code=status.HTTP_503_SERVICE_UNAVAILABLE, | ||
|  |                 detail=f"Required managers not available: {', '.join(missing)}" | ||
|  |             ) | ||
|  |         return True | ||
|  |     return Depends(dependency) |