| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | import threading | 
					
						
							|  |  |  | from datetime import datetime | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  | from pathlib import Path | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  | from typing import Optional, Union | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | from loguru import logger | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  | from sqlalchemy import exc, inspect, text | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | from sqlmodel import Session, SQLModel, and_, create_engine, select | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  | from ..datamodel import Response, Team | 
					
						
							|  |  |  | from ..teammanager import TeamManager | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | from .schema_manager import SchemaManager | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class DatabaseManager: | 
					
						
							|  |  |  |     _init_lock = threading.Lock() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-11 11:46:25 -07:00
										 |  |  |     def __init__(self, engine_uri: str, base_dir: Optional[Union[str, Path]] = None) -> None: | 
					
						
							| 
									
										
										
										
											2024-11-15 14:51:57 -08:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |         Initialize DatabaseManager with database connection settings. | 
					
						
							|  |  |  |         Does not perform any database operations. | 
					
						
							| 
									
										
										
										
											2024-11-15 14:51:57 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         Args: | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |             engine_uri: Database connection URI (e.g. sqlite:///db.sqlite3) | 
					
						
							|  |  |  |             base_dir: Base directory for migration files. If None, uses current directory | 
					
						
							| 
									
										
										
										
											2024-11-15 14:51:57 -08:00
										 |  |  |         """
 | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |         connection_args = {"check_same_thread": True} if "sqlite" in engine_uri else {} | 
					
						
							| 
									
										
										
										
											2024-11-15 14:51:57 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-11 11:46:25 -07:00
										 |  |  |         if base_dir is not None and isinstance(base_dir, str): | 
					
						
							|  |  |  |             base_dir = Path(base_dir) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |         self.engine = create_engine(engine_uri, connect_args=connection_args) | 
					
						
							|  |  |  |         self.schema_manager = SchemaManager( | 
					
						
							|  |  |  |             engine=self.engine, | 
					
						
							| 
									
										
										
										
											2024-11-15 14:51:57 -08:00
										 |  |  |             base_dir=base_dir, | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-25 07:21:08 -08:00
										 |  |  |     def _should_auto_upgrade(self) -> bool: | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Check if auto upgrade should run based on schema differences | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         needs_upgrade, _ = self.schema_manager.check_schema_status() | 
					
						
							|  |  |  |         return needs_upgrade | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |     def initialize_database(self, auto_upgrade: bool = False, force_init_alembic: bool = True) -> Response: | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Initialize database and migrations in the correct order. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Args: | 
					
						
							|  |  |  |             auto_upgrade: If True, automatically generate and apply migrations for schema changes | 
					
						
							|  |  |  |             force_init_alembic: If True, reinitialize alembic configuration even if it exists | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if not self._init_lock.acquire(blocking=False): | 
					
						
							|  |  |  |             return Response(message="Database initialization already in progress", status=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2025-03-03 13:46:57 -08:00
										 |  |  |             # Enable foreign key constraints for SQLite | 
					
						
							|  |  |  |             if "sqlite" in str(self.engine.url): | 
					
						
							|  |  |  |                 with self.engine.connect() as conn: | 
					
						
							|  |  |  |                     conn.execute(text("PRAGMA foreign_keys=ON")) | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |             inspector = inspect(self.engine) | 
					
						
							|  |  |  |             tables_exist = inspector.get_table_names() | 
					
						
							|  |  |  |             if not tables_exist: | 
					
						
							|  |  |  |                 logger.info("Creating database tables...") | 
					
						
							|  |  |  |                 SQLModel.metadata.create_all(self.engine) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if self.schema_manager.initialize_migrations(force=force_init_alembic): | 
					
						
							|  |  |  |                     return Response(message="Database initialized successfully", status=True) | 
					
						
							|  |  |  |                 return Response(message="Failed to initialize migrations", status=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Handle existing database | 
					
						
							| 
									
										
										
										
											2025-02-25 07:21:08 -08:00
										 |  |  |             if auto_upgrade or self._should_auto_upgrade(): | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                 logger.info("Checking database schema...") | 
					
						
							| 
									
										
										
										
											2025-02-25 07:21:08 -08:00
										 |  |  |                 if self.schema_manager.ensure_schema_up_to_date(): | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                     return Response(message="Database schema is up to date", status=True) | 
					
						
							|  |  |  |                 return Response(message="Database upgrade failed", status=False) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return Response(message="Database is ready", status=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         except Exception as e: | 
					
						
							|  |  |  |             error_msg = f"Database initialization failed: {str(e)}" | 
					
						
							|  |  |  |             logger.error(error_msg) | 
					
						
							|  |  |  |             return Response(message=error_msg, status=False) | 
					
						
							|  |  |  |         finally: | 
					
						
							|  |  |  |             self._init_lock.release() | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def reset_db(self, recreate_tables: bool = True): | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Reset the database by dropping all tables and optionally recreating them. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Args: | 
					
						
							|  |  |  |             recreate_tables (bool): If True, recreates the tables after dropping them. | 
					
						
							|  |  |  |                                 Set to False if you want to call create_db_and_tables() separately. | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         if not self._init_lock.acquire(blocking=False): | 
					
						
							|  |  |  |             logger.warning("Database reset already in progress") | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |             return Response(message="Database reset already in progress", status=False, data=None) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             # Dispose existing connections | 
					
						
							|  |  |  |             self.engine.dispose() | 
					
						
							|  |  |  |             with Session(self.engine) as session: | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     # Disable foreign key checks for SQLite | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                     if "sqlite" in str(self.engine.url): | 
					
						
							| 
									
										
										
										
											2025-03-14 15:02:05 -07:00
										 |  |  |                         session.exec(text("PRAGMA foreign_keys=OFF"))  # type: ignore | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     # Drop all tables | 
					
						
							|  |  |  |                     SQLModel.metadata.drop_all(self.engine) | 
					
						
							|  |  |  |                     logger.info("All tables dropped successfully") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     # Re-enable foreign key checks for SQLite | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                     if "sqlite" in str(self.engine.url): | 
					
						
							| 
									
										
										
										
											2025-03-14 15:02:05 -07:00
										 |  |  |                         session.exec(text("PRAGMA foreign_keys=ON"))  # type: ignore | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     session.commit() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 except Exception as e: | 
					
						
							|  |  |  |                     session.rollback() | 
					
						
							|  |  |  |                     raise e | 
					
						
							|  |  |  |                 finally: | 
					
						
							|  |  |  |                     session.close() | 
					
						
							|  |  |  |                     self._init_lock.release() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if recreate_tables: | 
					
						
							|  |  |  |                 logger.info("Recreating tables...") | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                 self.initialize_database(auto_upgrade=False, force_init_alembic=True) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             return Response( | 
					
						
							|  |  |  |                 message="Database reset successfully" if recreate_tables else "Database tables dropped successfully", | 
					
						
							|  |  |  |                 status=True, | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                 data=None, | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |             ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         except Exception as e: | 
					
						
							|  |  |  |             error_msg = f"Error while resetting database: {str(e)}" | 
					
						
							|  |  |  |             logger.error(error_msg) | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |             return Response(message=error_msg, status=False, data=None) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |         finally: | 
					
						
							|  |  |  |             if self._init_lock.locked(): | 
					
						
							|  |  |  |                 self._init_lock.release() | 
					
						
							|  |  |  |                 logger.info("Database reset lock released") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-25 07:21:08 -08:00
										 |  |  |     def upsert(self, model: SQLModel, return_json: bool = True) -> Response: | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |         """Create or update an entity
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Args: | 
					
						
							|  |  |  |             model (SQLModel): The model instance to create or update | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |             return_json (bool, optional): If True, returns the model as a dictionary. | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |                 If False, returns the SQLModel instance. Defaults to True. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Returns: | 
					
						
							|  |  |  |             Response: Contains status, message and data (either dict or SQLModel based on return_json) | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         status = True | 
					
						
							|  |  |  |         model_class = type(model) | 
					
						
							|  |  |  |         existing_model = None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with Session(self.engine) as session: | 
					
						
							|  |  |  |             try: | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                 existing_model = session.exec(select(model_class).where(model_class.id == model.id)).first() | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |                 if existing_model: | 
					
						
							|  |  |  |                     model.updated_at = datetime.now() | 
					
						
							|  |  |  |                     for key, value in model.model_dump().items(): | 
					
						
							|  |  |  |                         setattr(existing_model, key, value) | 
					
						
							|  |  |  |                     model = existing_model  # Use the updated existing model | 
					
						
							|  |  |  |                     session.add(model) | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     session.add(model) | 
					
						
							|  |  |  |                 session.commit() | 
					
						
							|  |  |  |                 session.refresh(model) | 
					
						
							|  |  |  |             except Exception as e: | 
					
						
							|  |  |  |                 session.rollback() | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                 logger.error("Error while updating/creating " + str(model_class.__name__) + ": " + str(e)) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |                 status = False | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return Response( | 
					
						
							|  |  |  |             message=( | 
					
						
							|  |  |  |                 f"{model_class.__name__} Updated Successfully" | 
					
						
							|  |  |  |                 if existing_model | 
					
						
							|  |  |  |                 else f"{model_class.__name__} Created Successfully" | 
					
						
							|  |  |  |             ), | 
					
						
							|  |  |  |             status=status, | 
					
						
							|  |  |  |             data=model.model_dump() if return_json else model, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def _model_to_dict(self, model_obj): | 
					
						
							|  |  |  |         return {col.name: getattr(model_obj, col.name) for col in model_obj.__table__.columns} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get( | 
					
						
							|  |  |  |         self, | 
					
						
							|  |  |  |         model_class: SQLModel, | 
					
						
							| 
									
										
										
										
											2025-03-11 11:46:25 -07:00
										 |  |  |         filters: dict | None = None, | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |         return_json: bool = False, | 
					
						
							|  |  |  |         order: str = "desc", | 
					
						
							|  |  |  |     ): | 
					
						
							|  |  |  |         """List entities""" | 
					
						
							|  |  |  |         with Session(self.engine) as session: | 
					
						
							|  |  |  |             result = [] | 
					
						
							|  |  |  |             status = True | 
					
						
							|  |  |  |             status_message = "" | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 statement = select(model_class) | 
					
						
							|  |  |  |                 if filters: | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                     conditions = [getattr(model_class, col) == value for col, value in filters.items()] | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |                     statement = statement.where(and_(*conditions)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if hasattr(model_class, "created_at") and order: | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                     order_by_clause = getattr(model_class.created_at, order)()  # Dynamically apply asc/desc | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |                     statement = statement.order_by(order_by_clause) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 items = session.exec(statement).all() | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                 result = [self._model_to_dict(item) if return_json else item for item in items] | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |                 status_message = f"{model_class.__name__} Retrieved Successfully" | 
					
						
							|  |  |  |             except Exception as e: | 
					
						
							|  |  |  |                 session.rollback() | 
					
						
							|  |  |  |                 status = False | 
					
						
							|  |  |  |                 status_message = f"Error while fetching {model_class.__name__}" | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                 logger.error("Error while getting items: " + str(model_class.__name__) + " " + str(e)) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |             return Response(message=status_message, status=status, data=result) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-02-25 07:21:08 -08:00
										 |  |  |     def delete(self, model_class: SQLModel, filters: dict = None) -> Response: | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |         """Delete an entity""" | 
					
						
							|  |  |  |         status_message = "" | 
					
						
							|  |  |  |         status = True | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         with Session(self.engine) as session: | 
					
						
							|  |  |  |             try: | 
					
						
							| 
									
										
										
										
											2025-03-03 13:46:57 -08:00
										 |  |  |                 if "sqlite" in str(self.engine.url): | 
					
						
							|  |  |  |                     session.exec(text("PRAGMA foreign_keys=ON")) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |                 statement = select(model_class) | 
					
						
							|  |  |  |                 if filters: | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  |                     conditions = [getattr(model_class, col) == value for col, value in filters.items()] | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |                     statement = statement.where(and_(*conditions)) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 rows = session.exec(statement).all() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if rows: | 
					
						
							|  |  |  |                     for row in rows: | 
					
						
							|  |  |  |                         session.delete(row) | 
					
						
							|  |  |  |                     session.commit() | 
					
						
							|  |  |  |                     status_message = f"{model_class.__name__} Deleted Successfully" | 
					
						
							|  |  |  |                 else: | 
					
						
							|  |  |  |                     status_message = "Row not found" | 
					
						
							|  |  |  |                     logger.info(f"Row with filters {filters} not found") | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             except exc.IntegrityError as e: | 
					
						
							|  |  |  |                 session.rollback() | 
					
						
							|  |  |  |                 status = False | 
					
						
							|  |  |  |                 status_message = f"Integrity error: The {model_class.__name__} is linked to another entity and cannot be deleted. {e}" | 
					
						
							|  |  |  |                 # Log the specific integrity error | 
					
						
							|  |  |  |                 logger.error(status_message) | 
					
						
							|  |  |  |             except Exception as e: | 
					
						
							|  |  |  |                 session.rollback() | 
					
						
							|  |  |  |                 status = False | 
					
						
							|  |  |  |                 status_message = f"Error while deleting: {e}" | 
					
						
							|  |  |  |                 logger.error(status_message) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return Response(message=status_message, status=status, data=None) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |     async def import_team( | 
					
						
							|  |  |  |         self, team_config: Union[str, Path, dict], user_id: str, check_exists: bool = False | 
					
						
							|  |  |  |     ) -> Response: | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             # Load config if path provided | 
					
						
							|  |  |  |             if isinstance(team_config, (str, Path)): | 
					
						
							|  |  |  |                 config = await TeamManager.load_from_file(team_config) | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 config = team_config | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # Check existence if requested | 
					
						
							|  |  |  |             if check_exists: | 
					
						
							|  |  |  |                 existing = await self._check_team_exists(config, user_id) | 
					
						
							|  |  |  |                 if existing: | 
					
						
							|  |  |  |                     return Response( | 
					
						
							|  |  |  |                         message="Identical team configuration already exists", status=True, data={"id": existing.id} | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  |                     ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |             # Store in database | 
					
						
							| 
									
										
										
										
											2025-02-28 19:23:59 -08:00
										 |  |  |             team_db = Team(user_id=user_id, component=config) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |             result = self.upsert(team_db) | 
					
						
							|  |  |  |             return result | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |         except Exception as e: | 
					
						
							|  |  |  |             logger.error(f"Failed to import team: {str(e)}") | 
					
						
							|  |  |  |             return Response(message=str(e), status=False) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |     async def import_teams_from_directory( | 
					
						
							|  |  |  |         self, directory: Union[str, Path], user_id: str, check_exists: bool = False | 
					
						
							|  |  |  |     ) -> Response: | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         Import all team configurations from a directory. | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |         Args: | 
					
						
							|  |  |  |             directory: Path to directory containing team configs | 
					
						
							|  |  |  |             user_id: User ID to associate with imported teams | 
					
						
							|  |  |  |             check_exists: Whether to check for existing teams | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |         Returns: | 
					
						
							|  |  |  |             Response containing import results for all files | 
					
						
							|  |  |  |         """
 | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             # Load all configs from directory | 
					
						
							|  |  |  |             configs = await TeamManager.load_from_directory(directory) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |             results = [] | 
					
						
							|  |  |  |             for config in configs: | 
					
						
							|  |  |  |                 try: | 
					
						
							|  |  |  |                     result = await self.import_team(team_config=config, user_id=user_id, check_exists=check_exists) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     # Add result info | 
					
						
							|  |  |  |                     results.append( | 
					
						
							|  |  |  |                         { | 
					
						
							|  |  |  |                             "status": result.status, | 
					
						
							|  |  |  |                             "message": result.message, | 
					
						
							|  |  |  |                             "id": result.data.get("id") if result.status else None, | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |                 except Exception as e: | 
					
						
							|  |  |  |                     logger.error(f"Failed to import team config: {str(e)}") | 
					
						
							|  |  |  |                     results.append({"status": False, "message": str(e), "id": None}) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |             return Response(message="Directory import complete", status=True, data=results) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |         except Exception as e: | 
					
						
							|  |  |  |             logger.error(f"Failed to import directory: {str(e)}") | 
					
						
							|  |  |  |             return Response(message=str(e), status=False) | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |     async def _check_team_exists(self, config: dict, user_id: str) -> Optional[Team]: | 
					
						
							|  |  |  |         """Check if identical team config already exists""" | 
					
						
							|  |  |  |         teams = self.get(Team, {"user_id": user_id}).data | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |         for team in teams: | 
					
						
							| 
									
										
										
										
											2025-02-28 19:23:59 -08:00
										 |  |  |             if team.component == config: | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |                 return team | 
					
						
							| 
									
										
										
										
											2024-11-26 15:39:36 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-01-30 16:24:37 -08:00
										 |  |  |         return None | 
					
						
							| 
									
										
										
										
											2024-11-09 14:32:24 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  |     async def close(self): | 
					
						
							|  |  |  |         """Close database connections and cleanup resources""" | 
					
						
							|  |  |  |         logger.info("Closing database connections...") | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             # Dispose of the SQLAlchemy engine | 
					
						
							|  |  |  |             self.engine.dispose() | 
					
						
							|  |  |  |             logger.info("Database connections closed successfully") | 
					
						
							|  |  |  |         except Exception as e: | 
					
						
							|  |  |  |             logger.error(f"Error closing database connections: {str(e)}") | 
					
						
							|  |  |  |             raise |