-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathsubmission_watcher_saver.py
170 lines (136 loc) · 6.02 KB
/
submission_watcher_saver.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
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
import os
import django
from time import sleep
from subprocess import call
from typing import List
from datetime import timedelta
from pycodestyle import Checker
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "autojudge.settings")
django.setup()
from judge import models, handler # noqa: E402
CONTENT_DIRECTORY = 'content'
TMP_DIRECTORY = 'tmp'
MONITOR_DIRECTORY = os.path.join(CONTENT_DIRECTORY, TMP_DIRECTORY)
DOCKER_IMAGE_NAME = 'autojudge_docker'
LS: List[str] = []
# Re-check the status of the submission folder if the number of unscored submissions
# is less than REFRESH_LS_TRIGGER
REFRESH_LS_TRIGGER = 5
# Sleep duration if number of unscored submissions is less than REFRESH_LS_TRIGGER
SLEEP_DUR_BEFORE_REFRESH = 10
def _compute_lint_score(report):
if len(report.lines) > 0:
score = 10.0 * (1 - report.total_errors / len(report.lines))
return max(0.0, score)
def saver(sub_id):
update_lb = False
# Based on the result populate SubmsissionTestCase table and return the result
with open(os.path.join(MONITOR_DIRECTORY, 'sub_run_' + sub_id + '.txt'), 'r') as f:
# Assumed format to sub_run_ID.txt file
# PROBLEM_CODE
# SUBMISSION_ID
# TESTCASEID VERDICT TIME MEMORY MESSAGE
# Read the output into verdict, memory and time.
lines = [line[:-1] for line in f.readlines()]
problem = lines[0]
submission = lines[1]
testcase_id, verdict, time, memory, msg = [], [], [], [], []
for line in lines[2:]:
sep = line.split(' ', maxsplit=4)
testcase_id.append(sep[0])
verdict.append(sep[1])
time.append(sep[2])
memory.append(sep[3])
with open(os.path.join(MONITOR_DIRECTORY, sep[4])) as log_file:
msg.append(str(log_file.read()))
os.remove(os.path.join(MONITOR_DIRECTORY, sep[4])) # Remove after reading
# Delete the file after reading
os.remove(os.path.join(MONITOR_DIRECTORY, 'sub_run_' + sub_id + '.txt'))
problem = models.Problem.objects.get(pk=problem)
s = models.Submission.objects.get(pk=submission)
score_received = 0
max_score = problem.max_score
for i in range(len(testcase_id)):
if verdict[i] == 'P':
score_received += max_score
st = models.SubmissionTestCase.objects.get(submission=submission,
testcase=testcase_id[i])
st.verdict = verdict[i]
st.memory_taken = int(memory[i])
st.time_taken = timedelta(seconds=float(time[i]))
if models.TestCase.objects.get(pk=testcase_id[i]).public:
st.message = msg[i] if len(msg[i]) < 1000 else msg[i][:1000] + '\\nMessage Truncated'
st.save()
s.judge_score = score_received
if s.problem.contest.enable_linter_score:
if s.file_type == '.py':
checker = Checker(
os.path.join(CONTENT_DIRECTORY,
'submissions', 'submission_{}.py'.format(submission)),
quiet=True)
checker.check_all()
s.linter_score = _compute_lint_score(checker.report)
current_final_score = s.judge_score + s.poster_score + s.linter_score
penalty_multiplier = 1.0
# If the submission crosses soft deadline
# Check if the submission has crossed the hard deadline
# If yes, penalty_multiplier = 0
# Else, penality_multiplier = 1 - num_of_days * penalty
remaining_time = problem.contest.soft_end_datetime - s.timestamp
if s.timestamp > problem.contest.soft_end_datetime:
if s.timestamp > problem.contest.hard_end_datetime:
penalty_multiplier = 0.0
else:
penalty_multiplier += remaining_time.days * problem.contest.penalty
# If num_of_days * penalty > 1.0, then the score is clamped to zero
s.final_score = max(0.0, current_final_score * penalty_multiplier)
s.save()
ppf, _ = models.PersonProblemFinalScore.objects.get_or_create(person=s.participant,
problem=problem)
if ppf.score <= s.final_score:
# <= because otherwise when someone submits for the first time and scores 0
# (s)he will not show up in leaderboard
ppf.score = s.final_score
update_lb = True
ppf.save()
if update_lb:
# Update the leaderboard only if the submission improved the final score
handler.update_leaderboard(problem.contest.pk, s.participant.email)
return True
# Move to ./content
cur_path = os.getcwd()
os.chdir(os.path.join(cur_path, CONTENT_DIRECTORY))
out = 1
while out != 0:
print("Building Docker image: {}....".format(DOCKER_IMAGE_NAME))
# Build docker image using docker run
out = call(['docker', 'build', '-t', DOCKER_IMAGE_NAME, './'])
if out != 0:
print("Build failed, retrying...")
# Move back to old directory
os.chdir(cur_path)
print("Docker image: {} built successfully!".format(DOCKER_IMAGE_NAME))
if not os.path.exists(MONITOR_DIRECTORY):
os.makedirs(MONITOR_DIRECTORY)
while True:
if len(LS) < REFRESH_LS_TRIGGER:
# Neglect .log files in tmp/; these are for error
# messages arising at any stage of the evaluation
sleep(SLEEP_DUR_BEFORE_REFRESH)
LS = [os.path.join(MONITOR_DIRECTORY, sub_file)
for sub_file in os.listdir(MONITOR_DIRECTORY) if sub_file[:-4] != '.log']
LS.sort(key=os.path.getctime)
if len(LS) > 0:
sub_file = LS[0] # The first file submission-wise
sub_id = os.path.basename(sub_file)[8:-4] # This is the submission ID
# Move to content
cur_dir = os.getcwd()
os.chdir(os.path.join(cur_dir, CONTENT_DIRECTORY))
# Run docker image
print("INFO: evaluating submission: {}".format(sub_id))
call(['docker', 'run', '--rm', '-v', '{}:/app'.format(os.getcwd()),
'-e', 'SUB_ID={}'.format(sub_id), DOCKER_IMAGE_NAME])
# Come back to parent directory
os.chdir(cur_dir)
saver(sub_id)
LS.remove(sub_file)