#!/usr/bin/env python3
"""
Docker sandbox utilities for running agent sessions.
"""

import os
import subprocess
import time
import yaml

from .convert_path import convert_windows_path_to_linux


def create_docker_compose_file(working_dir: str, log_dir: str, compose_path: str):
    """Create a generic Docker Compose file for the agent session"""
    project_root = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))))
    linux_project_root = convert_windows_path_to_linux(project_root)
    linux_log_dir = convert_windows_path_to_linux(log_dir)
    linux_working_dir = convert_windows_path_to_linux(working_dir)
    sample_id = os.path.basename(log_dir)
    
    container_name = os.path.basename(os.path.dirname(compose_path))
    db_dir = os.path.join(log_dir, "db")
    if not os.path.exists(db_dir):
        os.makedirs(db_dir)
    
    # Create the Docker Compose content with unique network name
    compose_content = {
        'services': {
            'workspace': {
                'container_name': container_name,
                'image': 'webgen-agent-postgres:latest',
                'tty': True,
                'stdin_open': True,
                'command': ['sleep', 'infinity'],
                'volumes': [
                    f'{working_dir}:{linux_working_dir}',
                    f'{log_dir}:{linux_log_dir}',
                    f'postgres_data_{container_name}:/var/lib/postgresql/14/main',
                    f'{project_root}:{linux_project_root}:ro'
                ],
                'environment': {
                    'DB_HOST': 'localhost',
                    'DB_PORT': 5432,
                    'DB_USERNAME': 'myappuser',
                    'DB_PASSWORD': 'myapppassword',
                    'DB_NAME': 'myapp'
                },
                'networks': [f'webgen_network_{container_name}']
            }
        },
        'volumes': {
            f'postgres_data_{container_name}': {
                'driver': 'local',
                'driver_opts': {
                    'type': 'none',
                    'o': 'bind',
                    'device': db_dir
                }
            }
        },
        'networks': {
            f'webgen_network_{container_name}': {
                'driver': 'bridge',
                'external': False
            }
        }
    }
    # Save the Docker Compose file
    with open(compose_path, 'w') as f:
        yaml.dump(compose_content, f, default_flow_style=False, sort_keys=False)
    
    print(f"Docker Compose file created at: {compose_path}")


def start_docker_containers(compose_path: str):
    """Start Docker containers using the compose file"""
    print("Starting Docker containers...")
    for attempt in range(3):  # Retry up to 3 times
        try:
            result = subprocess.run(["docker", "compose", "-f", compose_path, "up", "-d", "--remove-orphans"], 
                                  check=True, capture_output=True, text=True)
            print(result)
            
            print("Docker containers started successfully")
            return True
        except subprocess.CalledProcessError as e:
            error_output = e.stderr.lower()
            if "network" in error_output and "not found" in error_output and attempt < 2:
                print(f"Network error on attempt {attempt + 1}, retrying in 5 seconds...")
                print(f"Error details: {e.stderr}")
                time.sleep(5)
                continue
            else:
                print(f"Failed to start Docker containers after {attempt + 1} attempts")
                print(f"Stderr: {e.stderr}")
                print(f"Stdout: {e.stdout}")
                return False
    return False


def stop_docker_containers(compose_path: str):
    """Stop Docker containers using the compose file"""
    print("Stopping Docker containers...")
    try:
        # First try to stop gracefully
        subprocess.run(["docker", "compose", "-f", compose_path, "down", "-v", "--remove-orphans"], check=True, timeout=30)
        print("Docker containers stopped successfully")
        return True
    except subprocess.TimeoutExpired:
        print("Graceful shutdown timed out, forcing stop...")
        try:
            # Force stop if graceful shutdown fails
            subprocess.run(["docker", "compose", "-f", compose_path, "down", "-v", "--remove-orphans"], check=True, timeout=30)
            print("Docker containers force stopped successfully")
            return True
        except (subprocess.CalledProcessError, subprocess.TimeoutExpired) as e:
            print(f"Failed to stop Docker containers: {e}")
            # Try to clean up any remaining containers
            try:
                subprocess.run(["docker", "compose", "-f", compose_path, "kill"], check=True, timeout=10)
                subprocess.run(["docker", "compose", "-f", compose_path, "rm", "--stop", "-v", "--force"], check=True, timeout=10)
                print("Docker containers killed and removed")
            except Exception as cleanup_error:
                print(f"Failed to cleanup containers: {cleanup_error}")
            return False
    except subprocess.CalledProcessError as e:
        print(f"Failed to stop Docker containers: {e}")
        return False