2015-06-26 07:59:53 +00:00
|
|
|
|
# coding=utf-8
|
|
|
|
|
from django.db import models
|
2015-08-23 19:36:17 +00:00
|
|
|
|
from django.utils.timezone import now
|
2015-06-26 07:59:53 +00:00
|
|
|
|
|
2015-07-18 09:17:33 +00:00
|
|
|
|
from account.models import User
|
2015-06-26 07:59:53 +00:00
|
|
|
|
from problem.models import AbstractProblem
|
2015-08-18 12:12:27 +00:00
|
|
|
|
from group.models import Group
|
2015-09-24 06:55:20 +00:00
|
|
|
|
from utils.models import RichTextField, JsonField
|
|
|
|
|
from judge.judger.result import result
|
2015-09-22 09:03:53 +00:00
|
|
|
|
|
2015-06-26 07:59:53 +00:00
|
|
|
|
|
2015-09-09 11:39:42 +00:00
|
|
|
|
GROUP_CONTEST = 0
|
|
|
|
|
PUBLIC_CONTEST = 1
|
2015-09-13 08:29:48 +00:00
|
|
|
|
PASSWORD_PROTECTED_CONTEST = 2
|
2015-09-09 11:39:42 +00:00
|
|
|
|
|
2015-09-21 03:16:30 +00:00
|
|
|
|
CONTEST_NOT_START = 1
|
|
|
|
|
CONTEST_ENDED = -1
|
|
|
|
|
CONTEST_UNDERWAY = 0
|
|
|
|
|
|
2015-09-09 11:39:42 +00:00
|
|
|
|
|
2015-07-17 03:00:15 +00:00
|
|
|
|
class Contest(models.Model):
|
2015-08-19 09:53:43 +00:00
|
|
|
|
title = models.CharField(max_length=40, unique=True)
|
2015-09-22 09:03:53 +00:00
|
|
|
|
description = RichTextField()
|
2015-08-22 12:46:52 +00:00
|
|
|
|
# 比赛模式:0 即为是acm模式,1 即为是按照总的 ac 题目数量排名模式
|
2015-08-08 15:04:08 +00:00
|
|
|
|
mode = models.IntegerField()
|
2015-09-12 09:08:29 +00:00
|
|
|
|
# 是否显示实时排名结果
|
|
|
|
|
real_time_rank = models.BooleanField()
|
2015-08-18 12:12:27 +00:00
|
|
|
|
# 是否显示别人的提交记录
|
|
|
|
|
show_user_submission = models.BooleanField()
|
|
|
|
|
# 只能超级管理员创建公开赛,管理员只能创建小组内部的比赛
|
|
|
|
|
# 如果这一项不为空,即为有密码的公开赛,没有密码的可以为小组赛或者是公开赛(此时用比赛的类型来表示)
|
2015-07-18 09:17:33 +00:00
|
|
|
|
password = models.CharField(max_length=30, blank=True, null=True)
|
2015-09-09 11:39:42 +00:00
|
|
|
|
# 比赛的类型: 0 即为是小组赛(GROUP_CONTEST),1 即为是无密码的公开赛(PUBLIC_CONTEST),
|
|
|
|
|
# 2 即为是有密码的公开赛(PASSWORD_PUBLIC_CONTEST)
|
2015-08-18 12:12:27 +00:00
|
|
|
|
contest_type = models.IntegerField()
|
2015-08-08 15:04:08 +00:00
|
|
|
|
# 开始时间
|
2015-07-18 09:17:33 +00:00
|
|
|
|
start_time = models.DateTimeField()
|
2015-08-08 15:04:08 +00:00
|
|
|
|
# 结束时间
|
2015-07-18 09:17:33 +00:00
|
|
|
|
end_time = models.DateTimeField()
|
2015-08-08 15:04:08 +00:00
|
|
|
|
# 创建时间
|
|
|
|
|
create_time = models.DateTimeField(auto_now_add=True)
|
|
|
|
|
# 最后修改时间
|
|
|
|
|
last_updated_time = models.DateTimeField(auto_now=True)
|
2015-08-18 12:12:27 +00:00
|
|
|
|
# 这个比赛是谁创建的
|
2015-07-18 09:17:33 +00:00
|
|
|
|
created_by = models.ForeignKey(User)
|
2015-08-18 12:12:27 +00:00
|
|
|
|
groups = models.ManyToManyField(Group)
|
2015-08-21 10:16:34 +00:00
|
|
|
|
# 是否可见 false的话相当于删除
|
|
|
|
|
visible = models.BooleanField(default=True)
|
2015-08-18 12:12:27 +00:00
|
|
|
|
|
2015-08-23 19:36:17 +00:00
|
|
|
|
@property
|
|
|
|
|
def status(self):
|
|
|
|
|
if self.start_time > now():
|
|
|
|
|
# 没有开始 返回1
|
2015-09-21 03:16:30 +00:00
|
|
|
|
return CONTEST_NOT_START
|
2015-08-23 19:36:17 +00:00
|
|
|
|
elif self.end_time < now():
|
2015-09-22 04:56:10 +00:00
|
|
|
|
# 已经结束 返回-1
|
2015-09-21 03:16:30 +00:00
|
|
|
|
return CONTEST_ENDED
|
2015-08-23 19:36:17 +00:00
|
|
|
|
else:
|
|
|
|
|
# 正在进行 返回0
|
2015-09-21 03:16:30 +00:00
|
|
|
|
return CONTEST_UNDERWAY
|
2015-08-23 19:36:17 +00:00
|
|
|
|
|
2015-08-08 15:04:08 +00:00
|
|
|
|
class Meta:
|
|
|
|
|
db_table = "contest"
|
|
|
|
|
|
2015-07-17 03:00:15 +00:00
|
|
|
|
|
|
|
|
|
class ContestProblem(AbstractProblem):
|
|
|
|
|
contest = models.ForeignKey(Contest)
|
2015-08-08 15:04:08 +00:00
|
|
|
|
# 比如A B 或者1 2 或者 a b 将按照这个排序
|
|
|
|
|
sort_index = models.CharField(max_length=30)
|
2015-08-24 04:21:46 +00:00
|
|
|
|
score = models.IntegerField(default=0)
|
2015-08-08 15:04:08 +00:00
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
db_table = "contest_problem"
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ContestProblemTestCase(models.Model):
|
|
|
|
|
"""
|
|
|
|
|
如果比赛是按照通过的测试用例总分计算的话,就需要这个model 记录每个测试用例的分数
|
|
|
|
|
"""
|
|
|
|
|
# 测试用例的id 这个还在测试用例的配置文件里面有对应
|
|
|
|
|
id = models.CharField(max_length=40, primary_key=True, db_index=True)
|
|
|
|
|
problem = models.ForeignKey(ContestProblem)
|
|
|
|
|
score = models.IntegerField()
|
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
db_table = "contest_problem_test_case"
|
2015-08-23 10:28:30 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ContestSubmission(models.Model):
|
|
|
|
|
"""
|
|
|
|
|
用于保存比赛提交和排名的一些数据,加快检索速度
|
|
|
|
|
"""
|
|
|
|
|
user = models.ForeignKey(User)
|
|
|
|
|
problem = models.ForeignKey(ContestProblem)
|
|
|
|
|
contest = models.ForeignKey(Contest)
|
|
|
|
|
total_submission_number = models.IntegerField(default=1)
|
|
|
|
|
# 这道题是 AC 还是没过
|
|
|
|
|
ac = models.BooleanField()
|
2015-09-13 13:31:38 +00:00
|
|
|
|
# ac 用时以秒计
|
2015-09-10 07:35:27 +00:00
|
|
|
|
ac_time = models.IntegerField(default=0)
|
2015-08-23 10:28:30 +00:00
|
|
|
|
# 总的时间,用于acm 类型的,也需要保存罚时
|
|
|
|
|
total_time = models.IntegerField(default=0)
|
2015-09-13 11:49:59 +00:00
|
|
|
|
# 第一个解出此题目
|
2015-09-13 12:00:42 +00:00
|
|
|
|
first_achieved = models.BooleanField(default=False)
|
2015-08-23 10:28:30 +00:00
|
|
|
|
|
|
|
|
|
class Meta:
|
|
|
|
|
db_table = "contest_submission"
|
2015-09-24 06:55:20 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class ContestRank(models.Model):
|
|
|
|
|
user = models.ForeignKey(User)
|
|
|
|
|
contest = models.ForeignKey(Contest)
|
|
|
|
|
total_submission_number = models.IntegerField(default=0)
|
|
|
|
|
total_ac_number = models.IntegerField(default=0)
|
|
|
|
|
# ac 的题目才要加到这个字段里面 = ac 时间 + 错误次数 * 20 * 60
|
|
|
|
|
# 没有 ac 的题目不计算罚时 单位是秒
|
|
|
|
|
total_time = models.IntegerField(default=0)
|
|
|
|
|
# 数据结构{23: {"is_ac": True, "ac_time": 8999, "error_number": 2, "is_first_ac": True}}
|
|
|
|
|
# key 是比赛题目的id
|
|
|
|
|
submission_info = JsonField(default={})
|
|
|
|
|
|
|
|
|
|
def update_rank(self, submission):
|
|
|
|
|
if not submission.contest_id or submission.contest_id != self.contest_id:
|
|
|
|
|
raise ValueError("Error submission type")
|
|
|
|
|
|
|
|
|
|
# 这道题以前提交过
|
|
|
|
|
if submission.problem_id in self.problem_info:
|
|
|
|
|
info = self.submission_info[submission.problem_id]
|
|
|
|
|
# 如果这道题目已经 ac 了就跳过
|
|
|
|
|
if info["is_ac"]:
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
self.total_submission_number += 1
|
|
|
|
|
|
|
|
|
|
if submission.result == result["accepted"]:
|
|
|
|
|
|
|
|
|
|
self.total_ac_number += 1
|
|
|
|
|
|
|
|
|
|
info["is_ac"] = True
|
|
|
|
|
info["ac_time"] = (submission.create_time - self.contest.start_time).total_seconds()
|
|
|
|
|
|
|
|
|
|
# 之前已经提交过,但是是错误的,这次提交是正确的。错误的题目不计入罚时
|
|
|
|
|
self.total_time += (info["ac_time"] + info["error_time"] * 20 * 60)
|
|
|
|
|
problem = ContestProblem.objects.get(id=submission.problem_id)
|
|
|
|
|
if problem.total_accepted_number == 0:
|
|
|
|
|
info["is_first_ac"] = True
|
|
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
info["error_number"] += 1
|
|
|
|
|
info["is_ac"] = False
|
|
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
# 第一次提交这道题目
|
|
|
|
|
self.total_submission_number += 1
|
|
|
|
|
info = {"is_ac": False, "ac_time": 0, "error_number": 0, "is_first_ac": False}
|
|
|
|
|
if submission.result == result["accepted"]:
|
|
|
|
|
self.total_ac_number += 1
|
|
|
|
|
info["is_ac"] = True
|
|
|
|
|
info["ac_time"] = (submission.create_time - self.contest.start_time).total_seconds()
|
|
|
|
|
self.total_time += info["ac_time"]
|
|
|
|
|
problem = ContestProblem.objects.get(id=submission.problem_id)
|
|
|
|
|
|
|
|
|
|
if problem.total_accepted_number == 0:
|
|
|
|
|
info["is_first_ac"] = True
|
|
|
|
|
|
|
|
|
|
else:
|
|
|
|
|
info["is_ac"] = False
|
|
|
|
|
info["error_number"] = 1
|
|
|
|
|
self.submission_info[submission.problem_id] = info
|
|
|
|
|
self.save()
|