mirror of
https://github.com/renbaoshuo/S2OJ.git
synced 2025-01-11 09:41:53 +00:00
183 lines
5.6 KiB
PHP
183 lines
5.6 KiB
PHP
<?php
|
|
|
|
class UOJRemoteProblem {
|
|
static $providers = [
|
|
'codeforces' => [
|
|
'name' => 'Codeforces',
|
|
'short_name' => 'CF',
|
|
'url' => 'https://codeforces.com',
|
|
'not_exists_texts' => [
|
|
'<th>Actions</th>',
|
|
'Statement is not available on English language',
|
|
'ограничение по времени на тест',
|
|
],
|
|
'languages' => ['C', 'C++', 'C++17', 'C++20', 'Java17', 'Pascal', 'Python2', 'Python3'],
|
|
],
|
|
];
|
|
|
|
static function getCodeforcesProblemUrl($id) {
|
|
if (str_starts_with($id, 'GYM')) {
|
|
return static::$providers['codeforces']['url'] . '/gym/' . preg_replace_callback('/GYM([1-9][0-9]{0,5})([A-Z][1-9]?)/', fn ($matches) => $matches[1] . '/problem/' . $matches[2], $id);
|
|
}
|
|
|
|
return static::$providers['codeforces']['url'] . '/problemset/problem/' . preg_replace_callback('/([1-9][0-9]{0,5})([A-Z][1-9]?)/', fn ($matches) => $matches[1] . '/' . $matches[2], $id);
|
|
}
|
|
|
|
static function getCodeforcesProblemBasicInfoFromHtml($id, $html) {
|
|
$remote_provider = static::$providers['codeforces'];
|
|
|
|
$html = preg_replace('/\$\$\$/', '$', $html);
|
|
$dom = new \IvoPetkov\HTML5DOMDocument();
|
|
$dom->loadHTML($html);
|
|
|
|
$judgestatement = $dom->querySelector('html')->innerHTML;
|
|
|
|
foreach ($remote_provider['not_exists_texts'] as $text) {
|
|
if (str_contains($judgestatement, $text)) {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
$statement_dom = $dom->querySelector('.problem-statement');
|
|
$title_prefix = str_starts_with($id, 'GYM') ? '' : 'CF';
|
|
$title = explode('. ', trim($statement_dom->querySelector('.title')->innerHTML))[1];
|
|
$title = "【{$title_prefix}{$id}】{$title}";
|
|
$time_limit = intval(substr($statement_dom->querySelector('.time-limit')->innerHTML, 53));
|
|
$memory_limit = intval(substr($statement_dom->querySelector('.memory-limit')->innerHTML, 55));
|
|
$difficulty = -1;
|
|
|
|
foreach ($dom->querySelectorAll('.tag-box') as &$elem) {
|
|
$matches = [];
|
|
|
|
if (preg_match('/\*([0-9]{3,4})/', trim($elem->innerHTML), $matches)) {
|
|
$difficulty = intval($matches[1]);
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ($difficulty != -1) {
|
|
$closest = null;
|
|
|
|
foreach (UOJProblem::$difficulty as $val) {
|
|
if ($closest === null || abs($val - $difficulty) < abs($closest - $difficulty)) {
|
|
$closest = $val;
|
|
}
|
|
}
|
|
|
|
$difficulty = $closest;
|
|
}
|
|
|
|
$statement_dom->removeChild($statement_dom->querySelector('.header'));
|
|
$statement_dom->childNodes->item(0)->insertBefore($dom->createElement('h3', 'Description'), $statement_dom->childNodes->item(0)->childNodes->item(0));
|
|
|
|
foreach ($statement_dom->querySelectorAll('.section-title') as &$elem) {
|
|
$elem->outerHTML = '<h3>' . $elem->innerHTML . '</h3>';
|
|
}
|
|
|
|
$sample_input_cnt = 0;
|
|
$sample_output_cnt = 0;
|
|
|
|
foreach ($statement_dom->querySelectorAll('.input') as &$input_dom) {
|
|
$sample_input_cnt++;
|
|
$input_text = '';
|
|
|
|
if ($input_dom->querySelector('.test-example-line')) {
|
|
foreach ($input_dom->querySelectorAll('.test-example-line') as &$line) {
|
|
$input_text .= HTML::stripTags($line->innerHTML) . "\n";
|
|
}
|
|
} else {
|
|
$input_text = HTML::stripTags($input_dom->querySelector('pre')->innerHTML);
|
|
}
|
|
|
|
$input_dom->outerHTML = HTML::tag('h4', [], "Input #{$sample_input_cnt}") . HTML::tag('pre', [], HTML::tag('code', [], $input_text));
|
|
}
|
|
|
|
foreach ($statement_dom->querySelectorAll('.output') as &$output_dom) {
|
|
$sample_output_cnt++;
|
|
$output_text = '';
|
|
|
|
if ($output_dom->querySelector('.test-example-line')) {
|
|
foreach ($output_dom->querySelectorAll('.test-example-line') as &$line) {
|
|
$output_text .= HTML::stripTags($line->innerHTML) . "\n";
|
|
}
|
|
} else {
|
|
$output_text = HTML::stripTags($output_dom->querySelector('pre')->innerHTML);
|
|
}
|
|
|
|
$output_dom->outerHTML = HTML::tag('h4', [], "Output #{$sample_output_cnt}") . HTML::tag('pre', [], HTML::tag('code', [], $output_text));
|
|
}
|
|
|
|
return [
|
|
'type' => 'html',
|
|
'title' => $title,
|
|
'time_limit' => $time_limit,
|
|
'memory_limit' => $memory_limit,
|
|
'difficulty' => $difficulty,
|
|
'statement' => $statement_dom->innerHTML,
|
|
];
|
|
}
|
|
|
|
// 传入 ID 需确保有效
|
|
static function getCodeforcesProblemBasicInfo($id) {
|
|
$curl = new Curl();
|
|
$curl->setUserAgent('Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/100.0.0.0 Safari/537.36 S2OJ/3.1.0');
|
|
|
|
$res = retry_loop(function () use (&$curl, $id) {
|
|
$curl->get(static::getCodeforcesProblemUrl($id));
|
|
|
|
if ($curl->error) {
|
|
return false;
|
|
}
|
|
|
|
return [
|
|
'content-type' => $curl->response_headers['Content-Type'],
|
|
'response' => $curl->response,
|
|
];
|
|
});
|
|
|
|
if (!$res) return null;
|
|
|
|
if (str_starts_with($res['content-type'], 'text/html')) {
|
|
return static::getCodeforcesProblemBasicInfoFromHtml($id, $res['response']);
|
|
} else if (str_starts_with($res['content-type'], 'application/pdf')) {
|
|
$title_prefix = str_starts_with($id, 'GYM') ? '' : 'CF';
|
|
$title = "【{$title_prefix}{$id}】{$title_prefix}{$id}";
|
|
|
|
return [
|
|
'type' => 'pdf',
|
|
'title' => $title,
|
|
'time_limit' => null,
|
|
'memory_limit' => null,
|
|
'difficulty' => -1,
|
|
'statement' => HTML::tag('h3', [], '提示') .
|
|
HTML::tag(
|
|
'p',
|
|
[],
|
|
'本题题面为 PDF 题面,请' .
|
|
HTML::tag('a', ['href' => static::getCodeforcesProblemUrl($id), 'target' => '_blank'], '点此') .
|
|
'以查看题面。'
|
|
),
|
|
];
|
|
} else {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public static function getProblemRemoteUrl($oj, $id) {
|
|
if ($oj === 'codeforces') {
|
|
return static::getCodeforcesProblemUrl($id);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
public static function getProblemBasicInfo($oj, $id) {
|
|
if ($oj === 'codeforces') {
|
|
return static::getCodeforcesProblemBasicInfo($id);
|
|
}
|
|
|
|
return null;
|
|
}
|
|
}
|