JudgeServer/server.py

158 lines
5.4 KiB
Python
Raw Normal View History

2016-09-02 12:04:29 +00:00
# coding=utf-8
from __future__ import unicode_literals
import hashlib
import json
import os
2016-09-11 02:35:08 +00:00
import shutil
2016-09-02 12:04:29 +00:00
2016-09-28 12:43:46 +00:00
import web
2016-09-02 12:04:29 +00:00
from compiler import Compiler
from config import JUDGER_WORKSPACE_BASE, TEST_CASE_DIR
from exception import TokenVerificationFailed, CompileError, SPJCompileError,JudgeClientError
2016-09-28 12:43:46 +00:00
from judge_client import JudgeClient
2016-10-04 05:24:50 +00:00
from utils import server_info, get_token, logger
2016-10-03 08:13:46 +00:00
2016-09-02 12:04:29 +00:00
2016-09-28 12:43:46 +00:00
DEBUG = os.environ.get("judger_debug") == "1"
2016-09-02 12:04:29 +00:00
class InitSubmissionEnv(object):
def __init__(self, judger_workspace, submission_id):
self.path = os.path.join(judger_workspace, submission_id)
def __enter__(self):
try:
os.mkdir(self.path)
os.chmod(self.path, 0777)
except Exception as e:
2016-10-04 05:24:50 +00:00
logger.exception(e)
raise JudgeClientError("failed to create runtime dir")
2016-09-02 12:04:29 +00:00
return self.path
def __exit__(self, exc_type, exc_val, exc_tb):
2016-09-28 12:43:46 +00:00
if not DEBUG:
try:
shutil.rmtree(self.path)
except Exception as e:
2016-10-04 05:24:50 +00:00
logger.exception(e)
raise JudgeClientError("failed to clean runtime dir")
2016-09-02 12:04:29 +00:00
class JudgeServer(object):
def pong(self):
2016-10-03 08:13:46 +00:00
data = server_info()
data["action"] = "pong"
return data
2016-09-02 12:04:29 +00:00
@property
2016-09-11 02:35:08 +00:00
def _token(self):
2016-10-03 08:13:46 +00:00
t = get_token()
2016-09-02 12:04:29 +00:00
if not t:
raise TokenVerificationFailed("token not set")
2016-09-02 12:04:29 +00:00
return hashlib.sha256(t).hexdigest()
2016-10-01 18:05:16 +00:00
def judge(self, language_config, submission_id, src, max_cpu_time, max_memory, test_case_id,
spj_version=None, spj_config=None):
2016-09-02 12:04:29 +00:00
# init
compile_config = language_config["compile"]
2016-10-01 18:05:16 +00:00
with InitSubmissionEnv(JUDGER_WORKSPACE_BASE, submission_id=str(submission_id)) as submission_dir:
2016-09-02 12:04:29 +00:00
src_path = os.path.join(submission_dir, compile_config["src_name"])
# write source code into file
with open(src_path, "w") as f:
f.write(src.encode("utf-8"))
# compile source code, return exe file path
exe_path = Compiler().compile(compile_config=compile_config,
src_path=src_path,
output_dir=submission_dir)
2016-09-11 02:35:08 +00:00
judge_client = JudgeClient(run_config=language_config["run"],
exe_path=exe_path,
max_cpu_time=max_cpu_time,
max_memory=max_memory,
2016-10-01 18:05:16 +00:00
test_case_id=str(test_case_id),
submission_dir=submission_dir,
spj_version=str(spj_version),
spj_config=spj_config)
run_result = judge_client.run()
return run_result
2016-09-02 12:04:29 +00:00
2016-09-28 13:58:00 +00:00
def compile_spj(self, spj_version, src, spj_compile_config, test_case_id):
spj_compile_config["src_name"] = spj_compile_config["src_name"].format(spj_version=spj_version)
spj_compile_config["exe_name"] = spj_compile_config["exe_name"].format(spj_version=spj_version)
2016-09-28 12:43:46 +00:00
spj_src_path = os.path.join(TEST_CASE_DIR, test_case_id, spj_compile_config["src_name"])
# if spj source code not found, then write it into file
if not os.path.exists(spj_src_path):
with open(spj_src_path, "w") as f:
2016-09-28 13:58:00 +00:00
f.write(src.encode("utf-8"))
2016-09-28 12:43:46 +00:00
try:
Compiler().compile(compile_config=spj_compile_config,
src_path=spj_src_path,
output_dir=os.path.join(TEST_CASE_DIR, test_case_id))
# turn common CompileError into SPJCompileError
except CompileError as e:
raise SPJCompileError(e.message)
return "success"
def POST(self):
token = web.ctx.env.get("HTTP_X_JUDGE_SERVER_TOKEN", None)
try:
if token != self._token:
raise TokenVerificationFailed("invalid token")
if web.data():
try:
data = json.loads(web.data())
except Exception as e:
2016-10-04 05:24:50 +00:00
logger.info(web.data())
return {"ret": "ServerError", "data": "invalid json"}
else:
data = {}
if web.ctx["path"] == "/judge":
2016-09-28 12:43:46 +00:00
callback = self.judge
elif web.ctx["path"] == "/ping":
callback = self.pong
elif web.ctx["path"] == "/compile_spj":
callback = self.compile_spj
else:
return json.dumps({"err": "InvalidMethod", "data": None})
return json.dumps({"err": None, "data": callback(**data)})
2016-10-04 05:03:54 +00:00
except (CompileError, TokenVerificationFailed, SPJCompileError, JudgeClientError) as e:
2016-10-04 05:24:50 +00:00
logger.exception(e)
ret = dict()
ret["err"] = e.__class__.__name__
ret["data"] = e.message
2016-09-27 16:45:56 +00:00
return json.dumps(ret)
2016-10-04 05:03:54 +00:00
except Exception as e:
2016-10-04 05:24:50 +00:00
logger.exception(e)
2016-10-04 05:03:54 +00:00
ret = dict()
ret["err"] = "JudgeClientError"
ret["data"] =e.__class__.__name__ + ":" + e.message
return json.dumps(ret)
2016-09-02 12:04:29 +00:00
urls = (
"/judge", "JudgeServer",
2016-09-28 12:43:46 +00:00
"/ping", "JudgeServer",
2016-09-28 13:58:00 +00:00
"/compile_spj", "JudgeServer"
)
2016-09-02 12:04:29 +00:00
2016-09-28 05:38:12 +00:00
if DEBUG:
2016-10-04 05:24:50 +00:00
logger.info("DEBUG=ON")
2016-10-03 08:13:46 +00:00
# check token
JudgeServer()._token
app = web.application(urls, globals())
wsgiapp = app.wsgifunc()
# gunicorn -w 4 -b 0.0.0.0:8080 server:wsgiapp
if __name__ == "__main__":
app.run()