"""
Code execution with sandbox protection.

Provides safe execution of untrusted code with timeout and resource limits.
"""
import io
import os
import signal
import tempfile
import platform
import contextlib
import faulthandler
import multiprocessing
from typing import Dict, Optional


class TimeoutError(Exception):
    """Raised when code execution times out."""
    pass


class WriteOnlyStringIO(io.StringIO):
    """StringIO that prevents reading."""

    def read(self, *args, **kwargs):
        raise IOError

    def readline(self, *args, **kwargs):
        raise IOError

    def readlines(self, *args, **kwargs):
        raise IOError

    def readable(self, *args, **kwargs):
        return False


class _RedirectStdin(contextlib._RedirectStream):
    _stream = "stdin"


@contextlib.contextmanager
def _time_limit(seconds: float):
    """Context manager for execution timeout."""
    def handler(signum, frame):
        raise TimeoutError("Execution timed out")

    signal.setitimer(signal.ITIMER_REAL, seconds)
    signal.signal(signal.SIGALRM, handler)
    try:
        yield
    finally:
        signal.setitimer(signal.ITIMER_REAL, 0)


@contextlib.contextmanager
def _swallow_io():
    """Context manager to suppress all I/O."""
    stream = WriteOnlyStringIO()
    with contextlib.redirect_stdout(stream):
        with contextlib.redirect_stderr(stream):
            with _RedirectStdin(stream):
                yield


@contextlib.contextmanager
def _chdir(path: str):
    """Context manager for temporary directory change."""
    if path == ".":
        yield
        return
    cwd = os.getcwd()
    os.chdir(path)
    try:
        yield
    finally:
        os.chdir(cwd)


@contextlib.contextmanager
def _create_tempdir():
    """Create and enter a temporary directory."""
    with tempfile.TemporaryDirectory() as dirname:
        with _chdir(dirname):
            yield dirname


def _apply_sandbox(max_memory: Optional[int] = None):
    """
    Disable dangerous system functions.

    WARNING: This is NOT a security sandbox. Use proper containerization
    for production environments.
    """
    if max_memory is not None:
        import resource
        resource.setrlimit(resource.RLIMIT_AS, (max_memory, max_memory))
        resource.setrlimit(resource.RLIMIT_DATA, (max_memory, max_memory))
        if platform.uname().system != "Darwin":
            resource.setrlimit(resource.RLIMIT_STACK, (max_memory, max_memory))

    faulthandler.disable()

    import builtins
    builtins.exit = None
    builtins.quit = None

    os.environ["OMP_NUM_THREADS"] = "1"

    # Disable dangerous os functions
    for attr in [
        "kill", "system", "remove", "removedirs", "rmdir", "fchdir",
        "setuid", "fork", "forkpty", "killpg", "rename", "renames",
        "truncate", "replace", "unlink", "fchmod", "fchown", "chmod",
        "chown", "chroot", "lchflags", "lchmod", "lchown", "getcwd", "chdir"
    ]:
        setattr(os, attr, None)

    import shutil
    shutil.rmtree = None
    shutil.move = None
    shutil.chown = None

    import subprocess
    subprocess.Popen = None

    __builtins__["help"] = None

    import sys
    for mod in ["ipdb", "joblib", "resource", "psutil", "tkinter"]:
        sys.modules[mod] = None


def check_correctness(
    task_id: int,
    completion_id: int,
    solution: str,
    timeout: float = 3.0
) -> Dict:
    """
    Execute code and check correctness.

    Args:
        task_id: Task identifier
        completion_id: Completion identifier
        solution: Code to execute
        timeout: Maximum execution time in seconds

    Returns:
        Dict with task_id, completion_id, passed, result, solution
    """
    def _execute():
        with _create_tempdir():
            import os
            import shutil

            # Save functions needed for cleanup
            rmtree = shutil.rmtree
            rmdir = os.rmdir
            chdir = os.chdir
            unlink = os.unlink

            _apply_sandbox()

            try:
                with _swallow_io():
                    with _time_limit(timeout):
                        exec(solution, {})
                result.append("passed")
            except TimeoutError:
                result.append("timed out")
            except BaseException as e:
                result.append(f"failed: {e}")

            # Restore for cleanup
            shutil.rmtree = rmtree
            os.rmdir = rmdir
            os.chdir = chdir
            os.unlink = unlink

    manager = multiprocessing.Manager()
    result = manager.list()

    proc = multiprocessing.Process(target=_execute)
    proc.start()
    proc.join(timeout=timeout + 1)

    if proc.is_alive():
        proc.kill()

    if not result:
        result.append("timed out")

    return {
        "task_id": task_id,
        "completion_id": completion_id,
        "passed": result[0] == "passed",
        "result": result[0],
        "solution": solution
    }
