#!/usr/bin/env python3
import subprocess
from pathlib import Path
from typing import Tuple
import pandas as pd
import matplotlib.pyplot as plt
import matplotlib

matplotlib.use('Agg')


def run_code(python_file: Path, working_dir: Path = None) -> Tuple[str, bool, str]:
    try:
        if working_dir is None:
            working_dir = python_file.parent.parent
        result = subprocess.run(
            ['uv', 'run', 'python', str(python_file)], capture_output=True, text=True, cwd=working_dir, timeout=60
        )
        if result.returncode == 0:
            return (python_file.name, True, result.stdout)
        else:
            return (python_file.name, False, f'Exit code {result.returncode}: {result.stderr}')
    except subprocess.TimeoutExpired:
        return (python_file.name, False, 'Simulation timeout (60s limit)')
    except Exception as e:
        return (python_file.name, False, f'Exception: {str(e)}')


def save_csv_output(python_file: Path, csv_content: str, output_dir: Path) -> Path:
    csv_filename = python_file.stem + '.csv'
    csv_path = output_dir / csv_filename
    output_dir.mkdir(parents=True, exist_ok=True)
    limited_csv = limit_csv_precision(csv_content)
    with open(csv_path, 'w', encoding='utf-8') as f:
        f.write(limited_csv)
    return csv_path


def limit_csv_precision(csv_content: str, max_digits: int = 9) -> str:
    return csv_content


def sample_csv_rows(csv_content: str, sample_ratio: float) -> str:
    if sample_ratio == 1.0:
        return csv_content
    if sample_ratio <= 0.0:
        raise ValueError('csv_sample_ratio must be > 0.0')
    try:
        lines = csv_content.strip().split('\n')
        if len(lines) <= 2:
            return csv_content
        header = lines[0]
        data_rows = lines[1:]
        step_size = max(1, round(1.0 / sample_ratio))
        sampled_indices = set(range(0, len(data_rows), step_size))
        sampled_indices.add(0)
        sampled_indices.add(len(data_rows) - 1)
        sampled_data_rows = [data_rows[i] for i in sorted(sampled_indices)]
        return header + '\n' + '\n'.join(sampled_data_rows)
    except Exception:
        return csv_content


def load_csv_with_precision(csv_path: Path, max_digits: int = 9) -> pd.DataFrame:
    df = pd.read_csv(csv_path)
    for col in df.columns:
        if df[col].dtype in ['float64', 'float32']:
            df[col] = df[col].round(max_digits)
    return df


def plot_csv_file(csv_file: Path, output_dir: Path = None) -> Tuple[str, bool, str]:
    try:
        df = load_csv_with_precision(csv_file)
        if output_dir is None:
            plot_dir = csv_file.parent / f'{csv_file.stem}.plot'
        else:
            plot_dir = output_dir / f'{csv_file.stem}.plot'
        plot_dir.mkdir(parents=True, exist_ok=True)
        time_col = 't' if 't' in df.columns else df.columns[0]
        time_data = df[time_col]
        variables = [col for col in df.columns if col != time_col]
        plots_created = 0
        for var in variables:
            try:
                plt.figure(figsize=(10, 6))
                plt.plot(time_data, df[var], linewidth=2)
                plt.title(f'{var} vs Time')
                plt.xlabel('Time')
                plt.ylabel(var)
                plt.grid(True, alpha=0.3)
                plt.tight_layout()
                plot_path = plot_dir / f'{var}.png'
                plt.savefig(plot_path, dpi=150, bbox_inches='tight')
                plt.close()
                plots_created += 1
            except Exception as e:
                plt.close()
                return (csv_file.name, False, f'Error plotting {var}: {str(e)}')
        return (csv_file.name, True, f'Created {plots_created} plots in {plot_dir.name}/')
    except Exception as e:
        return (csv_file.name, False, f'Error reading CSV: {str(e)}')


def run_code_and_save(
    python_file: Path, output_dir: Path, working_dir: Path = None, create_plots: bool = True
) -> Tuple[bool, Path, str]:
    (filename, success, output) = run_code(python_file, working_dir)
    if not success:
        return (False, None, output)
    try:
        csv_path = save_csv_output(python_file, output, output_dir)
        message = f'Simulation completed, saved to {csv_path}'
        if create_plots:
            (plot_filename, plot_success, plot_message) = plot_csv_file(csv_path, output_dir)
            if plot_success:
                message += f', {plot_message}'
            else:
                message += f', plot failed: {plot_message}'
        return (True, csv_path, message)
    except Exception as e:
        return (False, None, f'Error saving results: {str(e)}')


def format_step_name(step: int, total_steps: int) -> str:
    width = len(str(total_steps))
    return f'step_{step:0{width}d}'


def sanitize_filename(name: str) -> str:
    import re

    name = Path(name).stem
    sanitized = re.sub('[^a-zA-Z0-9_-]', '_', name)
    sanitized = re.sub('_+', '_', sanitized)
    return sanitized.strip('_')[:50]


def get_csv_summary(csv_path: Path) -> str:
    try:
        df = load_csv_with_precision(csv_path)
        return f'{len(df)} rows, {len(df.columns)} variables ({", ".join(df.columns)})'
    except Exception as e:
        return f'Error reading CSV: {str(e)}'
