oicontest
Version:
OI Contest Management Tool
102 lines (101 loc) • 5.89 kB
JavaScript
;
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
return new (P || (P = Promise))(function (resolve, reject) {
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
step((generator = generator.apply(thisArg, _arguments || [])).next());
});
};
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.packageCommand = void 0;
const commander_1 = require("commander");
const chalk_1 = __importDefault(require("chalk"));
const fs_extra_1 = __importDefault(require("fs-extra"));
const path_1 = __importDefault(require("path"));
const archiver_1 = __importDefault(require("archiver"));
exports.packageCommand = new commander_1.Command('package')
.description('Generate contest directory structure readme.txt and zip the contest directory')
.action(() => __awaiter(void 0, void 0, void 0, function* () {
try {
const contestDir = process.cwd();
const parentDir = path_1.default.dirname(contestDir);
const contestName = path_1.default.basename(contestDir);
const readmePath = path_1.default.join(contestDir, 'readme.txt');
const zipPath = path_1.default.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 += yield 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`;
yield fs_extra_1.default.writeFile(readmePath, readmeContent, 'utf-8');
console.log(chalk_1.default.green(`readme.txt generated at ${readmePath}`));
// 2. 打包整个 contest 目录
yield zipDirectory(contestDir, zipPath);
console.log(chalk_1.default.green(`Contest directory zipped to ${zipPath}`));
}
catch (err) {
console.error(chalk_1.default.red(`Package failed: ${err.message}`));
process.exit(1);
}
}));
function generateTree(dir, prefix, root) {
return __awaiter(this, void 0, void 0, function* () {
let result = '';
const items = yield fs_extra_1.default.readdir(dir);
const entries = yield Promise.all(items.map((item) => __awaiter(this, void 0, void 0, function* () {
const fullPath = path_1.default.join(dir, item);
const stat = yield fs_extra_1.default.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 += yield generateTree(fullPath, prefix + (isLast ? ' ' : '│ '), root);
}
else {
result += `\n`;
}
}
return result;
});
}
function zipDirectory(source, out) {
return __awaiter(this, void 0, void 0, function* () {
return new Promise((resolve, reject) => {
const archive = (0, archiver_1.default)('zip', { zlib: { level: 9 } });
const stream = fs_extra_1.default.createWriteStream(out);
archive.directory(source, false);
archive.on('error', err => reject(err));
stream.on('close', () => resolve());
archive.pipe(stream);
archive.finalize();
});
});
}