# https://github.com/openai/human-eval

import io
import os
import sys
import json
import signal
import tempfile
import platform
import itertools
import contextlib
import faulthandler
import multiprocessing

import numpy as np

from tqdm import tqdm
from collections import Counter, defaultdict
from multiprocessing.managers import ListProxy
from concurrent.futures import ThreadPoolExecutor, as_completed

from logger_setup import _timestamp

from typing import List, Dict, Optional, Union


def reliability_guard(maximum_memory_bytes: Optional[int] = None):
    """
    This disables various destructive functions and prevents the generated code
    from interfering with the test (e.g. fork bomb, killing other processes,
    removing filesystem files, etc.)

    WARNING
    This function is NOT a security sandbox. Untrusted code, including, model-
    generated code, should not be blindly executed outside of one. See the
    Codex paper for more information about OpenAI's code sandbox, and proceed
    with caution.
    """

    if maximum_memory_bytes is not None:
        import resource

        resource.setrlimit(resource.RLIMIT_AS, (maximum_memory_bytes, maximum_memory_bytes))
        resource.setrlimit(resource.RLIMIT_DATA, (maximum_memory_bytes, maximum_memory_bytes))
        if not platform.uname().system == "Darwin":
            resource.setrlimit(resource.RLIMIT_STACK, (maximum_memory_bytes, maximum_memory_bytes))

    faulthandler.disable()

    import builtins

    builtins.exit = None
    builtins.quit = None

    # import os

    os.environ["OMP_NUM_THREADS"] = "1"

    os.kill = None
    os.system = None
    os.putenv = None
    os.remove = None
    os.removedirs = None
    os.rmdir = None
    os.fchdir = None
    os.setuid = None
    os.fork = None
    os.forkpty = None
    os.killpg = None
    os.rename = None
    os.renames = None
    os.truncate = None
    os.replace = None
    os.unlink = None
    os.fchmod = None
    os.fchown = None
    os.chmod = None
    os.chown = None
    os.chroot = None
    os.fchdir = None
    os.lchflags = None
    os.lchmod = None
    os.lchown = None
    os.getcwd = None
    os.chdir = None

    import shutil

    shutil.rmtree = None
    shutil.move = None
    shutil.chown = None

    import subprocess

    subprocess.Popen = None  # type: ignore

    # __builtins__["help"] = None

    import sys

    sys.modules["ipdb"] = None
    sys.modules["joblib"] = None
    sys.modules["resource"] = None
    sys.modules["psutil"] = None
    sys.modules["tkinter"] = None


@contextlib.contextmanager
def chdir(root):
    if root == ".":
        yield
        return
    cwd = os.getcwd()
    os.chdir(root)
    try:
        yield
    except BaseException as exc:
        raise exc
    finally:
        os.chdir(cwd)


class WriteOnlyStringIO(io.StringIO):
    """StringIO that throws an exception when it's read from"""

    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):
        """Returns True if the IO object can be read."""
        return False


class redirect_stdin(contextlib._RedirectStream):  # type: ignore
    _stream = "stdin"


@contextlib.contextmanager
def swallow_io():
    stream = WriteOnlyStringIO()
    with contextlib.redirect_stdout(stream):
        with contextlib.redirect_stderr(stream):
            with redirect_stdin(stream):
                yield


@contextlib.contextmanager
def create_tempdir():
    with tempfile.TemporaryDirectory() as dirname:
        with chdir(dirname):
            yield dirname


@contextlib.contextmanager
def time_limit(seconds: float):
    def signal_handler(signum, frame):
        raise TimeoutError("Timed out!")

    signal.setitimer(signal.ITIMER_REAL, seconds)
    signal.signal(signal.SIGALRM, signal_handler)
    try:
        yield
    finally:
        signal.setitimer(signal.ITIMER_REAL, 0)


CODE_PREAMBLE = """
import os
import sys
import math
import numpy as np
"""


def unsafe_execute(problem: Dict, completion: Dict, timeout: float, result_list: ListProxy):
    with create_tempdir():
        import os
        import shutil

        # needed for tempdir cleanup
        rmtree = shutil.rmtree
        rmdir = os.rmdir
        chdir = os.chdir

        reliability_guard()

        check_program = (
            CODE_PREAMBLE + "\n\n" + completion["code"] + "\n" + "\n".join(problem["test_list"])
        )

        try:
            globals = {}
            temp_io = io.StringIO()
            with contextlib.redirect_stdout(temp_io):
                with time_limit(timeout):
                    exec(check_program, globals)
                    result_list.append(("passed", temp_io.getvalue()))

        except TimeoutError:
            result_list.append(("failed", "timed out"))

        except BaseException as e:
            result_list.append(("failed", f"error: {e}"))

        shutil.rmtree = rmtree
        os.rmdir = rmdir
        os.chdir = chdir

        return


def check_correctness(
    problem: Dict, completion: Dict, timeout: float, completion_id: Optional[int] = None
) -> Dict:
    """
    Evaluates the functional correctness of a completion by running the test
    suite provided in the problem.

    :param completion_id: an optional completion ID so we can match
        the results later even if execution finishes asynchronously.
    """

    manager = multiprocessing.Manager()
    result = manager.list()

    p = multiprocessing.Process(target=unsafe_execute, args=(problem, completion, timeout, result))
    p.start()
    p.join(timeout=timeout + 1)
    if p.is_alive():
        p.kill()

    if not result:
        result.append("timed out")

    return dict(
        task_id=problem["task_id"],
        passed=result[0][0] == "passed",
        result=result[0],
        completion_id=completion_id,
    )


def estimate_pass_at_k(
    num_samples: Union[int, List[int], np.ndarray],
    num_correct: Union[List[int], np.ndarray],
    k: int,
) -> np.ndarray:
    """
    Estimates pass@k of each problem and returns them in an array.
    """

    def estimator(n: int, c: int, k: int) -> float:
        """
        Calculates 1 - comb(n - c, k) / comb(n, k).
        """
        if n - c < k:
            return 1.0
        return 1.0 - np.prod(1.0 - k / np.arange(n - c + 1, n + 1))

    if isinstance(num_samples, int):
        num_samples_it = itertools.repeat(num_samples, len(num_correct))
    else:
        assert len(num_samples) == len(num_correct)
        num_samples_it = iter(num_samples)

    return np.array([estimator(int(n), int(c), k) for n, c in zip(num_samples_it, num_correct)])


def evaluate_mbpp_functional_correctness(
    problems: Dict[int, Dict],
    completions: List[Dict],
    k: List[int] = [1, 10, 100],
    n_workers: int = 4,
    timeout: float = 5.0,
):
    with ThreadPoolExecutor(max_workers=n_workers) as executor:
        futures = []
        completion_id = Counter()
        n_samples = 0
        results = defaultdict(list)

        for completion in tqdm(completions):
            task_id = completion["task_id"]
            generation_id = completion["generation_id"]

            problem = problems[task_id]
            assert problem["task_id"] == task_id

            args = (problem, completion, timeout, generation_id)
            future = executor.submit(check_correctness, *args)
            futures.append(future)
            completion_id[task_id] += 1
            n_samples += 1

        if len(completion_id) != len(problems):
            raise RuntimeError(
                f"Some problems are not attempted ({len(completion_id)} != {len(problems)})"
            )

        print("Running test suites...")
        for future in tqdm(as_completed(futures), total=len(futures)):
            result = future.result()
            results[result["task_id"]].append((result["completion_id"], result))

    # Calculate pass@k.
    total, correct = [], []
    for result in results.values():
        result.sort()
        passed = [r[1]["passed"] for r in result]
        total.append(len(passed))
        correct.append(sum(passed))
    total = np.array(total)
    correct = np.array(correct)

    ks = k
    pass_at_k = {
        f"pass@{k}": estimate_pass_at_k(total, correct, k).mean() for k in ks if (total >= k).all()
    }

    return pass_at_k


def load_mbpp(data_path: str, start: int = 0, end: int = None):
    with open(data_path, "r") as f:
        objs = [json.loads(line) for line in f.readlines()]

    if end is None:
        end = len(objs)

    return objs[start:end]


def load_generations(data_path: str):
    with open(data_path, "r") as f:
        objs = [json.loads(line) for line in f.readlines()]

    return objs


def main():
    if len(sys.argv) != 2:
        raise ValueError("Usage: python mbpp_evaluator.py <path-to-completion-file>")

    # NOTE: MBPP task_id starts from 1, but here indexing starts from 0
    # so START=10, END=510 corresponds to loading problems [11, 510]
    MBPP_START = 10  # inclusive
    MBPP_END = 10 + 100  # exclusive
    MBPP_DATA_PATH = "dataset/mbpp/mbpp.jsonl"
    # COMPLETION_PATH = "outputs/default-codegen-350m/generations/mbpp/20250224-162456_dev_multinomial_n200_t0.2.jsonl"
    COMPLETION_PATH = sys.argv[1]
    print(f"completion path: {COMPLETION_PATH}")

    # load mbpp problems
    problems = load_mbpp(MBPP_DATA_PATH, MBPP_START, MBPP_END)
    # convert list of problems to dict of problems
    problems = {problem["task_id"]: problem for problem in problems}

    # load generated solutions
    completions = load_generations(COMPLETION_PATH)
    n_gen_problems = len(completions) // 200  # 200 per generation
    if n_gen_problems != len(problems):
        raise ValueError(
            "Number of problems does not match expected number: "
            f"{n_gen_problems} != {len(problems)}"
        )

    pass_at_ks = evaluate_mbpp_functional_correctness(problems, completions)
    print(pass_at_ks)

    OUTPUT_LOG_PATH = os.path.join(os.path.dirname(COMPLETION_PATH), "mbpp_evaluator.log")
    with open(OUTPUT_LOG_PATH, "a") as f:
        f.write(f"Run: {_timestamp()}\n")
        f.write(f"File: {os.path.basename(COMPLETION_PATH)}\n")
        f.write(json.dumps(pass_at_ks, ensure_ascii=False))
        f.write("\n\n")

    print(f"Results written to {OUTPUT_LOG_PATH}")


if __name__ == "__main__":
    main()
