"""
General utility functions for the package.
"""

import os
from typing import Any, Dict, Optional, Union
import logging
from pathlib import Path
from enum import Enum

def setup_logging(
   name: str = "nltgcr",
   level: str = "INFO",
   log_dir: Optional[str] = None,
   log_file: Optional[str] = None
) -> logging.Logger:
   """
   Set up logging configuration.
   
   Args:
      name (str, optional):
         Logger name. Defaults to "nltgcr".
      level (str, optional):
         Logging level. Defaults to "INFO".
         DEBUG < INFO < WARNING < ERROR < CRITICAL.
      log_dir (str, optional):
         Directory to save log file. Defaults to None.
         If None, the log file will be saved in the folder containing the main file being executed.
      log_file (str, optional):
         Name of the log file. Defaults to None.
         If None, no log file will be created.
   
   Returns:
      logging.Logger: 
         Configured logger
   """
   logger = logging.getLogger(name)
   logger.setLevel(level)

   # Create formatters
   console_formatter = logging.Formatter(
      '%(asctime)s - %(name)s - %(levelname)s - %(message)s'
   )
   
   # Create console handler
   console_handler = logging.StreamHandler()
   console_handler.setFormatter(console_formatter)
   logger.addHandler(console_handler)
   
   # Create file handler if log_file specified
   if log_file:
      if log_dir:
         log_file = Path(log_dir) / log_file
      else:
         log_file = Path(get_file_directory()) / log_file
      file_handler = logging.FileHandler(log_file)
      file_handler.setFormatter(console_formatter)
      logger.addHandler(file_handler)
   
   return logger

def log_platform_info(logger: logging.Logger):
   """
   Log platform information including time, system, and environment.

   Args:
      logger (logging.Logger):
         Logger to use for logging
   """
   try:
      import platform
      import time
      import os
      
      logger.info(f"Time: {time.strftime('%Y-%m-%d %H:%M:%S')}")
      logger.info(f"System: {platform.system()} {platform.version()}")
      logger.info(f"Platform: {platform.platform()}")
      logger.info(f"Platform architecture: {platform.machine()}")
      logger.info(f"Python: {platform.python_version()}")
      logger.info(f"Current Working Directory: {os.getcwd()}")
   except Exception as e:
      logger.error(f"Exception: {e}")

def log_environment_info(logger: logging.Logger):
   """
   Log environment information. This is used to generate a comprehensive log file for debugging.

   Args:
      logger (logging.Logger):
         Logger to use for logging
   """
   try:
      logger.info(f"Environment Variables: {os.environ}")
      import importlib_metadata
      installed_packages = {dist.name: dist.version for dist in importlib_metadata.distributions()}
      logger.info(f"Installed packages: {installed_packages}")
   except Exception as e:
      logger.error(f"Exception: {e}")

logger = setup_logging()  # Create default logger

def is_notebook() -> bool:
   """
   Check if the code is running in a notebook environment.
   
   Returns:
      bool: 
         True if running in a notebook environment, False otherwise
   """
   try:
      shell = get_ipython().__class__.__name__
      if shell in ['ZMQInteractiveShell',  # Jupyter notebook or qtconsole
                  'Shell',                  # Google Colab
                  'SparkShell',            # Databricks
                  'AzureNotebookShell']:   # Azure Notebooks
         return True
      
      # Additional check for JupyterLab
      try:
         import IPython
         if 'IPKernelApp' in IPython.get_ipython().config:
            return True
      except:
         logger.warning("ImportError: IPython support not available")
            
      return False
   except (NameError, ImportError):  # Probably standard Python interpreter
      return False

def get_file_directory() -> Path:
   """
   Get the absolute directory path containing the main file being executed.
   
   Returns:
      Path: Absolute directory path containing the main file
   """
   try:
      # For Python scripts
      import __main__
      if hasattr(__main__, '__file__'):
         return Path(__main__.__file__).resolve().parent
            
      # For Jupyter notebooks
      if is_notebook():
         try:
            from IPython import get_ipython
            kernel = get_ipython().kernel
            if hasattr(kernel, 'session') and hasattr(kernel.session, 'config'):
               if 'notebook_path' in kernel.session.config:
                  return Path(kernel.session.config['notebook_path']).parent
                  
            # Alternative method using connection file
            import ipykernel
            try:
               from jupyter_server import serverapp
            except ImportError:
               from notebook import notebookapp as serverapp  # Fallback for Jupyter versions < 7.0.0
            import json
            import urllib.parse
            
            connection_file = ipykernel.get_connection_file()
            kernel_id = connection_file.split('kernel-')[-1].split('.')[0]
            
            servers = serverapp.list_running_servers()
            for srv in servers:
               sessions = json.load(urllib.request.urlopen(f"{srv['url']}api/sessions"))
               for session in sessions:
                  if session['kernel']['id'] == kernel_id:
                     notebook_path = session['notebook']['path']
                     return Path(srv['notebook_dir']) / Path(notebook_path).parent
                     
         except Exception as e:
            logger.warning(f"Failed to get notebook path: {e}, using current working directory")
            
      # Final fallback: use current working directory
      return Path.cwd()
      
   except Exception as e:
      logger.warning(f"Could not determine file directory: {e}, using current working directory")
      return Path.cwd()

def get_git_root() -> Optional[Path]:
   """
   Get the git repository root directory.
   
   Returns:
      Optional[Path]: Path to git root if found, None otherwise
   """
   def is_git_root(path: Path) -> bool:
      return (path / '.git').exists()
   
   def find_git_root(start_path: Path) -> Optional[Path]:
      current = start_path.resolve()
      while current != current.parent:
         if is_git_root(current):
               return current
         current = current.parent
      return None

   # Try from module location
   try:
      module_path = Path(__file__).resolve()
      git_root = find_git_root(module_path)
      if git_root:
         return git_root
   except:
      pass
   
   # Try from current working directory
   try:
      git_root = find_git_root(Path.cwd())
      if git_root:
         return git_root
   except:
      pass
   
   return None

def convert_scientific_notation(obj: Any) -> Any:
   """
   Recursively convert scientific notation strings to numbers in nested structures.
   
   Args:
      obj: Any Python object that might contain scientific notation strings
      
   Returns:
      Same object with scientific notation strings converted to numbers
   """
   if isinstance(obj, dict):
      return {key: convert_scientific_notation(value) for key, value in obj.items()}
   elif isinstance(obj, list):
      return [convert_scientific_notation(item) for item in obj]
   elif isinstance(obj, str):
      try:
         # Check if string matches scientific notation pattern
         if 'e' in obj.lower() or '.' in obj:
               return float(obj)
         return int(obj)
      except ValueError:
         return obj
   return obj

def load_config(config_path: Union[str, Path]) -> Dict[str, Any]:
   """
   Load configuration from YAML or JSON file.
   Automatically converts scientific notation strings to numbers.
   
   Args:
      config_path (str or Path):
         Path to configuration file
      
   Returns:
      Dict:
         Dictionary containing configuration with proper number types
   """
   config_path = Path(config_path)
   
   # If path is not absolute, make it relative to project root
   if not config_path.is_absolute():
      config_path = get_file_directory() / config_path
   
   if not config_path.exists():
      logger.error(f"Config file not found: {config_path}")
      return {}
   
   with open(config_path, 'r') as f:
      if config_path.suffix in ['.yaml', '.yml']:
         try:
            import yaml
            config = yaml.safe_load(f)
         except Exception as e:
            logger.error(f"Exception: {e}")
            return {}
      elif config_path.suffix == '.json':
         import json
         config = json.load(f)
      else:
         logger.error(f"Unsupported config file format: {config_path.suffix}")
         return {}
   
   # Convert scientific notation strings to numbers
   return convert_scientific_notation(config)

def memory_usage() -> Dict[str, float]:
   """
   Get current memory usage.
   
   Returns:
      Dict:
         Dictionary containing memory usage statistics
   """
   try:
      import psutil
      process = psutil.Process(os.getpid())
      mem_info = process.memory_info()
   
      return {
      'rss': mem_info.rss / 1024 / 1024,  # RSS in MB
      'vms': mem_info.vms / 1024 / 1024,  # VMS in MB
         'percent': process.memory_percent()
      }
   except Exception as e:
      logger.error(f"Exception: {e}")
      return {}

def get_gpu_info() -> Optional[Dict[str, Any]]:
   """
   Get GPU information if available.
   
   Returns:
      Dict:
         Dictionary containing GPU information or None if no GPU
   """
   try:
      import torch
      if torch.cuda.is_available():
         return {
               'device_count': torch.cuda.device_count(),
               'current_device': torch.cuda.current_device(),
               'device_name': torch.cuda.get_device_name(),
               'memory_allocated': torch.cuda.memory_allocated() / 1024 / 1024,  # MB
               'memory_cached': torch.cuda.memory_reserved() / 1024 / 1024  # MB
         }
   except Exception as e:
      logger.error(f"Exception: {e}")
   return None

def timer(func):
   """
   Decorator to measure function execution time.
   
   Args:
      func (Callable):
         Function to time
      
   Returns:
      Callable:
         Wrapped function
   """
   import time
   from functools import wraps
   
   @wraps(func)
   def wrapper(*args, **kwargs):
      start_time = time.time()
      result = func(*args, **kwargs)
      end_time = time.time()
      print(f"{func.__name__} took {end_time - start_time:.4f} seconds to execute")
      return result
   return wrapper

def get_device_str() -> str:
   """
   Get string representation of current compute device.
   
   Returns:
      str:
         Device description
   """
   try:
      import torch
      if torch.cuda.is_available():
         return f"cuda:{torch.cuda.current_device()}"
      elif torch.backends.mps.is_available():
         return "mps"
      else:
         return "cpu"
   except Exception as e:
      logger.error(f"Exception: {e}")
      return "cpu"

# Example usage of the timer decorator
@timer
def example_function():
   """Example function to demonstrate timer decorator"""
   import time
   time.sleep(1)
   return "Done"

def get_project_root() -> Path:
   """
   Get the project root directory.
   Works in both standard Python and Jupyter notebooks.
   
   Returns:
      Path: Path to project root
   """
   try:
      # Try to get root from module location
      module_path = Path(__file__).resolve()
      if "site-packages" in str(module_path):
         # Package is installed in site-packages
         import foo
         return Path(foo.__file__).parent.parent
      else:
         # Package is in development mode
         return module_path.parent.parent.parent
   except:
      # Fallback for Jupyter notebooks
      import os
      current_path = Path(os.getcwd()).resolve()
      
      # Look for common project markers
      for path in [current_path] + list(current_path.parents):
         # Check for project markers
         if any((path / marker).exists() for marker in 
               ['setup.py', 'pyproject.toml', '.git', 'src/nltgcr']):
            return path
      
      # If no markers found, return current directory
      return current_path
