You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Hydro/hydro/handler/judge.js

123 lines
3.8 KiB
JavaScript

const
{ requirePerm } = require('./tools'),
{ sleep } = require('../utils'),
{ PERM_JUDGE } = require('../permission'),
record = require('../model/record'),
problem = require('../model/problem'),
bus = require('../service/bus'),
queue = require('../service/queue'),
{ GET, POST, SOCKET } = require('../service/server');
queue.assert('judge');
async function next(body) {
let rdoc = await record.get(body.rid);
let $set = {};
if (body.case) {
rdoc.cases.push(body.case);
$set.cases = rdoc.cases;
}
if (body.judge_text) {
rdoc.judgeTexts.push(body.judge_text);
$set.judgeTexts = rdoc.judgeTexts;
}
if (body.compiler_text) {
rdoc.compilerTexts.push(body.compiler_text);
$set.compilerTexts = rdoc.compilerTexts;
}
if (body.status) $set.status = body.status;
if (body.score) $set.score = body.score;
if (body.time_ms) $set.time = body.time_ms;
if (body.memory_kb) $set.memory = body.memory_kb;
rdoc = await record.update(body.rid, $set);
bus.publish('record_change', rdoc);
}
async function end(body) {
let rdoc = await record.get(body.rid);
let $set = {};
if (body.case) {
rdoc.cases.push(body.case);
$set.cases = rdoc.cases;
}
if (body.judge_text) {
rdoc.judgeTexts.push(body.judge_text);
$set.judgeTexts = rdoc.judgeTexts;
}
if (body.compiler_text) {
rdoc.compilerTexts.push(body.compiler_text);
$set.compilerTexts = rdoc.compilerTexts;
}
if (body.status) $set.status = body.status;
if (body.score) $set.score = body.score;
if (body.time_ms) $set.time = body.time_ms;
if (body.memory_kb) $set.memory = body.memory_kb;
$set.judgeAt = new Date();
$set.judger = body.judger;
rdoc = await record.update(body.rid, $set);
bus.publish('record_change', rdoc);
}
GET('/judge/noop', requirePerm(PERM_JUDGE), async ctx => {
ctx.body = {};
});
GET('/judge/fetch', requirePerm(PERM_JUDGE), async ctx => {
let rid = await queue.get('judge', false);
if (rid) {
let rdoc = await record.get(rid);
let pdoc = await problem.getById(rdoc.pid);
let task = {
event: 'judge',
rid, type: 0,
pid: rdoc.pid,
data: pdoc.data,
lang: rdoc.lang,
code: rdoc.code
};
ctx.body = { task };
}
else ctx.body = {};
});
SOCKET('/judge/conn', [requirePerm(PERM_JUDGE)], async conn => {
let isOpen = true, processing = null;
conn.on('close', async () => {
isOpen = false;
if (processing) {
await record.reset(processing);
queue.push('judge', processing);
}
});
conn.on('data', async message => {
message = JSON.parse(message);
if (message.key == 'next') await next(message);
else if (message.key == 'end') {
await end(Object.assign({ judger: conn.state.user._id }, message));
processing = null;
}
});
while (isOpen) {
if (!processing) {
let rid = await queue.get('judge');
let rdoc = await record.get(rid);
let pdoc = await problem.getById(rdoc.pid);
let task = {
event: 'judge',
rid, type: 0,
pid: rdoc.pid,
data: pdoc.data,
lang: rdoc.lang,
code: rdoc.code
};
conn.write(JSON.stringify({ task }));
processing = task.rid;
} else await sleep(100);
}
});
POST('/judge/next', requirePerm(PERM_JUDGE), async ctx => {
await next(ctx.request.body);
ctx.body = {};
});
POST('/judge/end', requirePerm(PERM_JUDGE), async ctx => {
await end(Object.assign({ judger: ctx.state.user._id }, ctx.request.body));
ctx.body = {};
});