2016-07-19 00:39:37 +08:00
< ? php
define ( " CONTEST_NOT_STARTED " , 0 );
define ( " CONTEST_IN_PROGRESS " , 1 );
define ( " CONTEST_PENDING_FINAL_TEST " , 2 );
define ( " CONTEST_TESTING " , 10 );
define ( " CONTEST_FINISHED " , 20 );
function genMoreContestInfo ( & $contest ) {
$contest [ 'start_time_str' ] = $contest [ 'start_time' ];
$contest [ 'start_time' ] = new DateTime ( $contest [ 'start_time' ]);
$contest [ 'end_time' ] = clone $contest [ 'start_time' ];
$contest [ 'end_time' ] -> add ( new DateInterval ( " PT ${ contest['last_min'] } M " ));
if ( $contest [ 'status' ] == 'unfinished' ) {
if ( UOJTime :: $time_now < $contest [ 'start_time' ]) {
$contest [ 'cur_progress' ] = CONTEST_NOT_STARTED ;
2020-06-25 20:41:16 +08:00
} elseif ( UOJTime :: $time_now < $contest [ 'end_time' ]) {
2016-07-19 00:39:37 +08:00
$contest [ 'cur_progress' ] = CONTEST_IN_PROGRESS ;
} else {
$contest [ 'cur_progress' ] = CONTEST_PENDING_FINAL_TEST ;
}
2020-06-25 20:41:16 +08:00
} elseif ( $contest [ 'status' ] == 'testing' ) {
2016-07-19 00:39:37 +08:00
$contest [ 'cur_progress' ] = CONTEST_TESTING ;
2020-06-25 20:41:16 +08:00
} elseif ( $contest [ 'status' ] == 'finished' ) {
2016-07-19 00:39:37 +08:00
$contest [ 'cur_progress' ] = CONTEST_FINISHED ;
}
$contest [ 'extra_config' ] = json_decode ( $contest [ 'extra_config' ], true );
if ( ! isset ( $contest [ 'extra_config' ][ 'standings_version' ])) {
$contest [ 'extra_config' ][ 'standings_version' ] = 2 ;
}
}
function updateContestPlayerNum ( $contest ) {
DB :: update ( " update contests set player_num = (select count(*) from contests_registrants where contest_id = { $contest [ 'id' ] } ) where id = { $contest [ 'id' ] } " );
}
2018-07-09 10:40:30 +08:00
// problems: pos => id
// data : id, submit_time, submitter, problem_pos, score
2022-03-17 11:02:44 +08:00
// people : username
2018-07-09 10:40:30 +08:00
function queryContestData ( $contest , $config = array ()) {
mergeConfig ( $config , [
'pre_final' => false
]);
$problems = [];
$prob_pos = [];
$n_problems = 0 ;
$result = DB :: query ( " select problem_id from contests_problems where contest_id = { $contest [ 'id' ] } order by problem_id " );
while ( $row = DB :: fetch ( $result , MYSQLI_NUM )) {
$prob_pos [ $problems [] = ( int ) $row [ 0 ]] = $n_problems ++ ;
}
$data = [];
if ( $config [ 'pre_final' ]) {
$result = DB :: query ( " select id, submit_time, submitter, problem_id, result from submissions "
. " where contest_id = { $contest [ 'id' ] } and score is not null order by id " );
while ( $row = DB :: fetch ( $result , MYSQLI_NUM )) {
$r = json_decode ( $row [ 4 ], true );
if ( ! isset ( $r [ 'final_result' ])) {
continue ;
}
$row [ 0 ] = ( int ) $row [ 0 ];
$row [ 3 ] = $prob_pos [ $row [ 3 ]];
$row [ 4 ] = ( int )( $r [ 'final_result' ][ 'score' ]);
$data [] = $row ;
}
} else {
if ( $contest [ 'cur_progress' ] < CONTEST_FINISHED ) {
$result = DB :: query ( " select id, submit_time, submitter, problem_id, score from submissions "
. " where contest_id = { $contest [ 'id' ] } and score is not null order by id " );
} else {
$result = DB :: query ( " select submission_id, date_add(' { $contest [ 'start_time_str' ] } ', interval penalty second), "
. " submitter, problem_id, score from contests_submissions where contest_id = { $contest [ 'id' ] } " );
}
while ( $row = DB :: fetch ( $result , MYSQLI_NUM )) {
$row [ 0 ] = ( int ) $row [ 0 ];
$row [ 3 ] = $prob_pos [ $row [ 3 ]];
$row [ 4 ] = ( int ) $row [ 4 ];
$data [] = $row ;
}
}
$people = [];
2022-03-17 18:26:29 +08:00
$result = DB :: query ( " select a.username, b.realname from contests_registrants a inner join user_info b on a.username = b.username where a.contest_id = { $contest [ 'id' ] } and a.has_participated = 1 " );
2018-07-09 10:40:30 +08:00
while ( $row = DB :: fetch ( $result , MYSQLI_NUM )) {
$people [] = $row ;
}
return [ 'problems' => $problems , 'data' => $data , 'people' => $people ];
}
function calcStandings ( $contest , $contest_data , & $score , & $standings , $update_contests_submissions = false ) {
// score: username, problem_pos => score, penalty, id
$score = array ();
$n_people = count ( $contest_data [ 'people' ]);
$n_problems = count ( $contest_data [ 'problems' ]);
foreach ( $contest_data [ 'people' ] as $person ) {
$score [ $person [ 0 ]] = array ();
}
2020-06-25 20:41:16 +08:00
foreach ( $contest_data [ 'data' ] as $submission ) {
2018-07-09 10:40:30 +08:00
$penalty = ( new DateTime ( $submission [ 1 ])) -> getTimestamp () - $contest [ 'start_time' ] -> getTimestamp ();
if ( $contest [ 'extra_config' ][ 'standings_version' ] >= 2 ) {
if ( $submission [ 4 ] == 0 ) {
$penalty = 0 ;
}
}
$score [ $submission [ 2 ]][ $submission [ 3 ]] = array ( $submission [ 4 ], $penalty , $submission [ 0 ]);
}
2022-03-17 18:26:29 +08:00
// standings: rank => score, penalty, [username, realname], virtual_rank
2018-07-09 10:40:30 +08:00
$standings = array ();
foreach ( $contest_data [ 'people' ] as $person ) {
$cur = array ( 0 , 0 , $person );
for ( $i = 0 ; $i < $n_problems ; $i ++ ) {
if ( isset ( $score [ $person [ 0 ]][ $i ])) {
$cur_row = $score [ $person [ 0 ]][ $i ];
$cur [ 0 ] += $cur_row [ 0 ];
$cur [ 1 ] += $cur_row [ 1 ];
if ( $update_contests_submissions ) {
DB :: insert ( " insert into contests_submissions (contest_id, submitter, problem_id, submission_id, score, penalty) values ( { $contest [ 'id' ] } , ' { $person [ 0 ] } ', { $contest_data [ 'problems' ][ $i ] } , { $cur_row [ 2 ] } , { $cur_row [ 0 ] } , { $cur_row [ 1 ] } ) " );
}
}
}
$standings [] = $cur ;
}
usort ( $standings , function ( $lhs , $rhs ) {
if ( $lhs [ 0 ] != $rhs [ 0 ]) {
return $rhs [ 0 ] - $lhs [ 0 ];
2020-06-25 20:41:16 +08:00
} elseif ( $lhs [ 1 ] != $rhs [ 1 ]) {
2018-07-09 10:40:30 +08:00
return $lhs [ 1 ] - $rhs [ 1 ];
} else {
return strcmp ( $lhs [ 2 ][ 0 ], $rhs [ 2 ][ 0 ]);
}
});
$is_same_rank = function ( $lhs , $rhs ) {
return $lhs [ 0 ] == $rhs [ 0 ] && $lhs [ 1 ] == $rhs [ 1 ];
};
for ( $i = 0 ; $i < $n_people ; $i ++ ) {
if ( $i == 0 || ! $is_same_rank ( $standings [ $i - 1 ], $standings [ $i ])) {
$standings [ $i ][] = $i + 1 ;
} else {
$standings [ $i ][] = $standings [ $i - 1 ][ 3 ];
}
}
}