feat(problem/manage/data): new data configure page

This commit is contained in:
Baoshuo Ren 2023-02-04 21:52:59 +08:00
parent 5d8f194293
commit 38635106c0
Signed by: baoshuo
GPG Key ID: 00CB9680AB29F51A
7 changed files with 269 additions and 166 deletions

View File

@ -0,0 +1,23 @@
<?php
requireLib('bootstrap5');
requirePHPLib('form');
requirePHPLib('judger');
requirePHPLib('data');
UOJProblem::init(UOJRequest::get('id')) || UOJResponse::page404();
UOJProblem::cur()->userCanManage(Auth::user()) || UOJResponse::page403();
$problem = UOJProblem::info();
$problem_configure = new UOJProblemConfigure(UOJProblem::cur());
$problem_configure->runAtServer();
?>
<?php echoUOJPageHeader('数据配置 - ' . HTML::stripTags(UOJProblem::cur()->getTitle(['with' => 'id']))) ?>
<h1>
<?= UOJProblem::cur()->getTitle(['with' => 'id']) ?> 数据配置
</h1>
<?php $problem_configure->printHTML() ?>
<?php echoUOJPageFooter() ?>

View File

@ -90,58 +90,6 @@ if ($_POST['problem_data_file_submit'] == 'submit') {
}
}
// 添加配置文件
if ($_POST['problem_settings_file_submit'] == 'submit') {
if ($_POST['use_builtin_checker'] and $_POST['n_tests']) {
$set_filename = "/var/uoj_data/upload/{$problem['id']}/problem.conf";
$has_legacy = false;
if (file_exists($set_filename)) {
$has_legacy = true;
unlink($set_filename);
}
$setfile = fopen($set_filename, "w");
fwrite($setfile, "use_builtin_judger on\n");
if ($_POST['use_builtin_checker'] != 'ownchk') {
fwrite($setfile, "use_builtin_checker " . $_POST['use_builtin_checker'] . "\n");
}
fwrite($setfile, "n_tests " . $_POST['n_tests'] . "\n");
if ($_POST['n_ex_tests']) {
fwrite($setfile, "n_ex_tests " . $_POST['n_ex_tests'] . "\n");
} else {
fwrite($setfile, "n_ex_tests 0\n");
}
if ($_POST['n_sample_tests']) {
fwrite($setfile, "n_sample_tests " . $_POST['n_sample_tests'] . "\n");
} else {
fwrite($setfile, "n_sample_tests 0\n");
}
if (isset($_POST['input_pre'])) {
fwrite($setfile, "input_pre " . $_POST['input_pre'] . "\n");
}
if (isset($_POST['input_suf'])) {
fwrite($setfile, "input_suf " . $_POST['input_suf'] . "\n");
}
if (isset($_POST['output_pre'])) {
fwrite($setfile, "output_pre " . $_POST['output_pre'] . "\n");
}
if (isset($_POST['output_suf'])) {
fwrite($setfile, "output_suf " . $_POST['output_suf'] . "\n");
}
fwrite($setfile, "time_limit " . ($_POST['time_limit'] ?: 1) . "\n");
fwrite($setfile, "memory_limit " . ($_POST['memory_limit'] ?: 256) . "\n");
fclose($setfile);
if (!$has_legacy) {
echo "<script>alert('添加成功!请点击「检验配置并同步数据」按钮以应用新配置文件。')</script>";
} else {
echo "<script>alert('替换成功!请点击「检验配置并同步数据」按钮以应用新配置文件。')</script>";
}
} else {
$errmsg = "添加配置文件失败,请检查是否所有必填输入框都已填写!";
becomeMsgPage('<div>' . $errmsg . '</div><a href="/problem/' . $problem['id'] . '/manage/data">返回</a>');
}
}
$info_form = new UOJForm('info');
$attachment_url = UOJProblem::cur()->getAttachmentUri();
$info_form->appendHTML(<<<EOD
@ -670,7 +618,7 @@ if ($problem['hackable']) {
<button type="button" class="btn d-block w-100 btn-primary" data-bs-toggle="modal" data-bs-target="#UploadDataModal">上传数据</button>
</div>
<div class="mt-2">
<button type="button" class="btn d-block w-100 btn-primary" data-bs-toggle="modal" data-bs-target="#ProblemSettingsFileModal">试题配置</button>
<a role="button" class="btn d-block w-100 btn-primary" href="<?= UOJProblem::cur()->getUri('/manage/data/configure') ?>">试题配置</a>
</div>
</div>
</aside>
@ -705,108 +653,4 @@ if ($problem['hackable']) {
</div>
</div>
<?php $problem_conf = UOJProblem::cur()->getProblemConf() ?>
<div class="modal fade" id="ProblemSettingsFileModal" tabindex="-1" role="dialog" aria-labelledby="myModalLabel" aria-hidden="true">
<div class="modal-dialog modal-lg">
<div class="modal-content">
<div class="modal-header">
<h4 class="modal-title" id="myModalLabel">试题配置</h4>
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<form class="form-horizontal" action="" method="post" role="form">
<div class="modal-body">
<div class="form-group row">
<label for="use_builtin_checker" class="col-sm-5 control-label">比对函数</label>
<div class="col-sm-7">
<?php $checker_value = $problem_conf instanceof UOJProblemConf ? $problem_conf->getVal('use_builtin_checker', 'ownchk') : ""; ?>
<select class="form-select" id="use_builtin_checker" name="use_builtin_checker">
<option value="ncmp" <?= $checker_value == "ncmp" ? 'selected' : '' ?>>ncmp: 整数序列</option>
<option value="wcmp" <?= $checker_value == "wcmp" ? 'selected' : '' ?>>wcmp: 字符串序列</option>
<option value="lcmp" <?= $checker_value == "lcmp" ? 'selected' : '' ?>>lcmp: 多行数据(忽略行内与行末的多余空格,同时忽略文末回车)</option>
<option value="fcmp" <?= $checker_value == "fcmp" ? 'selected' : '' ?>>fcmp: 多行数据(不忽略行末空格,但忽略文末回车)</option>
<option value="rcmp4" <?= $checker_value == "rcmp4" ? 'selected' : '' ?>>rcmp4: 浮点数序列(误差不超过 1e-4</option>
<option value="rcmp6" <?= $checker_value == "rcmp6" ? 'selected' : '' ?>>rcmp6: 浮点数序列(误差不超过 1e-6</option>
<option value="rcmp9" <?= $checker_value == "rcmp9" ? 'selected' : '' ?>>rcmp9: 浮点数序列(误差不超过 1e-9</option>
<option value="yesno" <?= $checker_value == "yesno" ? 'selected' : '' ?>>yesno: Yes、No不区分大小写</option>
<option value="uncmp" <?= $checker_value == "uncmp" ? 'selected' : '' ?>>uncmp: 整数集合</option>
<option value="bcmp" <?= $checker_value == "bcmp" ? 'selected' : '' ?>>bcmp: 二进制文件</option>
<option value="ownchk" <?= $checker_value == "ownchk" ? 'selected' : '' ?>>自定义校验器(需上传 chk.cpp</option>
</select>
</div>
</div>
<div class="form-group row">
<label for="n_tests" class="col-sm-5 control-label">n_tests</label>
<div class="col-sm-7">
<?php $n_tests_value = $problem_conf instanceof UOJProblemConf ? $problem_conf->getVal('n_tests', '') : ""; ?>
<input type="number" class="form-control" id="n_tests" name="n_tests" placeholder="数据点个数(必填)" value="<?= $n_tests_value ?>">
</div>
</div>
<div class="form-group row">
<label for="n_ex_tests" class="col-sm-5 control-label">n_ex_tests</label>
<div class="col-sm-7">
<?php $n_ex_tests_value = $problem_conf instanceof UOJProblemConf ? $problem_conf->getVal('n_ex_tests', 0) : ""; ?>
<input type="number" class="form-control" id="n_ex_tests" name="n_ex_tests" placeholder="额外数据点个数(默认为 0" value="<?= $n_ex_tests_value ?>">
</div>
</div>
<div class="form-group row">
<label for="n_sample_tests" class="col-sm-5 control-label">n_sample_tests</label>
<div class="col-sm-7">
<?php $n_sample_tests_value = $problem_conf instanceof UOJProblemConf ? $problem_conf->getVal('n_sample_tests', 0) : ""; ?>
<input type="number" class="form-control" id="n_sample_tests" name="n_sample_tests" placeholder="样例测试点个数(默认为 0" value="<?= $n_sample_tests_value ?>">
</div>
</div>
<div class="form-group row">
<label for="input_pre" class="col-sm-5 control-label">input_pre</label>
<div class="col-sm-7">
<?php $input_pre_value = $problem_conf instanceof UOJProblemConf ? $problem_conf->getVal('input_pre', 'input') : ""; ?>
<input type="text" class="form-control" id="input_pre" name="input_pre" placeholder="输入文件名称(默认为 input" value="<?= $input_pre_value ?>">
</div>
</div>
<div class="form-group row">
<label for="input_suf" class="col-sm-5 control-label">input_suf</label>
<div class="col-sm-7">
<?php $input_suf_value = $problem_conf instanceof UOJProblemConf ? $problem_conf->getVal('input_suf', 'txt') : ""; ?>
<input type="text" class="form-control" id="input_suf" name="input_suf" placeholder="输入文件后缀(默认为 txt" value="<?= $input_suf_value ?>">
</div>
</div>
<div class="form-group row">
<label for="output_pre" class="col-sm-5 control-label">output_pre</label>
<div class="col-sm-7">
<?php $output_pre_value = $problem_conf instanceof UOJProblemConf ? $problem_conf->getVal('output_pre', 'output') : ''; ?>
<input type="text" class="form-control" id="output_pre" name="output_pre" placeholder="输出文件名称(默认为 output" value="<?= $output_pre_value ?>">
</div>
</div>
<div class="form-group row">
<label for="output_suf" class="col-sm-5 control-label">output_suf</label>
<div class="col-sm-7">
<?php $output_suf_value = $problem_conf instanceof UOJProblemConf ? $problem_conf->getVal('output_suf', 'txt') : ""; ?>
<input type="text" class="form-control" id="output_suf" name="output_suf" placeholder="输出文件后缀(默认为 txt" value="<?= $output_suf_value ?>">
</div>
</div>
<div class="form-group row">
<label for="time_limit" class="col-sm-5 control-label">time_limit</label>
<div class="col-sm-7">
<?php $time_limit_value = $problem_conf instanceof UOJProblemConf ? $problem_conf->getVal('time_limit', 1) : ""; ?>
<input type="text" class="form-control" id="time_limit" name="time_limit" placeholder="时间限制(默认为 1s" value="<?= $time_limit_value ?>">
</div>
</div>
<div class="form-group row">
<label for="memory_limit" class="col-sm-5 control-label">memory_limit</label>
<div class="col-sm-7">
<?php $memory_limit_value = $problem_conf instanceof UOJProblemConf ? $problem_conf->getVal('memory_limit', 256) : ""; ?>
<input type="number" class="form-control" id="memory_limit" name="memory_limit" placeholder="内存限制(默认为 256 MB" value="<?= $memory_limit_value ?>">
</div>
</div>
<input type="hidden" name="problem_settings_file_submit" value="submit">
</div>
<div class="modal-footer">
<button type="submit" class="btn btn-success">确定</button>
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">关闭</button>
</div>
</form>
</div>
</div>
</div>
<?php echoUOJPageFooter() ?>

View File

@ -201,7 +201,9 @@ class SyncProblemDataHandler {
public function _updateProblemConf($new_problem_conf) {
try {
putUOJConf("{$this->data_dir}/problem.conf", $new_problem_conf);
putUOJConf("{$this->upload_dir}/problem.conf", $new_problem_conf);
$this->_sync();
return '';
} catch (Exception $e) {
return $e->getMessage();
@ -506,3 +508,7 @@ function dataAddHackPoint($problem, $uploaded_input_file, $uploaded_output_file,
return (new SyncProblemDataHandler($problem, $user))->addHackPoint($uploaded_input_file, $uploaded_output_file, $reason);
}
function dataUpdateProblemConf($problem, $new_problem_conf) {
return (new SyncProblemDataHandler($problem))->updateProblemConf($new_problem_conf);
}

View File

@ -125,6 +125,8 @@ class UOJForm {
'type' => 'text',
'div_class' => '',
'input_class' => 'form-control',
'input_attrs' => [],
'input_div_class' => null,
'default_value' => '',
'label' => '',
'label_class' => 'form-label',
@ -148,6 +150,10 @@ class UOJForm {
], $config['label']);
}
if ($config['input_div_class'] !== null) {
$html .= HTML::tag_begin('div', ['class' => $config['input_div_class']]);
}
$html .= HTML::empty_tag('input', [
'class' => $config['input_class'],
'type' => $config['type'],
@ -155,13 +161,17 @@ class UOJForm {
'id' => "input-$name",
'value' => $config['default_value'],
'placeholder' => $config['placeholder'],
]);
] + $config['input_attrs']);
$html .= HTML::tag('div', ['class' => 'invalid-feedback', 'id' => "help-$name"], '');
if ($config['help']) {
$html .= HTML::tag('div', ['class' => $config['help_class']], $config['help']);
}
if ($config['input_div_class'] !== null) {
$html .= HTML::tag_end('div');
}
$html .= HTML::tag_end('div');
$this->add($name, $html, $config['validator_php'], $config['validator_js']);
@ -255,6 +265,7 @@ class UOJForm {
$config += [
'div_class' => '',
'select_class' => 'form-select',
'select_div_class' => null,
'options' => [],
'default_value' => '',
'label' => '',
@ -275,6 +286,10 @@ class UOJForm {
], $config['label']);
}
if ($config['select_div_class'] !== null) {
$html .= HTML::tag_begin('div', ['class' => $config['select_div_class']]);
}
// Select
$html .= HTML::tag_begin('select', ['id' => "input-$name", 'name' => $name, 'class' => $config['select_class']]);
@ -293,6 +308,10 @@ class UOJForm {
$html .= HTML::tag('div', ['class' => $config['help_class']], $config['help']);
}
if ($config['select_div_class'] !== null) {
$html .= HTML::tag_end('div');
}
$html .= HTML::tag_end('div');
$this->add(
@ -503,6 +522,13 @@ class UOJForm {
if (!$this->config['no_submit']) {
echo HTML::tag_begin('div', ['class' => $this->config['submit_container']['class']]);
if ($this->config['back_button']['href'] !== null) {
echo HTML::tag('a', [
'class' => $this->config['back_button']['class'],
'href' => $this->config['back_button']['href']
], '返回');
}
echo HTML::tag('button', [
'type' => 'submit',
'id' => "button-submit-{$this->form_name}",
@ -511,13 +537,6 @@ class UOJForm {
'class' => $this->config['submit_button']['class']
], $this->config['submit_button']['text']);
if ($this->config['back_button']['href'] !== null) {
echo HTML::tag('a', [
'class' => $this->config['back_button']['class'],
'href' => $this->config['back_button']['href']
], '返回');
}
echo HTML::tag_end('div');
}

View File

@ -663,6 +663,10 @@ class UOJProblem {
return "/var/uoj_data/{$this->info['id']}";
}
public function getUploadFolderPath() {
return "/var/uoj_data/upload/{$this->info['id']}";
}
public function getDataZipPath() {
return "/var/uoj_data/{$this->info['id']}.zip";
}
@ -672,6 +676,10 @@ class UOJProblem {
return "{$this->getDataFolderPath()}/$name";
}
public function getUploadFilePath($name = '') {
return "{$this->getUploadFolderPath()}/$name";
}
public function getResourcesFolderPath() {
return UOJContext::storagePath() . "/problem_resources/" . $this->info['id'];
}

View File

@ -0,0 +1,202 @@
<?php
class UOJProblemConfigure {
public UOJProblem $problem;
public UOJProblemConf $problem_conf;
public string $href;
public array $conf_keys;
public UOJForm $simple_form;
public static $supported_checkers = [
'ownchk' => '自定义校验器',
'ncmp' => 'ncmp: 单行或多行整数序列',
'wcmp' => 'wcmp: 单行或多行字符串序列',
'fcmp' => 'fcmp: 单行或多行数据(不忽略行末空格,但忽略文末回车)',
'bcmp' => 'bcmp: 逐字节比较',
'uncmp' => 'uncmp: 单行或多行整数集合',
'yesno' => 'yesno: YES、NO 序列(不区分大小写)',
'rcmp4' => 'rcmp4: 浮点数序列,绝对或相对误差在 1e-4 以内则视为答案正确',
'rcmp6' => 'rcmp6: 浮点数序列,绝对或相对误差在 1e-6 以内则视为答案正确',
'rcmp9' => 'rcmp9: 浮点数序列,绝对或相对误差在 1e-9 以内则视为答案正确',
];
public static $supported_score_types = [
'int' => '整数,每个测试点的部分分向下取整',
'real-0' => '整数,每个测试点的部分分四舍五入到整数',
'real-1' => '实数,四舍五入到小数点后 1 位',
'real-2' => '实数,四舍五入到小数点后 2 位',
'real-3' => '实数,四舍五入到小数点后 3 位',
'real-4' => '实数,四舍五入到小数点后 4 位',
'real-5' => '实数,四舍五入到小数点后 5 位',
'real-6' => '实数,四舍五入到小数点后 6 位',
'real-7' => '实数,四舍五入到小数点后 7 位',
'real-8' => '实数,四舍五入到小数点后 8 位',
];
private static function getCardHeader($title) {
return <<<EOD
<div class="col-12 col-md-6">
<div class="card">
<div class="card-header fw-bold">
{$title}
</div>
<div class="card-body vstack gap-3">
EOD;
}
private static function getCardFooter() {
return <<<EOD
</div>
</div>
</div>
EOD;
}
public function __construct(UOJProblem $problem) {
$this->problem = $problem;
$problem_conf = $this->problem->getProblemConf('data');
if (!($problem_conf instanceof UOJProblemConf)) {
$problem_conf = new UOJProblemConf([]);
}
$this->problem_conf = $problem_conf;
$this->href = "/problem/{$this->problem->info['id']}/manage/data";
$this->simple_form = new UOJForm('simple');
$this->simple_form->appendHTML(static::getCardHeader('基本信息'));
$this->addSelect($this->simple_form, 'use_builtin_judger', ['on' => '默认', 'off' => '自定义 Judger'], '测评逻辑', 'on');
$this->addSelect($this->simple_form, 'use_builtin_checker', self::$supported_checkers, '比对函数', 'ncmp');
$this->addSelect($this->simple_form, 'score_type', self::$supported_score_types, '测试点分数数值类型', 'int');
$this->simple_form->appendHTML(static::getCardFooter());
$this->simple_form->appendHTML(static::getCardHeader('数据配置'));
$this->addNumberInput($this->simple_form, 'n_tests', '数据点个数', 10);
$this->addNumberInput($this->simple_form, 'n_ex_tests', '额外数据点个数', 0);
$this->addNumberInput($this->simple_form, 'n_sample_tests', '样例数据点个数', 0);
$this->simple_form->appendHTML(static::getCardFooter());
$this->simple_form->appendHTML(static::getCardHeader('文件配置'));
$this->addTextInput($this->simple_form, 'input_pre', '输入文件名称', '');
$this->addTextInput($this->simple_form, 'input_suf', '输入文件后缀', '');
$this->addTextInput($this->simple_form, 'output_pre', '输出文件名称', '');
$this->addTextInput($this->simple_form, 'output_suf', '输出文件后缀', '');
$this->simple_form->appendHTML(static::getCardFooter());
$this->simple_form->appendHTML(static::getCardHeader('运行时限制'));
$this->addTimeLimitInput($this->simple_form, 'time_limit', '时间限制', 1, ['help' => '单位为秒,至多三位小数。']);
$this->addNumberInput($this->simple_form, 'memory_limit', '内存限制', 256, ['help' => '单位为 MiB。']);
$this->addNumberInput($this->simple_form, 'output_limit', '输出长度限制', 64, ['help' => '单位为 MiB。']);
$this->simple_form->appendHTML(static::getCardFooter());
$this->simple_form->succ_href = $this->href;
$this->simple_form->config['form']['class'] = 'row gy-3 mt-2';
$this->simple_form->config['submit_container']['class'] = 'col-12 text-center mt-3';
$this->simple_form->config['back_button']['href'] = $this->href;
$this->simple_form->config['back_button']['class'] = 'btn btn-secondary me-2';
$this->simple_form->handle = fn (&$vdata) => $this->onUpload($vdata);
}
public function addSelect(UOJForm $form, $key, $options, $label, $default_val = '', $cfg = []) {
$this->conf_keys[$key] = true;
$form->addSelect($key, [
'options' => $options,
'label' => $label,
'div_class' => 'row',
'label_class' => 'col-form-label col-4',
'select_div_class' => 'col-8',
'default_value' => $this->problem_conf->getVal($key, $default_val),
] + $cfg);
}
public function addNumberInput(UOJForm $form, $key, $label, $default_val = '', $cfg = []) {
$this->conf_keys[$key] = true;
$form->addInput($key, [
'type' => 'number',
'label' => $label,
'div_class' => 'row',
'label_class' => 'col-form-label col-4',
'input_div_class' => 'col-8',
'default_value' => $this->problem_conf->getVal($key, $default_val),
'validator_php' => function ($x) {
return validateInt($x) ? '' : '必须为一个整数';
},
] + $cfg);
}
public function addTimeLimitInput(UOJForm $form, $key, $label, $default_val = '', $cfg = []) {
$this->conf_keys[$key] = true;
$form->addInput($key, [
'type' => 'number',
'label' => $label,
'input_attrs' => ['step' => 0.001],
'div_class' => 'row',
'label_class' => 'col-form-label col-4',
'input_div_class' => 'col-8',
'default_value' => $this->problem_conf->getVal($key, $default_val),
'validator_php' => function ($x) {
if (!validateUFloat($x)) {
return '必须为整数或小数,且值大于等于零';
} elseif (round($x * 1000) != $x * 1000) {
return '至多包含三位小数';
} else {
return '';
}
},
] + $cfg);
}
public function addTextInput(UOJForm $form, $key, $label, $default_val = '', $cfg = []) {
$this->conf_keys[$key] = true;
$form->addInput($key, [
'label' => $label,
'div_class' => 'row',
'label_class' => 'col-form-label col-4',
'input_div_class' => 'col-8',
'default_value' => $this->problem_conf->getVal($key, $default_val),
'validator_php' => function ($x) {
return ctype_graph($x) ? '' : '必须仅包含除空格以外的可见字符';
},
] + $cfg);
}
public function runAtServer() {
$this->simple_form->runAtServer();
}
public function onUpload(array &$vdata) {
$conf = $this->problem_conf->conf;
foreach (array_keys($this->conf_keys) as $key) {
$val = UOJRequest::post($key);
if ($key === 'use_builtin_judger') {
if ($val === 'off') {
unset($conf[$key]);
} else {
$conf[$key] = $val;
}
} elseif ($key === 'use_builtin_checker') {
if ($val === 'ownchk') {
unset($conf[$key]);
} else {
$conf[$key] = $val;
}
} else {
if ($val !== '') {
$conf[$key] = $val;
}
}
}
$err = dataUpdateProblemConf($this->problem->info, $conf);
if ($err) {
UOJResponse::message('<div>' . $err . '</div><a href="' . $this->href . '">返回</a>');
}
}
public function printHTML() {
$this->simple_form->printHTML();
}
}

View File

@ -27,6 +27,7 @@ Route::group(
Route::any('/problem/{id}/manage/statement', '/problem_statement_manage.php');
Route::any('/problem/{id}/manage/permissions', '/problem_permissions_manage.php');
Route::any('/problem/{id}/manage/data', '/problem_data_manage.php');
Route::any('/problem/{id}/manage/data/configure', '/problem_data_configure.php');
Route::any('/download/testlib.h', '/download.php?type=testlib.h');
Route::any('/download/problem/{id}/data.zip', '/download.php?type=problem');
Route::any('/download/problem/{id}/attachment.zip', '/download.php?type=attachment');