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/module/migrate-vijos/script.js

339 lines
11 KiB
JavaScript

/* eslint-disable no-await-in-loop */
const { mongodb } = global.Hydro.nodeModules;
const dst = global.Hydro.service.db;
const { problem } = global.Hydro.model;
const map = {};
const pid = (id) => {
if (map[id.toString()]) return map[id.toString()];
if (id.generationTime) return id;
id = id.toString();
return id;
};
const tasks = {
user: async (docs) => docs.map((doc) => ({
_id: doc._id,
uname: doc.uname,
unameLower: doc.uname_lower,
salt: doc.salt,
hash: doc.hash.split('|')[1],
hashType: doc.hash.split('|')[0] === 'vj4' ? 'hydro' : doc.hash.split('|')[0],
role: doc.priv === -1 ? 'root' : 'default',
gravatar: doc.gravatar,
email: doc.mail,
emailLower: doc.mail_lower,
regat: doc.regat,
regip: doc.regip,
loginat: doc.loginat,
loginip: doc.loginip,
codeLang: doc.code_lang,
codeTemplate: doc.code_template,
timezone: doc.timezone,
viewLang: doc.viewLang,
bio: doc.bio,
gender: doc.gender,
qq: doc.qq,
})),
problem: async (docs) => {
const problems = [];
for (const doc of docs) {
if (typeof doc.doc_id === 'number') {
map[doc.doc_id] = doc._id;
if (doc.pname) map[doc.pname] = doc._id;
} else {
map[doc.doc_id] = doc.doc_id;
if (doc.pname) map[doc.pname] = doc.doc_id;
}
problems.push({
_id: doc._id,
pid: doc.pname || pid(doc.doc_id),
title: doc.title,
content: doc.content,
owner: doc.owner_uid,
data: doc.data,
category: doc.category,
tag: doc.tag,
nSubmit: doc.num_submit,
nAccept: doc.num_accept,
hidden: doc.hidden || false,
difficulty: doc.difficulty_admin,
});
}
return problems;
},
'problem.status': async (docs) => {
const problems = [];
for (const doc of docs) {
problems.push({
pid: pid(doc.doc_id),
rid: doc.rid,
uid: doc.uid,
status: doc.status,
});
}
return problems;
},
solution: async (docs) => {
const solutions = [];
for (const doc of docs) {
const reply = doc.reply.map((r) => ({
owner: r.owner_uid,
content: r.content,
_id: r._id,
}));
solutions.push({
_id: doc._id,
owner: doc.owner_uid,
content: doc.content,
vote: doc.vote,
reply,
pid: pid(doc.parent_doc_id),
});
}
return solutions;
},
discussion: async (docs) => {
const TYPE = {
10: 'problem',
20: 'node',
30: 'contest',
};
const discussions = [];
for (const doc of docs) {
discussions.push({
_id: doc._id,
title: doc.title,
content: doc.content,
owner: doc.owner_uid,
views: doc.views,
highlight: doc.highlight,
nReply: doc.num_replies,
ip: doc.ip,
updateAt: doc.update_at,
parentType: TYPE[doc.parentType],
parentId: doc.parentId,
});
}
return discussions;
},
'discussion.reply': async (docs) => docs.map((doc) => ({
_id: doc._id,
owner: doc.owner_uid,
ip: doc.ip,
content: doc.content,
parentId: doc.parent_doc_id,
})),
contest: async (docs) => {
const RULES = {
2: 'oi',
3: 'acm',
11: 'homework',
};
const contests = [];
for (const doc of docs) {
contests.push({
_id: doc._id,
title: doc.title,
content: doc.content,
owner: doc.owner_uid,
rule: RULES[doc.rule],
beginAt: doc.begin_at,
endAt: doc.end_at,
pids: doc.pids.map((id) => pid(id)),
attend: doc.attend,
});
}
return contests;
},
'contest.status': async (docs) => {
const contests = [];
for (const doc of docs) {
const journal = [];
const detail = [];
for (const i of doc.journal || []) {
const pdoc = await problem.get(pid(i.pid));
journal.push({ ...i, id: pdoc._id });
}
for (const i of doc.detail || []) {
const pdoc = await problem.get(pid(i.pid));
detail.push({ ...i, id: pdoc._id });
}
contests.push({
_id: doc._id,
tid: doc.doc_id,
uid: doc.uid,
attend: doc.attend,
rev: doc.rev,
journal,
detail,
score: doc.score,
accept: doc.accept || 0,
time: doc.time || 0,
});
}
return contests;
},
training: async (docs) => {
const trainings = [];
for (const doc of docs) {
const dag = [];
for (const t of doc.dag) {
dag.push({
_id: t._id,
title: t.title,
requireNids: t.require_nids,
pids: t.pids.map((id) => pid(id)),
});
}
trainings.push({
_id: doc._id,
title: doc.title,
content: doc.content,
description: doc.desc,
enroll: doc.enroll,
owner: doc.owner_uid,
dag,
});
}
return trainings;
},
'training.status': async (docs) => {
const trainings = [];
for (const doc of docs) {
trainings.push({
_id: doc._id,
pid: pid(doc.doc_id),
rid: doc.rid,
uid: doc.uid,
done: doc.done,
enroll: doc.enroll,
tid: doc.doc_id,
doneNids: doc.done_nids,
donePids: doc.done_pids.map((id) => pid(id)),
});
}
return trainings;
},
record: async (docs) => {
const records = [];
for (const doc of docs) {
const testCases = [];
for (const c of doc.cases) {
testCases.push({
status: c.status,
score: c.score,
time: c.time,
memory: c.memory,
judge_text: c.judge_text,
});
}
records.push({
_id: doc._id,
status: doc.status,
score: doc.score,
time: doc.time_ms,
memory: doc.memory_kn,
code: doc.code,
lang: doc.lang,
uid: doc.uid,
pid: pid(doc.pid),
tid: doc.tid,
judger: doc.judge_uid,
judgeAt: doc.judge_at,
judgeTexts: doc.judge_texts,
compileTexts: doc.compile_texts,
rejudged: doc.rejudged,
testCases,
});
}
return records;
},
'fs.files': async (docs) => docs,
'fs.chunks': async (docs) => docs,
file: async (docs) => docs.map((doc) => ({
_id: doc._id,
count: doc.metadata.link,
secret: doc.metadata.secret,
size: doc.length,
md5: doc.md5,
})),
};
const cursor = {
user: (s) => s.collection('user').find(),
problem: (s) => s.collection('document').find({ doc_type: 10 }),
'problem.status': (s) => s.collection('document.status').find({ doc_type: 10 }),
solution: (s) => s.collection('document').find({ doc_type: 11 }),
discussion: (s) => s.collection('document').find({ doc_type: 21 }),
'discussion.reply': (s) => s.collection('document').find({ doc_type: 22 }),
contest: (s) => s.collection('document').find({ doc_type: 30 }),
'contest.status': (s) => s.collection('document.status').find({ doc_type: 30 }),
training: (s) => s.collection('document').find({ doc_type: 40 }),
'training.status': (s) => s.collection('document.status').find({ doc_type: 40 }),
record: (s) => s.collection('record').find(),
'fs.files': (s) => s.collection('fs.files').find(),
'fs.chunks': (s) => s.collection('fs.chunks').find(),
file: (s) => s.collection('fs.files').find(),
};
async function task(name, src, report) {
const count = await cursor[name](src).count();
await report({ progress: 1, message: `${name}: ${count}` });
const total = Math.floor(count / 50);
for (let i = 0; i <= total; i++) {
const docs = await cursor[name](src).skip(i * 50).limit(50).toArray();
const t = await tasks[name](docs);
if (t.length) await dst.collection(name).insertMany(t);
await report({ progress: Math.round(100 * ((i + 1) / (total + 1))) });
}
}
async function migrateVijos({
host, port, name, username, password,
}, report = () => { }) {
let mongourl = 'mongodb://';
if (username) mongourl += `${username}:${password}@`;
mongourl += `${host}:${port}/${name}`;
const Database = await mongodb.MongoClient.connect(mongourl, {
useNewUrlParser: true, useUnifiedTopology: true,
});
const src = Database.db(name);
5 years ago
await report({ progress: 0, message: 'Database connected.' });
await dst.collection('system').updateOne(
{ _id: 'user' },
{
$set: {
value: (await src.collection('system').findOne({ _id: 'user_counter' })).value,
},
},
{ upsert: true },
);
5 years ago
await report({ progress: 1, message: 'Collection:system done.' });
await dst.collection('user').deleteMany({ _id: { $nin: [0, 1] } });
if (!await dst.collection('system').findOne({ _id: 'migrateVijosFs' })) {
const f = ['fs.files', 'fs.chunks'];
for (const i of f) {
await dst.collection(i).deleteMany();
await task(i, src, report);
}
await dst.collection('system').insertOne({ _id: 'migrateVijosFs', value: true });
}
const d = [
'problem', 'problem.status', 'solution', 'discussion', 'discussion.reply',
'contest', 'training', 'training.status', 'record', 'file',
];
for (const i of d) {
await dst.collection(i).deleteMany();
}
const t = [
'user', 'problem', 'problem.status', 'solution', 'discussion',
'discussion.reply', 'contest', 'training', 'training.status', 'record',
'file',
];
for (const i of t) await task(i, src, report);
}
global.Hydro.script.migrateVijos = module.exports = { run: migrateVijos };