-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathhandler.py
131 lines (111 loc) · 4.32 KB
/
handler.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
import os
import traceback
from io import StringIO
from werkzeug.contrib.cache import MemcachedCache
from config.config import Verdict, TRACEBACK_LIMIT, DEBUG
from core.case import Case
from core.exception import CompileError
from core.interaction import InteractiveRunner
from core.judge import SpecialJudge
from core.runner import CaseRunner
from core.submission import Submission
cache = MemcachedCache([os.environ.get("MEMCACHED", "localhost") + ':11211'])
def reject_with_traceback():
return {'status': 'reject', 'message': traceback.format_exc(TRACEBACK_LIMIT)}
def trace_group_dependencies(dep):
def dfs(x, graph, reachable):
reachable.add(x)
if x in graph:
for y in graph[x]:
if y not in reachable:
dfs(y, graph, reachable)
ret = dict()
if dep is None:
return ret
for x, y in dep:
if y not in ret:
ret[y] = set()
ret[y].add(x)
for start in ret.keys():
p = set()
dfs(start, ret, p)
ret[start] = p
return ret
def judge_handler(sub_fingerprint, sub_code, sub_lang,
case_list, max_time, max_memory,
checker_fingerprint='',
interactor_fingerprint=None,
run_until_complete=False,
group_list=None,
group_dependencies=None):
try:
assert group_list is None or len(group_list) == len(case_list)
# group should be like [1,1,2,2,2,3,3,3,3] and similar
# group dependencies are a list of tuples, something like [(2,1),(3,2),(3,1)]
group_dependencies = trace_group_dependencies(group_dependencies)
try:
detail = []
skipped_groups = set()
# enum is converted into value manually for json serialization
response = {'status': 'received', 'verdict': Verdict.JUDGING.value, 'detail': detail}
sum_verdict_value = Verdict.ACCEPTED.value
time_verdict = -1
memory_verdict = -1
report = StringIO()
submission = Submission(sub_lang)
submission.compile(sub_code, max(max_time * 5, 15))
if not checker_fingerprint:
checker_fingerprint = 'defaultspj'
checker = SpecialJudge.fromExistingFingerprint(checker_fingerprint)
if interactor_fingerprint:
interactor = SpecialJudge.fromExistingFingerprint(interactor_fingerprint)
case_runner = InteractiveRunner(submission, interactor, checker, max_time, max_memory, report_file=report)
else:
case_runner = CaseRunner(submission, checker, max_time, max_memory, report_file=report)
for case_idx, case_fingerprint in enumerate(case_list):
if group_list is not None:
case_result = {'group': group_list[case_idx], 'verdict': -3}
if not run_until_complete and group_list[case_idx] in skipped_groups:
detail.append(case_result)
continue
else:
case_result = dict()
case = Case(case_fingerprint)
case_result.update(case_runner.run(case))
case_result['verdict'] = case_result['verdict'].value
detail.append(case_result)
cache.set(sub_fingerprint, response, timeout=3600)
if case_result.get('time'):
time_verdict = max(time_verdict, case_result['time'])
if case_result.get('memory'):
memory_verdict = max(memory_verdict, case_result['memory'])
if case_result['verdict'] != Verdict.ACCEPTED.value:
if sum_verdict_value == Verdict.ACCEPTED.value:
sum_verdict_value = case_result['verdict']
if 'group' in case_result:
skipped_groups |= group_dependencies.get(case_result['group'], {case_result['group']})
if not run_until_complete and not group_list:
break
except CompileError as ce:
sum_verdict_value = Verdict.COMPILE_ERROR.value
response.update(message=ce.detail)
response.update(verdict=sum_verdict_value)
if time_verdict >= 0:
response.update(time=time_verdict)
if memory_verdict >= 0:
response.update(memory=memory_verdict)
cache.set('report_%s' % sub_fingerprint, report.getvalue(), timeout=1800)
except:
response = reject_with_traceback()
finally:
cache.set(sub_fingerprint, response, timeout=3600)
if not DEBUG:
try:
submission.clean()
except NameError:
pass
try:
case_runner.clean()
except NameError:
pass
return response