#!/usr/bin/env python3
"""
Docker sandbox utilities for running agent sessions.
"""

import hashlib
import os
import time
import subprocess
import yaml

from .convert_path import convert_windows_path_to_linux


def _hash_name(name: str, length: int = 12) -> str:
    """
    Return the first `length` hex chars of the SHA-256 digest of `name`.

    The result is
    * deterministic      – same `name` ⇒ same hash every run
    * practically unique – collision probability ≪ 10⁻¹² for `length` ≥ 12
    * Docker-friendly    – lowercase hex, matches `[a-z0-9]+`
    """
    return hashlib.sha256(name.encode("utf-8")).hexdigest()[:length]


def write_logging_conf(conf_path: str) -> None:
    """
    Create a tiny PostgreSQL conf.d fragment that enables statement logging
    into CSV files.  The file is idempotent – it is only rewritten if missing.
    """
    if os.path.exists(conf_path):
        return

    os.makedirs(os.path.dirname(conf_path), exist_ok=True)
    with open(conf_path, "w", encoding="utf-8") as f:
        f.write(
            "# Automatically generated – do not edit inside the container\n"
            "logging_collector = on\n"
            "log_destination   = 'csvlog'\n"
            "log_statement     = 'all'\n"
            "log_duration      = on\n"
        )
    print(f"✔ logging.conf written to {conf_path}")


def create_docker_compose_file(
    working_dir: str,
    log_dir: str,
    compose_path: str,
    image_name: str = "webgen-agent-postgres-django:latest",
):
    """
    Generate a docker-compose.yml in `compose_path` that *always* uses
    short, deterministic component names so the host filesystem path
    never exceeds 255 characters.
    """
    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)

    readable_name  = os.path.basename(os.path.dirname(compose_path))
    name_hash      = _hash_name(readable_name)                # 12-char hex

    # Stable, < 40-char identifiers
    project_name   = f"pgproj_{name_hash}"
    container_name = f"ctr_{name_hash}"
    volume_name    = f"pgdata_{name_hash}"
    network_name   = f"net_{name_hash}"

    db_dir = os.path.join(log_dir, "db")
    os.makedirs(db_dir, exist_ok=True)

    # ---------- write the logging.conf next to the compose file -------------
    host_conf_path = os.path.join(os.path.dirname(compose_path), "logging.conf")
    write_logging_conf(host_conf_path)
    linux_host_conf_path = convert_windows_path_to_linux(host_conf_path)

    compose_content = {
        # v2+ compose files can contain a built-in project name.
        # This prevents the CLI from prepending the (long) directory name.
        "name": project_name,

        "services": {
            "workspace": {
                "container_name": container_name,
                "image": image_name,
                "tty": True,
                "stdin_open": True,
                "command": ["sleep", "infinity"],
                "volumes": [
                    f"{working_dir}:{linux_working_dir}",
                    f"{log_dir}:{linux_log_dir}",
                    # the Postgres data volume
                    f"{volume_name}:/var/lib/postgresql/14/main",
                    f"{project_root}:{linux_project_root}:ro",
                    f"{linux_host_conf_path}:/etc/postgresql/14/main/conf.d/90-logging.conf:ro",
                ],
                "environment": {
                    "DB_HOST": "localhost",
                    "DB_PORT": 5432,
                    "DB_USERNAME": "myappuser",
                    "DB_PASSWORD": "myapppassword",
                    "DB_NAME": "myapp",
                },
                "networks": [network_name],
            }
        },

        "volumes": {
            # ‘name’ forces the *actual* volume directory to be exactly
            # what we want; Compose will not prefix it.
            volume_name: {
                "name": volume_name,
                "driver": "local",
                "driver_opts": {
                    "type": "none",
                    "o": "bind",
                    "device": db_dir,
                },
            }
        },

        "networks": {
            network_name: {
                "driver": "bridge",
            }
        },
    }

    with open(compose_path, "w", encoding="utf-8") as fh:
        yaml.dump(compose_content, fh, default_flow_style=False, sort_keys=False)

    print(f"Docker Compose file written to: {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