|
| 1 | +from __future__ import absolute_import, print_function |
| 2 | +from cyaron import IO, log |
| 3 | +from cyaron.utils import * |
| 4 | +from cyaron.consts import * |
| 5 | +from cyaron.graders import CYaRonGraders |
| 6 | +import subprocess |
| 7 | +import multiprocessing |
| 8 | +import sys |
| 9 | +from io import open |
| 10 | +import os |
| 11 | + |
| 12 | + |
| 13 | +class CompareMismatch(ValueError): |
| 14 | + def __init__(self, name, mismatch): |
| 15 | + super(CompareMismatch, self).__init__(name, mismatch) |
| 16 | + self.name = name |
| 17 | + self.mismatch = mismatch |
| 18 | + |
| 19 | + def __str__(self): |
| 20 | + return 'In program: \'{}\'. {}'.format(self.name,self.mismatch) |
| 21 | + |
| 22 | + |
| 23 | +class Compare: |
| 24 | + @staticmethod |
| 25 | + def __compare_two(name, content, std, grader): |
| 26 | + (result, info) = CYaRonGraders.invoke(grader, content, std) |
| 27 | + status = "Correct" if result else "!!!INCORRECT!!!" |
| 28 | + info = info if info is not None else "" |
| 29 | + log.debug("{}: {} {}".format(name, status, info)) |
| 30 | + if not result: |
| 31 | + raise CompareMismatch(name, info) |
| 32 | + |
| 33 | + @staticmethod |
| 34 | + def __process_file(file): |
| 35 | + if isinstance(file, IO): |
| 36 | + file.flush_buffer() |
| 37 | + file.output_file.seek(0) |
| 38 | + return file.output_filename, file.output_file.read() |
| 39 | + else: |
| 40 | + with open(file, "r", newline='\n') as f: |
| 41 | + return file, f.read() |
| 42 | + |
| 43 | + @staticmethod |
| 44 | + def __normal_max_workers(workers): |
| 45 | + if workers is None: |
| 46 | + if sys.version_info < (3, 5): |
| 47 | + cpu = multiprocessing.cpu_count() |
| 48 | + return cpu * 5 if cpu is not None else 1 |
| 49 | + return workers |
| 50 | + |
| 51 | + @classmethod |
| 52 | + def output(cls, *files, **kwargs): |
| 53 | + kwargs = unpack_kwargs('output', kwargs, ('std', ('grader', DEFAULT_GRADER), ('max_workers', -1), |
| 54 | + ('job_pool', None), ('stop_on_incorrect', None))) |
| 55 | + std = kwargs['std'] |
| 56 | + grader = kwargs['grader'] |
| 57 | + max_workers = kwargs['max_workers'] |
| 58 | + job_pool = kwargs['job_pool'] |
| 59 | + if kwargs['stop_on_incorrect'] is not None: |
| 60 | + log.warn("parameter stop_on_incorrect is deprecated and has no effect.") |
| 61 | + |
| 62 | + if (max_workers is None or max_workers >= 0) and job_pool is None: |
| 63 | + max_workers = cls.__normal_max_workers(max_workers) |
| 64 | + try: |
| 65 | + from concurrent.futures import ThreadPoolExecutor |
| 66 | + with ThreadPoolExecutor(max_workers=max_workers) as job_pool: |
| 67 | + return cls.output(*files, std=std, grader=grader, max_workers=max_workers, job_pool=job_pool) |
| 68 | + except ImportError: |
| 69 | + pass |
| 70 | + |
| 71 | + def get_std(): |
| 72 | + return cls.__process_file(std)[1] |
| 73 | + if job_pool is not None: |
| 74 | + std = job_pool.submit(get_std).result() |
| 75 | + else: |
| 76 | + std = get_std() |
| 77 | + |
| 78 | + def do(file): |
| 79 | + (file_name, content) = cls.__process_file(file) |
| 80 | + cls.__compare_two(file_name, content, std, grader) |
| 81 | + |
| 82 | + if job_pool is not None: |
| 83 | + job_pool.map(do, files) |
| 84 | + else: |
| 85 | + [x for x in map(do, files)] |
| 86 | + |
| 87 | + @classmethod |
| 88 | + def program(cls, *programs, **kwargs): |
| 89 | + kwargs = unpack_kwargs('program', kwargs, ('input', ('std', None), ('std_program', None), |
| 90 | + ('grader', DEFAULT_GRADER), ('max_workers', -1), |
| 91 | + ('job_pool', None), ('stop_on_incorrect', None))) |
| 92 | + input = kwargs['input'] |
| 93 | + std = kwargs['std'] |
| 94 | + std_program = kwargs['std_program'] |
| 95 | + grader = kwargs['grader'] |
| 96 | + max_workers = kwargs['max_workers'] |
| 97 | + job_pool = kwargs['job_pool'] |
| 98 | + if kwargs['stop_on_incorrect'] is not None: |
| 99 | + log.warn("parameter stop_on_incorrect is deprecated and has no effect.") |
| 100 | + |
| 101 | + if (max_workers is None or max_workers >= 0) and job_pool is None: |
| 102 | + max_workers = cls.__normal_max_workers(max_workers) |
| 103 | + try: |
| 104 | + from concurrent.futures import ThreadPoolExecutor |
| 105 | + with ThreadPoolExecutor(max_workers=max_workers) as job_pool: |
| 106 | + return cls.program(*programs, input=input, std=std, std_program=std_program, grader=grader, max_workers=max_workers, job_pool=job_pool) |
| 107 | + except ImportError: |
| 108 | + pass |
| 109 | + |
| 110 | + if not isinstance(input, IO): |
| 111 | + raise TypeError("expect {}, got {}".format(type(IO).__name__, type(input).__name__)) |
| 112 | + input.flush_buffer() |
| 113 | + input.input_file.seek(0) |
| 114 | + |
| 115 | + if std_program is not None: |
| 116 | + def get_std(): |
| 117 | + with open(os.dup(input.input_file.fileno()), 'r', newline='\n') as input_file: |
| 118 | + content = make_unicode(subprocess.check_output(std_program, shell=(not list_like(std_program)), stdin=input.input_file, universal_newlines=True)) |
| 119 | + input_file.seek(0) |
| 120 | + return content |
| 121 | + if job_pool is not None: |
| 122 | + std = job_pool.submit(get_std).result() |
| 123 | + else: |
| 124 | + std = get_std() |
| 125 | + elif std is not None: |
| 126 | + def get_std(): |
| 127 | + return cls.__process_file(std)[1] |
| 128 | + if job_pool is not None: |
| 129 | + std = job_pool.submit(get_std).result() |
| 130 | + else: |
| 131 | + std = get_std() |
| 132 | + else: |
| 133 | + raise TypeError('program() missing 1 required non-None keyword-only argument: \'std\' or \'std_program\'') |
| 134 | + |
| 135 | + def do(program_name): |
| 136 | + timeout = None |
| 137 | + if list_like(program_name) and len(program_name) == 2 and int_like(program_name[-1]): |
| 138 | + program_name, timeout = program_name |
| 139 | + with open(os.dup(input.input_file.fileno()), 'r', newline='\n') as input_file: |
| 140 | + if timeout is None: |
| 141 | + content = make_unicode(subprocess.check_output(program_name, shell=(not list_like(program_name)), stdin=input_file, universal_newlines=True)) |
| 142 | + else: |
| 143 | + content = make_unicode(subprocess.check_output(program_name, shell=(not list_like(program_name)), stdin=input_file, universal_newlines=True, timeout=timeout)) |
| 144 | + input_file.seek(0) |
| 145 | + cls.__compare_two(program_name, content, std, grader) |
| 146 | + |
| 147 | + if job_pool is not None: |
| 148 | + job_pool.map(do, programs) |
| 149 | + else: |
| 150 | + [x for x in map(do, programs)] |
0 commit comments