import { isSafeInteger, flatten } from 'lodash'; import { ObjectID } from 'mongodb'; import paginate from '../lib/paginate'; import { checkPid, isTitle, isContent, isPid, } from '../lib/validator'; import * as file from '../model/file'; import * as problem from '../model/problem'; import * as record from '../model/record'; import * as domain from '../model/domain'; import * as user from '../model/user'; import * as solution from '../model/solution'; import * as system from '../model/system'; import { PERM, PRIV } from '../model/builtin'; import * as bus from '../service/bus'; import { Route, Connection, Handler, ConnectionHandler, Types, param, } from '../service/server'; import { NoProblemError, ProblemDataNotFoundError, BadRequestError, SolutionNotFoundError, ProblemNotFoundError, } from '../error'; import { Pdoc, User, Rdoc } from '../interface'; const parseCategory = (value: string) => { if (!value) return []; console.log(value); return flatten(value.split('+').map((e) => e.split(','))).map((e) => e.trim()); }; const parsePid = (value: string) => (isSafeInteger(value) ? parseInt(value, 10) : value); class ProblemHandler extends Handler { async __prepare() { this.checkPerm(PERM.PERM_VIEW_PROBLEM); } async cleanup() { if (this.response.template === 'problem_main.html' && this.request.json) { const { path, page, pcount, ppcount, pdocs, psdict, category, } = this.response.body; this.response.body = { title: this.renderTitle(category), fragments: (await Promise.all([ this.renderHTML('partials/problem_list.html', { page, ppcount, pcount, pdocs, psdict, }), this.renderHTML('partials/problem_stat.html', { pcount }), this.renderHTML('partials/problem_lucky.html', { category }), this.renderHTML('partials/path.html', { path }), ])).map((i) => ({ html: i })), raw: { path, page, pcount, ppcount, pdocs, psdict, category, }, }; } } } class ProblemMainHandler extends ProblemHandler { @param('page', Types.UnsignedInt, true) @param('q', Types.String, true) async get(domainId: string, page = 1, q = '') { this.response.template = 'problem_main.html'; const query: any = {}; let psdict = {}; const path: any = [ ['Hydro', 'homepage'], ['problem_main', null], ]; if (q) { q = q.toLowerCase(); const $regex = new RegExp(`\\A\\Q${q}\\E`, 'gmi'); query.title = { $regex }; path.push([q, null, null, true]); } if (!this.user.hasPerm(PERM.PERM_VIEW_PROBLEM_HIDDEN)) query.hidden = false; const [pdocs, ppcount, pcount] = await paginate( problem.getMulti(domainId, query).sort({ pid: 1 }), page, await system.get('PROBLEM_PER_PAGE'), ); if (this.user.hasPriv(PRIV.PRIV_USER_PROFILE)) { psdict = await problem.getListStatus( domainId, this.user._id, pdocs.map((pdoc) => pdoc.docId), ); } this.response.body = { path, page, pcount, ppcount, pdocs, psdict, category: q, }; } @param('pid', Types.UnsignedInt) async postStar(domainId: string, pid: number) { await problem.setStar(domainId, pid, this.user._id, true); this.back({ star: true }); } @param('pid', Types.UnsignedInt) async postUnstar(domainId: string, pid: number) { await problem.setStar(domainId, pid, this.user._id, false); this.back({ star: false }); } } class ProblemCategoryHandler extends ProblemHandler { @param('page', Types.UnsignedInt, true) @param('category', Types.String, null, parseCategory) async get(domainId: string, page = 1, category: string[]) { this.response.template = 'problem_main.html'; const q: any = { $and: [] }; for (const name of category) { q.$and.push({ $or: [ { category: { $elemMatch: { $eq: name } } }, { tag: { $elemMatch: { $eq: name } } }, ], }); } let psdict = {}; if (!this.user.hasPerm(PERM.PERM_VIEW_PROBLEM_HIDDEN)) q.hidden = false; const [pdocs, ppcount, pcount] = await paginate( problem.getMulti(domainId, q).sort({ pid: 1 }), page, await system.get('PROBLEM_PER_PAGE'), ); if (this.user.hasPriv(PRIV.PRIV_USER_PROFILE)) { psdict = await problem.getListStatus( domainId, this.user._id, pdocs.map((pdoc) => pdoc.docId), ); } const path = [ ['Hydro', 'homepage'], ['problem_main', 'problem_main'], [category, null, null, true], ]; this.response.body = { path, page, pcount, ppcount, pdocs, psdict, category: category.join('+'), }; } } class ProblemRandomHandler extends ProblemHandler { @param('category', Types.String, true, null, parseCategory) async get(domainId: string, category: string[] = []) { const q: any = category.length ? { $and: [] } : {}; for (const name of category) { if (name) { q.$and.push({ $or: [ { category: { $elemMatch: { $eq: name } } }, { tag: { $elemMatch: { $eq: name } } }, ], }); } } if (!this.user.hasPerm(PERM.PERM_VIEW_PROBLEM_HIDDEN)) q.hidden = false; const pid = await problem.random(domainId, q); if (!pid) throw new NoProblemError(); this.response.body = { pid }; this.response.redirect = this.url('problem_detail', { pid }); } } class ProblemDetailHandler extends ProblemHandler { pdoc: Pdoc; udoc: User; @param('pid', Types.String, null, parsePid) async _prepare(domainId: string, pid: number | string) { this.response.template = 'problem_detail.html'; this.pdoc = await problem.get(domainId, pid, this.user._id, true); if (this.pdoc.hidden && this.pdoc.owner !== this.user._id) { this.checkPerm(PERM.PERM_VIEW_PROBLEM_HIDDEN); } this.udoc = await user.getById(domainId, this.pdoc.owner); this.response.body = { pdoc: this.pdoc, udoc: this.udoc, title: this.pdoc.title, path: [ ['Hydro', 'homepage'], ['problem_main', 'problem_main'], [this.pdoc.title, null, true], ], }; } // eslint-disable-next-line async get(...args: any[]) { } } class ProblemSubmitHandler extends ProblemDetailHandler { @param('pid', Types.String, null, parsePid) async get(domainId: string, pid: string | number) { this.response.template = 'problem_submit.html'; const rdocs = await record .getUserInProblemMulti(domainId, this.user._id, this.pdoc.docId) .sort({ _id: -1 }) .limit(10) .toArray(); this.response.body = { path: [ ['Hydro', 'homepage'], ['problem_main', 'problem_main'], [this.pdoc.title, 'problem_detail', { pid }, true], ['problem_submit', null], ], pdoc: this.pdoc, udoc: this.udoc, rdocs, title: this.pdoc.title, }; } @param('lang', Types.String) @param('code', Types.String) async post(domainId: string, lang: string, code: string) { const rid = await record.add(domainId, { uid: this.user._id, lang, code, pid: this.pdoc.docId, }, true); const [rdoc] = await Promise.all([ record.get(domainId, rid), domain.incUserInDomain(domainId, this.user._id, 'nSubmit'), ]); bus.publish('record_change', rdoc); this.response.body = { rid }; this.response.redirect = this.url('record_detail', { rid }); } } class ProblemPretestHandler extends ProblemDetailHandler { @param('lang', Types.String) @param('code', Types.String) @param('input', Types.String, true) async post(domainId: string, lang: string, code: string, input: string = '') { this.limitRate('add_record', 3600, 100); const rid = await record.add(domainId, { uid: this.user._id, lang, code, pid: this.pdoc.docId, input, }, true); const rdoc = await record.get(domainId, rid); bus.publish('record_change', rdoc); this.response.body = { rid }; } } class ProblemPretestConnectionHandler extends ConnectionHandler { pid: string; domainId: string; @param('pid', Types.String) async prepare(domainId: string, pid: string) { const pdoc = await problem.get(domainId, pid); if (!pdoc) throw new ProblemNotFoundError(domainId, pid); this.pid = pdoc.docId.toString(); this.domainId = domainId; bus.subscribe(['record_change'], this, 'onRecordChange'); } async onRecordChange(data) { const rdoc: Rdoc = data.value; if ( rdoc.uid !== this.user._id || rdoc.pid.toString() !== this.pid || rdoc.domainId !== this.domainId ) return; if (rdoc.tid) return; this.send({ rdoc }); } async cleanup() { bus.unsubscribe(['record_change'], this, 'onRecordChange'); } } class ProblemStatisticsHandler extends ProblemDetailHandler { async get(domainId: string) { const udoc = await user.getById(domainId, this.pdoc.owner); const path = [ ['problem_main', 'problem_main'], [this.pdoc.title, 'problem_detail', { pid: this.pdoc.pid }, true], ['problem_statistics', null], ]; this.response.template = 'problem_statistics.html'; this.response.body = { pdoc: this.pdoc, udoc, path }; } } class ProblemManageHandler extends ProblemDetailHandler { async prepare() { if (this.pdoc.owner !== this.user._id) this.checkPerm(PERM.PERM_EDIT_PROBLEM); else this.checkPerm(PERM.PERM_EDIT_PROBLEM_SELF); } } class ProblemSettingsHandler extends ProblemManageHandler { @param('pid', Types.String) async get(domainId: string, pid: string) { this.response.template = 'problem_settings.html'; this.response.body.path = [ ['Hydro', 'homepage'], ['problem_main', 'problem_main'], [this.pdoc.title, 'problem_detail', { pid }, true], ['problem_settings', null], ]; } @param('pid', Types.String, null, parsePid) @param('yaml', Types.String) async postConfig(domainId: string, pid: string | number, yaml: string) { const pdoc = await problem.get(domainId, pid); await problem.edit(domainId, pdoc.docId, { config: yaml }); this.back(); } @param('pid', Types.String, null, parsePid) @param('hidden', Types.Boolean) @param('category', Types.String, true, null, parseCategory) @param('tag', Types.String, true, null, parseCategory) async postSetting( domainId: string, pid: string | number, hidden = false, category: string[] = [], tag: string[] = [], ) { const pdoc = await problem.get(domainId, pid); await problem.edit(domainId, pdoc.docId, { hidden, category, tag }); this.back(); } } class ProblemEditHandler extends ProblemManageHandler { async get({ pid }) { this.response.template = 'problem_edit.html'; this.response.body.path = [ ['Hydro', 'homepage'], ['problem_main', 'problem_main'], [this.pdoc.title, 'problem_detail', { pid }, true], ['problem_edit', null], ]; this.response.body.page_name = 'problem_edit'; } @param('title', Types.String, isTitle) @param('content', Types.String, isContent) async post(domainId: string, title: string, content: string) { const pid = checkPid(this.request.body.pid); const pdoc = await problem.get(domainId, this.request.params.pid); await problem.edit(domainId, pdoc.docId, { title, content, pid }); this.back(); } } class ProblemDataUploadHandler extends ProblemManageHandler { async prepare() { this.response.template = 'problem_upload.html'; } async get() { let md5; if (this.pdoc.data && typeof this.pdoc.data === 'object') { const f = await file.getMeta(this.pdoc.data); md5 = f.md5; } this.response.body.md5 = md5; } async post({ domainId }) { let md5; if (!this.request.files.file) throw new BadRequestError(); this.pdoc = await problem.setTestdata( domainId, this.pdoc.docId, this.request.files.file.path, ); if (this.pdoc.data && typeof this.pdoc.data === 'object') { const f = await file.getMeta(this.pdoc.data); md5 = f.md5; } this.response.body.md5 = md5; } } class ProblemDataDownloadHandler extends ProblemDetailHandler { async get({ pid }) { if (!this.user.hasPriv(PRIV.PRIV_JUDGE)) { if (this.user._id !== this.pdoc.owner) { this.checkPerm(PERM.PERM_READ_PROBLEM_DATA); } else this.checkPerm(PERM.PERM_READ_PROBLEM_DATA_SELF); } if (!this.pdoc.data) throw new ProblemDataNotFoundError(pid); else if (typeof this.pdoc.data === 'string') [, this.response.redirect] = this.pdoc.data.split('from:'); else this.response.redirect = await file.url(this.pdoc.data, `${this.pdoc.title}.zip`); } } class ProblemSolutionHandler extends ProblemDetailHandler { @param('page', Types.UnsignedInt, true) async get(domainId: string, page = 1) { this.response.template = 'problem_solution.html'; this.checkPerm(PERM.PERM_VIEW_PROBLEM_SOLUTION); const [psdocs, pcount, pscount] = await paginate( solution.getMulti(domainId, this.pdoc.docId), page, await system.get('SOLUTION_PER_PAGE'), ); const uids = [this.pdoc.owner]; const docids = []; for (const psdoc of psdocs) { docids.push(psdoc.docId); uids.push(psdoc.owner); if (psdoc.reply.length) { for (const psrdoc of psdoc.reply) uids.push(psrdoc.owner); } } const udict = await user.getList(domainId, uids); const pssdict = solution.getListStatus(domainId, docids, this.user._id); const path = [ ['problem_main', 'problem_main'], [this.pdoc.title, 'problem_detail', { pid: this.pdoc.pid }, true], ['problem_solution', null], ]; this.response.body = { path, psdocs, page, pcount, pscount, udict, pssdict, pdoc: this.pdoc, }; } @param('content', Types.String, isContent) async postSubmit(domainId: string, content: string) { this.checkPerm(PERM.PERM_CREATE_PROBLEM_SOLUTION); await solution.add(domainId, this.pdoc.docId, this.user._id, content); this.back(); } @param('content', Types.String, isContent) @param('psid', Types.ObjectID) async postEditSolution(domainId: string, content: string, psid: ObjectID) { let psdoc = await solution.get(domainId, psid); if (psdoc.owner !== this.user._id) this.checkPerm(PERM.PERM_EDIT_PROBLEM_SOLUTION); else this.checkPerm(PERM.PERM_EDIT_PROBLEM_SOLUTION_SELF); psdoc = await solution.edit(domainId, psdoc.docId, content); this.back({ psdoc }); } @param('psid', Types.ObjectID) async postDeleteSolution(domainId: string, psid: ObjectID) { const psdoc = await solution.get(domainId, psid); if (psdoc.owner !== this.user._id) this.checkPerm(PERM.PERM_DELETE_PROBLEM_SOLUTION); else this.checkPerm(PERM.PERM_DELETE_PROBLEM_SOLUTION_SELF); await solution.del(domainId, psdoc.docId); this.back(); } @param('psid', Types.ObjectID) @param('content', Types.String, isContent) async postReply(domainId: string, psid: ObjectID, content: string) { this.checkPerm(PERM.PERM_REPLY_PROBLEM_SOLUTION); const psdoc = await solution.get(domainId, psid); await solution.reply(domainId, psdoc.docId, this.user._id, content); this.back(); } @param('psid', Types.ObjectID) @param('psrid', Types.ObjectID) @param('content', Types.String, isContent) async postEditReply(domainId: string, psid: ObjectID, psrid: ObjectID, content: string) { const [psdoc, psrdoc] = await solution.getReply(domainId, psid, psrid); if ((!psdoc) || psdoc.pid !== this.pdoc.docId) throw new SolutionNotFoundError(psid); if (!(psrdoc.owner === this.user._id && this.user.hasPerm(PERM.PERM_EDIT_PROBLEM_SOLUTION_REPLY_SELF))) { this.checkPerm(PERM.PERM_EDIT_PROBLEM_SOLUTION_REPLY); } await solution.editReply(domainId, psid, psrid, content); this.back(); } @param('psid', Types.ObjectID) @param('psrid', Types.ObjectID) async postDeleteReply(domainId: string, psid: ObjectID, psrid: ObjectID) { const [psdoc, psrdoc] = await solution.getReply(domainId, psid, psrid); if ((!psdoc) || psdoc.pid !== this.pdoc.docId) throw new SolutionNotFoundError(psid); if (!(psrdoc.owner === this.user._id && this.user.hasPerm(PERM.PERM_DELETE_PROBLEM_SOLUTION_REPLY_SELF))) { this.checkPerm(PERM.PERM_DELETE_PROBLEM_SOLUTION_REPLY); } await solution.delReply(domainId, psid, psrid); this.back(); } @param('psid', Types.ObjectID) async postUpvote(domainId: string, psid: ObjectID) { const [psdoc, pssdoc] = await solution.vote(domainId, psid, this.user._id, 1); this.back({ vote: psdoc.vote, user_vote: pssdoc.vote }); } @param('psid', Types.ObjectID) async postDownvote(domainId: string, psid: ObjectID) { const [psdoc, pssdoc] = await solution.vote(domainId, psid, this.user._id, -1); this.back({ vote: psdoc.vote, user_vote: pssdoc.vote }); } } class ProblemSolutionRawHandler extends ProblemDetailHandler { @param('psid', Types.ObjectID) async get(domainId: string, psid: ObjectID) { this.checkPerm(PERM.PERM_VIEW_PROBLEM_SOLUTION); const psdoc = await solution.get(domainId, psid); this.response.type = 'text/markdown'; this.response.body = psdoc.content; } } class ProblemSolutionReplyRawHandler extends ProblemDetailHandler { @param('psid', Types.ObjectID) @param('psrid', Types.ObjectID) async get(domainId: string, psid: ObjectID, psrid: ObjectID) { this.checkPerm(PERM.PERM_VIEW_PROBLEM_SOLUTION); const [psdoc, psrdoc] = await solution.getReply(domainId, psid, psrid); if ((!psdoc) || psdoc.pid !== this.pdoc.docId) throw new SolutionNotFoundError(psid, psrid); this.response.type = 'text/markdown'; this.response.body = psrdoc.content; } } class ProblemCreateHandler extends Handler { async get() { this.response.template = 'problem_edit.html'; this.checkPerm(PERM.PERM_CREATE_PROBLEM); this.response.body = { path: [ ['Hydro', 'homepage'], ['problem_main', 'problem_main'], ['problem_create', null], ], page_name: 'problem_create', }; } @param('title', Types.String, isTitle) @param('pid', Types.String, isPid, true) @param('content', Types.String, isContent) @param('hidden', Types.Boolean) async post(domainId: string, title: string, pid: string, content: string, hidden = false) { pid = await problem.add(domainId, pid, title, content, this.user._id, [], [], null, hidden); this.response.body = { pid }; this.response.redirect = this.url('problem_settings', { pid }); } } export async function apply() { Route('problem_main', '/p', ProblemMainHandler); Route('problem_category', '/p/category/:category', ProblemCategoryHandler); Route('problem_random', '/problem/random', ProblemRandomHandler); Route('problem_detail', '/p/:pid', ProblemDetailHandler); Route('problem_submit', '/p/:pid/submit', ProblemSubmitHandler, PERM.PERM_SUBMIT_PROBLEM); Route('problem_pretest', '/p/:pid/pretest', ProblemPretestHandler); Route('problem_settings', '/p/:pid/settings', ProblemSettingsHandler); Route('problem_statistics', '/p/:pid/statistics', ProblemStatisticsHandler); Route('problem_edit', '/p/:pid/edit', ProblemEditHandler); Route('problem_upload', '/p/:pid/upload', ProblemDataUploadHandler); Route('problem_data', '/p/:pid/data', ProblemDataDownloadHandler); Route('problem_solution', '/p/:pid/solution', ProblemSolutionHandler); Route('problem_solution_raw', '/p/:pid/solution/:psid/raw', ProblemSolutionRawHandler); Route('problem_solution_reply_raw', '/p/:pid/solution/:psid/:psrid/raw', ProblemSolutionReplyRawHandler); Route('problem_create', '/problem/create', ProblemCreateHandler); Connection('problem_pretest_conn', '/conn/pretest', ProblemPretestConnectionHandler); } global.Hydro.handler.problem = apply;