summaryrefslogtreecommitdiff
path: root/yaksh/code_server.py
diff options
context:
space:
mode:
authorPrabhu Ramachandran2017-08-20 01:06:08 +0530
committerPrabhu Ramachandran2017-08-20 01:06:08 +0530
commit46d9ccf47dd3cdc54732fdd833f0ee5b30fcd00f (patch)
treeeecedfd0201433a436bbe43e4d38cdba89c8aa5e /yaksh/code_server.py
parentd108cd03e446dcac2b4eeeaec1ea17c47b205c78 (diff)
downloadonline_test-46d9ccf47dd3cdc54732fdd833f0ee5b30fcd00f.tar.gz
online_test-46d9ccf47dd3cdc54732fdd833f0ee5b30fcd00f.tar.bz2
online_test-46d9ccf47dd3cdc54732fdd833f0ee5b30fcd00f.zip
Safely handle code checking process being killed.
While waiting for a result, if the process is not alive, it returns an error status and restarts another process to continue working.
Diffstat (limited to 'yaksh/code_server.py')
-rwxr-xr-xyaksh/code_server.py30
1 files changed, 27 insertions, 3 deletions
diff --git a/yaksh/code_server.py b/yaksh/code_server.py
index d74d35b..75dd9b2 100755
--- a/yaksh/code_server.py
+++ b/yaksh/code_server.py
@@ -44,12 +44,12 @@ def run_as_nobody():
os.seteuid(nobody.pw_uid)
-def check_code(job_queue, results):
+def check_code(pid, job_queue, results):
"""Check the code, this runs forever.
"""
while True:
uid, json_data, user_dir = job_queue.get(True)
- results[uid] = dict(status='running', result=None)
+ results[uid] = dict(status='running', pid=pid, result=None)
data = json.loads(json_data)
grader = Grader(user_dir)
result = grader.evaluate(data)
@@ -81,7 +81,7 @@ class ServerPool(object):
self.job_queue = Queue()
processes = []
for i in range(n):
- p = Process(target=check_code, args=(self.job_queue, self.results))
+ p = self._make_process(i)
processes.append(p)
self.processes = processes
self.app = self._make_app()
@@ -93,11 +93,33 @@ class ServerPool(object):
app.listen(self.my_port)
return app
+ def _make_process(self, pid):
+ return Process(
+ target=check_code, args=(pid, self.job_queue, self.results)
+ )
+
def _start_code_servers(self):
for proc in self.processes:
if proc.pid is None:
proc.start()
+ def _handle_dead_process(self, result):
+ if result.get('status') == 'running':
+ pid = result.get('pid')
+ proc = self.processes[pid]
+ if not proc.is_alive():
+ # If the processes is dead, something bad happened so
+ # restart that process.
+ new_proc = self._make_process(pid)
+ self.processes[pid] = new_proc
+ new_proc.start()
+ result['status'] = 'done'
+ result['result'] = json.dumps(dict(
+ success=False, weight=0.0,
+ error=['Process ended with exit code %s.'
+ % proc.exitcode]
+ ))
+
# Public Protocol ##########
def get_status(self):
@@ -117,6 +139,7 @@ class ServerPool(object):
def get_result(self, uid):
result = self.results.get(uid, dict(status='unknown'))
+ self._handle_dead_process(result)
if result.get('status') == 'done':
self.results.pop(uid)
return json.dumps(result)
@@ -239,6 +262,7 @@ def main(args=None):
server_pool.run()
+
if __name__ == '__main__':
args = sys.argv[1:]
main(args)