oicontest
Version:
OI Contest Management Tool
83 lines (80 loc) • 4.26 kB
text/typescript
import { Command } from 'commander';
import chalk from 'chalk';
import fs from 'fs-extra';
import path from 'path';
import archiver from 'archiver';
export const packageCommand = new Command('package')
.description('Generate contest directory structure readme.txt and zip the contest directory')
.action(async () => {
try {
const contestDir = process.cwd();
const parentDir = path.dirname(contestDir);
const contestName = path.basename(contestDir);
const readmePath = path.join(contestDir, 'readme.txt');
const zipPath = path.join(parentDir, `${contestName}.zip`);
const now = new Date();
// 1. 生成目录结构说明
let readmeContent = `OIContest Package\n=================\n\n`;
readmeContent += `本包为信息学竞赛/题库项目“${contestName}”的完整目录结构归档。\n`;
readmeContent += `导出时间: ${now.toISOString()}\n`;
readmeContent += `\n用途:\n- 便于命题人、教研组、竞赛组委会等同行之间分发、交流、归档整套 contest 资料\n- 便于快速了解题库/比赛的目录结构和内容组成\n\n使用说明:\n- 解压本 zip 包后,即可获得完整的 contest 目录结构\n- 目录下的 readme.txt 为本说明文件\n- 题目描述、数据、题解、附加文件等均在各自子目录下\n\n目录结构如下:\n\n`;
readmeContent += await generateTree(contestDir, '', contestDir);
// 追加目录说明
readmeContent += `\n目录说明:\n`;
readmeContent += `- oicontest.json :比赛全局配置文件\n`;
readmeContent += `- problem/ :所有题目目录\n`;
readmeContent += `- problem/<id>/ :单个题目目录\n`;
readmeContent += `- problem/<id>/config.json :单题配置信息\n`;
readmeContent += `- problem/<id>/status.json :单题状态信息\n`;
readmeContent += `- problem/<id>/problem.md :题目描述\n`;
readmeContent += `- problem/<id>/testdata/ :测试数据目录\n`;
readmeContent += `- problem/<id>/src/ :题目源代码、标准程序等\n`;
readmeContent += `- problem/<id>/solution/ :题解目录\n`;
readmeContent += `- problem/<id>/additional_file/ :附加文件(图片、数据等)\n`;
readmeContent += `- html/ :生成的 HTML 题面\n`;
readmeContent += `- output/ :LEMON 评测包\n`;
readmeContent += `- pdf/ :PDF 题面(如有,已经弃用)\n`;
readmeContent += `- readme.txt :本说明文件\n`;
await fs.writeFile(readmePath, readmeContent, 'utf-8');
console.log(chalk.green(`readme.txt generated at ${readmePath}`));
// 2. 打包整个 contest 目录
await zipDirectory(contestDir, zipPath);
console.log(chalk.green(`Contest directory zipped to ${zipPath}`));
} catch (err: any) {
console.error(chalk.red(`Package failed: ${err.message}`));
process.exit(1);
}
});
async function generateTree(dir: string, prefix: string, root: string): Promise<string> {
let result = '';
const items = await fs.readdir(dir);
const entries = await Promise.all(items.map(async (item) => {
const fullPath = path.join(dir, item);
const stat = await fs.stat(fullPath);
return { item, fullPath, stat };
}));
for (let i = 0; i < entries.length; i++) {
const { item, fullPath, stat } = entries[i];
const isLast = i === entries.length - 1;
const branch = isLast ? '└── ' : '├── ';
result += `${prefix}${branch}${item}`;
if (stat.isDirectory()) {
result += '/\n';
result += await generateTree(fullPath, prefix + (isLast ? ' ' : '│ '), root);
} else {
result += `\n`;
}
}
return result;
}
async function zipDirectory(source: string, out: string): Promise<void> {
return new Promise((resolve, reject) => {
const archive = archiver('zip', { zlib: { level: 9 } });
const stream = fs.createWriteStream(out);
archive.directory(source, false);
archive.on('error', err => reject(err));
stream.on('close', () => resolve());
archive.pipe(stream);
archive.finalize();
});
}