# Coode based on: https://github.com/microsoft/CodeT/blob/main/CodeT/src/_execution.py
from typing import Optional, Dict
import contextlib
import faulthandler
import io
import os
import multiprocessing
import platform
import signal
import tempfile
import sys

blank_4 = ' ' * 4
blank_8 = ' ' * 8
blank_12 = ' ' * 12

    
def get_outputs(program_str, inputs, timeout):
    outputs = []
    for input in inputs:
        check_program = program_str + "\n"
        check_program += f'def check():\n'
        input = input.strip().replace('\n', f'\n{blank_12}')
        check_program += f'\n{blank_8}try:\n{blank_12}with time_limit({timeout}):\
                    \n{blank_12}{blank_4}output={input}\
                    \n{blank_8}except TimeoutException as err:\n{blank_12}output="timed out"\n \
                    \n{blank_8}except Exception as err:\n{blank_12}output="error"\n{blank_12}print(err)\n'
        check_program += f'\n{blank_8}return output'
        check_program += f'\n\noutput = check()\n'
        try: 
            exec_globals = {'time_limit': time_limit, 'TimeoutException': TimeoutException}
            #blockPrint()
            exec(check_program, exec_globals)
            #enablePrint()
            output = exec_globals['output']
        except Exception as e:
            print(e)
            output = 'error'
        outputs.append(output)
    return outputs

def run_test_case(program_str, test_case, entry_point, timeout):
    check_program = program_str + "\n" + test_case
    check_program += f'\n\nresult = False'
    check_program += f'\n\ntry:'
    check_program += f'\n{blank_4}check({entry_point})'
    check_program += f'\n{blank_4}result = True'
    check_program += f'\nexcept AssertionError as e:'
    check_program += f'\n{blank_4}result = False'
    try:
        with time_limit(timeout):
            exec_globals = {}
            blockPrint()
            exec(check_program, exec_globals)
            enablePrint()
            result = exec_globals['result']
    except Exception as e:
       result = 'error'
    return result

def test_case_ls_to_check(test_case_ls):
    check_program = 'def check(candidate):\n'
    for t in test_case_ls:
        check_program += f'{blank_4}{t}\n'
    return check_program

def test_case_ls_to_check_with_answers(test_case_ls, answers):
    check_program = 'def check(candidate):\n'
    for t, a in zip(test_case_ls, answers):
        if a == True:
            check_program += f'{blank_4}{t}\n'
        elif a == False:
            t = t.replace('assert ', 'assert not ')
            check_program += f'{blank_4}{t}\n'
        else:
            continue

    return check_program
        
def get_expected_outputs(program_str, inputs_str_ls, timeout):
    """
    Returns the ouput of the program given the inputs.
    """
    extend_timeout = timeout*len(inputs_str_ls)

    def unsafe_execute():

        with create_tempdir():

            # These system calls are needed when cleaning up tempdir.
            import os
            import shutil
            rmtree = shutil.rmtree
            rmdir = os.rmdir
            chdir = os.chdir

            # Disable functionalities that can make destructive changes to the test.
            reliability_guard()

            outputs.append(get_outputs(program_str, inputs_str_ls, timeout))


            # Needed for cleaning up.
            shutil.rmtree = rmtree
            os.rmdir = rmdir
            os.chdir = chdir

    manager = multiprocessing.Manager()
    outputs = manager.list()

    p = multiprocessing.Process(target=unsafe_execute)
    p.start()
    p.join(timeout=extend_timeout + 0.1)
    if p.is_alive():
        p.kill()
        
    if not outputs:
        outputs.append("timed out")

    return outputs[0]


def get_test_case_result(program_str, test_case, entry_point, timeout):
    """
    Evaluates the correctness of a program against a set of test cases.
    """
    extend_timeout = timeout*len(test_case)

    def unsafe_execute():

        with create_tempdir():

            # These system calls are needed when cleaning up tempdir.
            import os
            import shutil
            rmtree = shutil.rmtree
            rmdir = os.rmdir
            chdir = os.chdir

            # Disable functionalities that can make destructive changes to the test.
            reliability_guard()

            results.append(run_test_case(program_str, test_case, entry_point, timeout))


            # Needed for cleaning up.
            shutil.rmtree = rmtree
            os.rmdir = rmdir
            os.chdir = chdir

    manager = multiprocessing.Manager()
    results = manager.list()

    p = multiprocessing.Process(target=unsafe_execute)
    p.start()
    p.join(timeout=extend_timeout + 0.1)
    if p.is_alive():
        p.kill()
        
    if not results:
        results.append("timed out")

    return results[0]

def blockPrint():
    sys.stdout = open(os.devnull, 'w')

def enablePrint():
    sys.stdout = sys.__stdout__

@contextlib.contextmanager
def time_limit(seconds: float):
    def signal_handler(signum, frame):
        raise TimeoutException("Timed out!")
    signal.setitimer(signal.ITIMER_REAL, seconds)
    signal.signal(signal.SIGALRM, signal_handler)
    try:
        yield
    finally:
        signal.setitimer(signal.ITIMER_REAL, 0)


@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


class TimeoutException(Exception):
    pass


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 chdir(root):
    if root == ".":
        yield
        return
    cwd = os.getcwd()
    os.chdir(root)
    try:
        yield
    except BaseException as exc:
        raise exc
    finally:
        os.chdir(cwd)


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

