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/packages/hydrooj/src/handler/remote.ts

161 lines
7.0 KiB
TypeScript

/* eslint-disable no-await-in-loop */
import superagent from 'superagent';
import { PassThrough } from 'stream';
import { PermissionError, InvalidTokenError, RemoteOnlineJudgeError } from '../error';
import { Logger } from '../logger';
import { ProblemAdd } from '../lib/ui';
import { PERM } from '../model/builtin';
import * as token from '../model/token';
import * as problem from '../model/problem';
import * as system from '../model/system';
import {
Handler, Types, Route, post,
} from '../service/server';
import storage from '../service/storage';
import { logAndReturn } from '../utils';
const logger = new Logger('remote');
export class ProblemSendHandler extends Handler {
async get() {
this.checkPerm(PERM.PERM_VIEW_PROBLEM);
this.response.template = 'problem_send.html';
}
@post('target', Types.String)
@post('pids', Types.Array)
async postSend(domainId: string, target: any, _pids: (number | string)[]) {
this.checkPerm(PERM.PERM_VIEW_PROBLEM);
target = target.split('@');
const getHidden = this.user.hasPerm(PERM.PERM_VIEW_PROBLEM_HIDDEN);
const getData = this.user.hasPerm(PERM.PERM_READ_PROBLEM_DATA);
const pdocs = await problem.getList(domainId, _pids, true, true, ['docId', 'owner', 'hidden']);
const pids = new Set<number>();
for (const key in pdocs) {
pids.add(pdocs[key].docId);
if (pdocs[key].hidden && !getHidden && pdocs[key].owner !== this.user._id) {
throw new PermissionError(PERM.PERM_VIEW_PROBLEM_HIDDEN);
}
if (!getData && pdocs[key].owner !== this.user._id) {
throw new PermissionError(PERM.PERM_READ_PROBLEM_DATA);
}
}
const [url, expire] = system.getMany(['server.url', 'session.saved_expire_seconds']);
const tokenId = await token.createOrUpdate(token.TYPE_EXPORT, expire, { domainId, pids: Array.from(pids) });
await superagent.post(`${target[1]}/d/${target[0]}/problem/receive`)
.send({
operation: 'request', url: `${url}d/${domainId}/problem/send`, tokenId, expire,
}).catch(logAndReturn(logger));
this.back();
}
@post('token', Types.String)
async postInfo(domainId: string, tokenId: string) {
const data = await token.get(tokenId, token.TYPE_EXPORT);
if (!data) throw new InvalidTokenError(tokenId);
const r = {};
const pdocs = await problem.getMulti(domainId, { docId: { $in: data.pids } }, ['title', 'docId']).toArray();
for (const pdoc of pdocs) r[pdoc.docId] = pdoc.title;
this.response.body = { pdocs: r, pids: data.pids };
}
@post('token', Types.String, true)
@post('pids', Types.Array)
async postFetch(domainId: string, tokenId: string, pids: number[]) {
pids = pids.map((i) => (+i ? +i : i));
const pdocs = await problem.getMulti(domainId, { docId: { $in: pids } }, problem.PROJECTION_PUBLIC).toArray();
const data = await token.get(tokenId, token.TYPE_EXPORT);
if (!data) throw new InvalidTokenError(tokenId);
if (pids.filter((pid) => !data.pids.includes(pid)).length) throw new InvalidTokenError(tokenId);
const links = [];
for (const pdoc of pdocs) {
const files = await storage.list(`problem/${domainId}/${pdoc.docId}/`, true);
const l = [];
for (const file of files) {
l.push([
file.name,
await storage.signDownloadLink(file.prefix + file.name, file.name, false, 'user'),
]);
}
links.push(l);
}
this.response.body = { pdocs, links };
}
}
export class ProblemReceiveHandler extends Handler {
async get({ domainId }) {
this.checkPerm(PERM.PERM_CREATE_PROBLEM);
const requests = await token.getMulti(token.TYPE_IMPORT, { domainId }).toArray();
this.response.template = 'problem_receive.html';
this.response.body = { requests };
}
@post('url', Types.String)
@post('tokenId', Types.String)
@post('expire', Types.UnsignedInt)
async postRequest(domainId: string, url: string, tokenId: string, expire: number) {
const res = await superagent.post(url)
.send({ operation: 'info', token: tokenId })
.catch(logAndReturn(logger));
if (res instanceof Error) throw new RemoteOnlineJudgeError(res.message);
const [id] = await token.add(token.TYPE_IMPORT, expire, {
domainId, tokenId, source: url, pdocs: res.body.pdocs, pids: res.body.pids,
});
this.response.body = { token: id };
}
// eslint-disable-next-line class-methods-use-this
async syncFiles(domainId: string, baseUrl: string, pid: number, files: [string, string][]) {
for (const file of files) {
const content = new PassThrough();
superagent.get(file[1].startsWith('/') ? baseUrl + file[1] : file[1]).pipe(content);
if (file[0].startsWith('testdata/')) {
await problem.addTestdata(domainId, pid, file[0].split('testdata/')[1], content);
} else if (file[0].startsWith('additional_file/')) {
await problem.addAdditionalFile(domainId, pid, file[0].split('additional_file/')[1], content);
}
}
}
@post('tokenId', Types.String)
@post('pids', Types.Array, true)
async postConfirm(domainId: string, tokenId: string, filterPid?: number[]) {
this.checkPerm(PERM.PERM_CREATE_PROBLEM);
const data = await token.get(tokenId, token.TYPE_IMPORT);
if (!data) throw new InvalidTokenError(tokenId);
const res = await superagent.post(data.source).send({
operation: 'fetch',
token: data.tokenId,
pids: filterPid.length ? filterPid.map((i) => (+i ? +i : i)) : data.pids,
}).catch(logAndReturn(logger));
if (res instanceof Error) throw new RemoteOnlineJudgeError(res.message);
const pids = [];
const { pdocs } = res.body;
const files = res.body.links;
const source = new URL(data.source);
const baseUrl = `${source.protocol}//${source.host}`;
for (let i = 0; i < pdocs.length; i++) {
const pdoc = pdocs[i];
if (pdoc.pid) {
const exist = await problem.get(domainId, pdoc.pid);
if (exist) pdoc.pid = undefined;
}
const pid = await problem.add(domainId, pdoc.pid, pdoc.title, pdoc.content, this.user._id, pdoc.tag, pdoc.category);
pids.push(pid);
this.syncFiles(domainId, baseUrl, pid, files[i]).catch(logAndReturn(logger));
await problem.edit(domainId, pid, { html: pdoc.html });
}
this.back();
this.response.body = { pids };
}
}
export async function apply() {
ProblemAdd('problem_receive', {}, 'copy', 'Import From Hydro');
Route('problem_receive', '/problem/receive', ProblemReceiveHandler);
Route('problem_send', '/problem/send', ProblemSendHandler);
}
global.Hydro.handler.remote = apply;