/* * This file is part of SYZOJ. * * Copyright (c) 2016 Menci * * SYZOJ is free software: you can redistribute it and/or modify * it under the terms of the GNU Affero General Public License as * published by the Free Software Foundation, either version 3 of the * License, or (at your option) any later version. * * SYZOJ is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Affero General Public License for more details. * * You should have received a copy of the GNU Affero General Public * License along with SYZOJ. If not, see . */ 'use strict'; let User = syzoj.model('user'); let Problem = syzoj.model('problem'); let WaitingJudge = syzoj.model('waiting_judge'); let JudgeState = syzoj.model('judge_state'); let File = syzoj.model('file'); function setLoginCookie(username, password, res) { res.cookie('login', JSON.stringify([username, password])); } // Login app.post('/api/login', async (req, res) => { try { res.setHeader('Content-Type', 'application/json'); let user = await User.fromName(req.body.username); if (!user) res.send({ error_code: 1001 }); else if (user.password !== req.body.password) res.send({ error_code: 1002 }); else { req.session.user_id = user.id; setLoginCookie(user.username, user.password, res); res.send({ error_code: 1 }); } } catch (e) { syzoj.log(e); res.send({ error_code: e }); } }); // Sign up app.post('/api/sign_up', async (req, res) => { try { res.setHeader('Content-Type', 'application/json'); let user = await User.fromName(req.body.username); if (user) throw 2008; user = await User.findOne({ where: { email: req.body.email } }); if (user) throw 2009; // Because the salt is "syzoj2_xxx" and the "syzoj2_xxx" 's md5 is"59cb..." // the empty password 's md5 will equal "59cb.." let syzoj2_xxx_md5 = '59cb65ba6f9ad18de0dcd12d5ae11bd2'; if (req.body.password === syzoj2_xxx_md5) throw 2007; if (!(req.body.email = req.body.email.trim())) throw 2006; if (!syzoj.utils.isValidUsername(req.body.username)) throw 2002; if (syzoj.config.register_mail.enabled) { let sendmail = Promise.promisify(require('sendmail')()); let sendObj = { username: req.body.username, password: req.body.password, email: req.body.email, prevUrl: req.body.prevUrl, r: Math.random() }; let encrypted = encodeURIComponent(syzoj.utils.encrypt(JSON.stringify(sendObj), syzoj.config.register_mail.key).toString('base64')); let url = req.protocol + '://' + req.get('host') + syzoj.utils.makeUrl(['api', 'sign_up', encrypted]); try { await sendmail({ from: `${syzoj.config.title} <${syzoj.config.register_mail.address}>`, to: req.body.email, type: 'text/html', subject: `${req.body.username} 的 ${syzoj.config.title} 注册验证邮件`, html: `

请点击该链接完成您在 ${syzoj.config.title} 的注册:

${url}

如果您不是 ${req.body.username},请忽略此邮件。

` }); } catch (e) { throw 2010 } res.send(JSON.stringify({ error_code: 2 })); } else { user = await User.create({ username: req.body.username, password: req.body.password, email: req.body.email }); await user.save(); req.session.user_id = user.id; setLoginCookie(user.username, user.password, res); res.send(JSON.stringify({ error_code: 1 })); } } catch (e) { syzoj.log(e); res.send(JSON.stringify({ error_code: e })); } }); app.get('/api/sign_up/:token', async (req, res) => { try { let obj; try { let decrypted = syzoj.utils.decrypt(Buffer.from(req.params.token, 'base64'), syzoj.config.register_mail.key).toString(); obj = JSON.parse(decrypted); } catch (e) { throw new ErrorMessage('无效的注册验证链接。'); } let user = await User.fromName(obj.username); if (user) throw new ErrorMessage('用户名已被占用。'); user = await User.findOne({ where: { email: obj.email } }); if (user) throw new ErrorMessage('邮件地址已被占用。'); // Because the salt is "syzoj2_xxx" and the "syzoj2_xxx" 's md5 is"59cb..." // the empty password 's md5 will equal "59cb.." let syzoj2_xxx_md5 = '59cb65ba6f9ad18de0dcd12d5ae11bd2'; if (obj.password === syzoj2_xxx_md5) throw new ErrorMessage('密码不能为空。'); if (!(obj.email = obj.email.trim())) throw new ErrorMessage('邮件地址不能为空。'); if (!syzoj.utils.isValidUsername(obj.username)) throw new ErrorMessage('用户名不合法。'); user = await User.create({ username: obj.username, password: obj.password, email: obj.email }); await user.save(); req.session.user_id = user.id; setLoginCookie(user.username, user.password, res); res.redirect(obj.prevUrl || '/'); } catch (e) { syzoj.log(e); res.render('error', { err: e }); } }); // Markdown app.post('/api/markdown', async (req, res) => { try { let s = await syzoj.utils.markdown(req.body.s.toString()); res.send(s); } catch (e) { syzoj.log(e); res.send(e); } }); // APIs for judge client app.get('/api/waiting_judge', async (req, res) => { try { if (req.query.session_id !== syzoj.config.judge_token) return res.status(404).send({ err: 'Permission denied' }); let judge_state; await syzoj.utils.lock('/api/waiting_judge', async () => { let waiting_judge = await WaitingJudge.findOne(); if (!waiting_judge) { return; } judge_state = await waiting_judge.getJudgeState(); await judge_state.loadRelationships(); await judge_state.problem.loadRelationships(); await waiting_judge.destroy(); }); if (judge_state) { await judge_state.loadRelationships(); await judge_state.problem.loadRelationships(); if (judge_state.problem.type === 'submit-answer') { res.send({ have_task: 1, judge_id: judge_state.id, answer_file: judge_state.code, testdata: judge_state.problem.id, problem_type: judge_state.problem.type }); } else { res.send({ have_task: 1, judge_id: judge_state.id, code: judge_state.code, language: judge_state.language, testdata: judge_state.problem.id, time_limit: judge_state.problem.time_limit, memory_limit: judge_state.problem.memory_limit, file_io: judge_state.problem.file_io, file_io_input_name: judge_state.problem.file_io_input_name, file_io_output_name: judge_state.problem.file_io_output_name, problem_type: judge_state.problem.type }); } } else { res.send({ have_task: 0 }); } } catch (e) { res.status(500).send(e); } }); app.post('/api/update_judge/:id', async (req, res) => { try { if (req.query.session_id !== syzoj.config.judge_token) return res.status(404).send({ err: 'Permission denied' }); let judge_state = await JudgeState.fromID(req.params.id); await judge_state.updateResult(JSON.parse(req.body.result)); await judge_state.save(); await judge_state.updateRelatedInfo(); res.send({ return: 0 }); } catch (e) { syzoj.log(e); res.status(500).send(e); } }); app.get('/static/uploads/answer/:md5', async (req, res) => { try { res.sendFile(File.resolvePath('answer', req.params.md5)); } catch (e) { res.status(500).send(e); } });