#!/usr/bin/env python3
"""Experiment file management for HeuPSRO experiments."""

import os
import json
import glob
from typing import Dict, List, Optional, Tuple
from dataclasses import dataclass
from datetime import datetime


@dataclass
class ExperimentPaths:
    """Paths for a HeuPSRO experiment."""
    experiment_dir: str
    solver_eoh_dir: str
    generator_eoh_dir: str
    psro_results_dir: str
    logs_dir: str
    
    def __post_init__(self):
        """Ensure all directories exist."""
        for path in [self.experiment_dir, self.solver_eoh_dir, self.generator_eoh_dir, 
                    self.psro_results_dir, self.logs_dir]:
            os.makedirs(path, exist_ok=True)


class ExperimentFileManager:
    """Manages file paths and retrieval for HeuPSRO experiments."""
    
    def __init__(self, experiment_dir: str):
        self.experiment_dir = os.path.abspath(experiment_dir)
        self.paths = ExperimentPaths(
            experiment_dir=self.experiment_dir,
            solver_eoh_dir=os.path.join(self.experiment_dir, "solver_eoh"),
            generator_eoh_dir=os.path.join(self.experiment_dir, "generator_eoh"),
            psro_results_dir=os.path.join(self.experiment_dir, "psro_results"),
            logs_dir=os.path.join(self.experiment_dir, "logs")
        )
        
        # Create experiment metadata
        self.metadata_file = os.path.join(self.experiment_dir, "experiment_metadata.json")
        self._save_metadata()
    
    def _save_metadata(self):
        """Save experiment metadata."""
        metadata = {
            "experiment_dir": self.experiment_dir,
            "created_at": datetime.now().isoformat(),
            "paths": {
                "solver_eoh": self.paths.solver_eoh_dir,
                "generator_eoh": self.paths.generator_eoh_dir,
                "psro_results": self.paths.psro_results_dir,
                "logs": self.paths.logs_dir
            }
        }
        with open(self.metadata_file, 'w') as f:
            json.dump(metadata, f, indent=2)
    
    def get_solver_eoh_files(self, generation: Optional[int] = None) -> Dict[str, str]:
        """Get solver EoH result files.
        
        Args:
            generation: Specific generation number, or None for latest
            
        Returns:
            Dict with file paths: {'pops': path, 'pops_best': path, 'history': path}
        """
        files = {}
        
        # Population files
        pops_pattern = os.path.join(self.paths.solver_eoh_dir, "pops", "population_generation_*.json")
        pops_files = sorted(glob.glob(pops_pattern))
        if pops_files:
            if generation is not None:
                target_file = os.path.join(self.paths.solver_eoh_dir, "pops", f"population_generation_{generation}.json")
                files['pops'] = target_file if os.path.exists(target_file) else pops_files[-1]
            else:
                files['pops'] = pops_files[-1]  # Latest
        
        # Best population files
        best_pattern = os.path.join(self.paths.solver_eoh_dir, "pops_best", "population_generation_*.json")
        best_files = sorted(glob.glob(best_pattern))
        if best_files:
            if generation is not None:
                target_file = os.path.join(self.paths.solver_eoh_dir, "pops_best", f"population_generation_{generation}.json")
                files['pops_best'] = target_file if os.path.exists(target_file) else best_files[-1]
            else:
                files['pops_best'] = best_files[-1]  # Latest
        
        # History files
        history_dir = os.path.join(self.paths.solver_eoh_dir, "history")
        if os.path.exists(history_dir):
            files['history'] = history_dir
        
        return files
    
    def get_generator_eoh_files(self, generation: Optional[int] = None) -> Dict[str, str]:
        """Get generator EoH result files.
        
        Args:
            generation: Specific generation number, or None for latest
            
        Returns:
            Dict with file paths: {'pops': path, 'pops_best': path, 'history': path}
        """
        files = {}
        
        # Population files
        pops_pattern = os.path.join(self.paths.generator_eoh_dir, "pops", "population_generation_*.json")
        pops_files = sorted(glob.glob(pops_pattern))
        if pops_files:
            if generation is not None:
                target_file = os.path.join(self.paths.generator_eoh_dir, "pops", f"population_generation_{generation}.json")
                files['pops'] = target_file if os.path.exists(target_file) else pops_files[-1]
            else:
                files['pops'] = pops_files[-1]  # Latest
        
        # Best population files
        best_pattern = os.path.join(self.paths.generator_eoh_dir, "pops_best", "population_generation_*.json")
        best_files = sorted(glob.glob(best_pattern))
        if best_files:
            if generation is not None:
                target_file = os.path.join(self.paths.generator_eoh_dir, "pops_best", f"population_generation_{generation}.json")
                files['pops_best'] = target_file if os.path.exists(target_file) else best_files[-1]
            else:
                files['pops_best'] = best_files[-1]  # Latest
        
        # History files
        history_dir = os.path.join(self.paths.generator_eoh_dir, "history")
        if os.path.exists(history_dir):
            files['history'] = history_dir
        
        return files
    
    def get_psro_files(self) -> Dict[str, str]:
        """Get PSRO result files.
        
        Returns:
            Dict with file paths: {'pools': path, 'utilities': path, 'meta': path}
        """
        files = {}
        
        # Pools file
        pools_file = os.path.join(self.paths.psro_results_dir, "pools.json")
        if os.path.exists(pools_file):
            files['pools'] = pools_file
        
        # Utilities file
        utilities_file = os.path.join(self.paths.psro_results_dir, "utilities.npy")
        if os.path.exists(utilities_file):
            files['utilities'] = utilities_file
        
        # Meta file
        meta_file = os.path.join(self.paths.psro_results_dir, "meta.json")
        if os.path.exists(meta_file):
            files['meta'] = meta_file
        
        return files
    
    def get_latest_solver_code(self) -> Optional[str]:
        """Get the latest evolved solver code.
        
        Returns:
            Best solver code string, or None if not found
        """
        files = self.get_solver_eoh_files()
        if 'pops_best' in files and os.path.exists(files['pops_best']):
            try:
                with open(files['pops_best'], 'r') as f:
                    data = json.load(f)
                    return data.get('code', '')
            except (json.JSONDecodeError, KeyError):
                pass
        return None
    
    def get_latest_generator_code(self) -> Optional[str]:
        """Get the latest evolved generator code.
        
        Returns:
            Best generator code string, or None if not found
        """
        files = self.get_generator_eoh_files()
        if 'pops_best' in files and os.path.exists(files['pops_best']):
            try:
                with open(files['pops_best'], 'r') as f:
                    data = json.load(f)
                    return data.get('code', '')
            except (json.JSONDecodeError, KeyError):
                pass
        return None
    
    def list_all_experiments(self, base_dir: str = None) -> List[Dict[str, str]]:
        """List all experiments in a directory.
        
        Args:
            base_dir: Base directory to search, defaults to parent of current experiment
            
        Returns:
            List of experiment info dicts
        """
        if base_dir is None:
            base_dir = os.path.dirname(self.experiment_dir)
        
        experiments = []
        for exp_dir in glob.glob(os.path.join(base_dir, "*")):
            if os.path.isdir(exp_dir):
                metadata_file = os.path.join(exp_dir, "experiment_metadata.json")
                if os.path.exists(metadata_file):
                    try:
                        with open(metadata_file, 'r') as f:
                            metadata = json.load(f)
                            experiments.append({
                                'experiment_dir': exp_dir,
                                'created_at': metadata.get('created_at', 'Unknown'),
                                'metadata': metadata
                            })
                    except (json.JSONDecodeError, KeyError):
                        experiments.append({
                            'experiment_dir': exp_dir,
                            'created_at': 'Unknown',
                            'metadata': {}
                        })
        
        return sorted(experiments, key=lambda x: x['created_at'], reverse=True)
    
    def get_experiment_summary(self) -> Dict[str, any]:
        """Get a summary of the current experiment.
        
        Returns:
            Dict with experiment summary information
        """
        summary = {
            'experiment_dir': self.experiment_dir,
            'solver_eoh_files': len(glob.glob(os.path.join(self.paths.solver_eoh_dir, "pops", "*.json"))),
            'generator_eoh_files': len(glob.glob(os.path.join(self.paths.generator_eoh_dir, "pops", "*.json"))),
            'psro_files': len(self.get_psro_files()),
            'latest_solver_code': self.get_latest_solver_code() is not None,
            'latest_generator_code': self.get_latest_generator_code() is not None
        }
        return summary


def create_experiment_manager(experiment_name: str, base_dir: str = "./experiments") -> ExperimentFileManager:
    """Create a new experiment manager.
    
    Args:
        experiment_name: Name of the experiment
        base_dir: Base directory for experiments
        
    Returns:
        ExperimentFileManager instance
    """
    experiment_dir = os.path.join(base_dir, experiment_name)
    return ExperimentFileManager(experiment_dir)


def load_experiment_manager(experiment_dir: str) -> ExperimentFileManager:
    """Load an existing experiment manager.
    
    Args:
        experiment_dir: Path to existing experiment directory
        
    Returns:
        ExperimentFileManager instance
    """
    return ExperimentFileManager(experiment_dir)
