merge problem and contest_problem
This commit is contained in:
36
submission/migrations/0007_auto_20170923_1318.py
Normal file
36
submission/migrations/0007_auto_20170923_1318.py
Normal file
@@ -0,0 +1,36 @@
|
||||
# -*- coding: utf-8 -*-
|
||||
# Generated by Django 1.11.4 on 2017-09-23 13:18
|
||||
from __future__ import unicode_literals
|
||||
|
||||
from django.db import migrations, models
|
||||
import django.db.models.deletion
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('submission', '0006_auto_20170830_1154'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AlterField(
|
||||
model_name='submission',
|
||||
name='contest_id',
|
||||
field=models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, to='contest.Contest'),
|
||||
),
|
||||
migrations.AlterField(
|
||||
model_name='submission',
|
||||
name='problem_id',
|
||||
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='problem.Problem'),
|
||||
),
|
||||
migrations.RenameField(
|
||||
model_name='submission',
|
||||
old_name='contest_id',
|
||||
new_name='contest',
|
||||
),
|
||||
migrations.RenameField(
|
||||
model_name='submission',
|
||||
old_name='problem_id',
|
||||
new_name='problem',
|
||||
),
|
||||
]
|
||||
@@ -1,6 +1,8 @@
|
||||
from django.db import models
|
||||
from jsonfield import JSONField
|
||||
from account.models import AdminType
|
||||
from problem.models import Problem
|
||||
from contest.models import Contest
|
||||
|
||||
from utils.shortcuts import rand_str
|
||||
|
||||
@@ -21,8 +23,8 @@ class JudgeStatus:
|
||||
|
||||
class Submission(models.Model):
|
||||
id = models.CharField(max_length=32, default=rand_str, primary_key=True, db_index=True)
|
||||
contest_id = models.IntegerField(db_index=True, null=True)
|
||||
problem_id = models.IntegerField(db_index=True)
|
||||
contest = models.ForeignKey(Contest, null=True)
|
||||
problem = models.ForeignKey(Problem)
|
||||
create_time = models.DateTimeField(auto_now_add=True)
|
||||
user_id = models.IntegerField(db_index=True)
|
||||
username = models.CharField(max_length=30)
|
||||
|
||||
@@ -18,13 +18,13 @@ class SubmissionModelSerializer(serializers.ModelSerializer):
|
||||
model = Submission
|
||||
|
||||
|
||||
# 不显示submission info详情的serializer
|
||||
# 不显示submission info的serializer, 用于ACM rule_type
|
||||
class SubmissionSafeSerializer(serializers.ModelSerializer):
|
||||
statistic_info = serializers.JSONField()
|
||||
|
||||
class Meta:
|
||||
model = Submission
|
||||
exclude = ("info", "contest_id")
|
||||
exclude = ("info", "contest")
|
||||
|
||||
|
||||
class SubmissionListSerializer(serializers.ModelSerializer):
|
||||
@@ -37,7 +37,7 @@ class SubmissionListSerializer(serializers.ModelSerializer):
|
||||
|
||||
class Meta:
|
||||
model = Submission
|
||||
exclude = ("info", "contest_id", "code")
|
||||
exclude = ("info", "contest", "code")
|
||||
|
||||
def get_show_link(self, obj):
|
||||
# 没传user或为匿名user
|
||||
|
||||
57
submission/tests.py
Normal file
57
submission/tests.py
Normal file
@@ -0,0 +1,57 @@
|
||||
from unittest import mock
|
||||
from copy import deepcopy
|
||||
|
||||
from .models import Submission
|
||||
from problem.models import Problem, ProblemTag
|
||||
from utils.api.tests import APITestCase
|
||||
|
||||
DEFAULT_PROBLEM_DATA = {"_id": "110", "title": "test", "description": "<p>test</p>", "input_description": "test",
|
||||
"output_description": "test", "time_limit": 1000, "memory_limit": 256, "difficulty": "Low",
|
||||
"visible": True, "tags": ["test"], "languages": ["C", "C++", "Java", "Python2"], "template": {},
|
||||
"samples": [{"input": "test", "output": "test"}], "spj": False, "spj_language": "C",
|
||||
"spj_code": "", "test_case_id": "499b26290cc7994e0b497212e842ea85",
|
||||
"test_case_score": [{"output_name": "1.out", "input_name": "1.in", "output_size": 0,
|
||||
"stripped_output_md5": "d41d8cd98f00b204e9800998ecf8427e",
|
||||
"input_size": 0, "score": 0}],
|
||||
"rule_type": "ACM", "hint": "<p>test</p>", "source": "test"}
|
||||
|
||||
DEFAULT_SUBMISSION_DATA = {
|
||||
"problem_id": "110",
|
||||
"user_id": 1,
|
||||
"username": "test",
|
||||
"code": "xxxxxxxxxxxxxx",
|
||||
"result": -2,
|
||||
"info": {},
|
||||
"language": "C",
|
||||
"statistic_info": {}
|
||||
}
|
||||
|
||||
|
||||
class SubmissionListTest(APITestCase):
|
||||
def setUp(self):
|
||||
self.create_user("123", "345")
|
||||
self.url = self.reverse("submission_list_api")
|
||||
Submission.objects.create(**DEFAULT_SUBMISSION_DATA)
|
||||
|
||||
def test_get_submission_list(self):
|
||||
resp = self.client.get(self.url, data={"limit": "10"})
|
||||
self.assertSuccess(resp)
|
||||
|
||||
|
||||
@mock.patch("submission.views.oj.judge_task.delay")
|
||||
class SubmissionAPITest(APITestCase):
|
||||
def setUp(self):
|
||||
self.user = self.create_user("test", "test123")
|
||||
tag = ProblemTag.objects.create(name="test")
|
||||
problem_data = deepcopy(DEFAULT_PROBLEM_DATA)
|
||||
problem_data.pop("tags")
|
||||
problem_data["created_by"] = self.user
|
||||
self.problem = Problem.objects.create(**problem_data)
|
||||
self.problem.tags.add(tag)
|
||||
self.problem.save()
|
||||
self.url = self.reverse("submission_api")
|
||||
|
||||
def test_create_submission(self, judge_task):
|
||||
resp = self.client.post(self.url, DEFAULT_SUBMISSION_DATA)
|
||||
self.assertSuccess(resp)
|
||||
judge_task.assert_called()
|
||||
@@ -1,8 +1,7 @@
|
||||
|
||||
from account.decorators import login_required, check_contest_permission
|
||||
from judge.tasks import judge_task
|
||||
# from judge.dispatcher import JudgeDispatcher
|
||||
from problem.models import Problem, ProblemRuleType, ContestProblem
|
||||
from problem.models import Problem, ProblemRuleType
|
||||
from contest.models import Contest, ContestStatus
|
||||
from utils.api import APIView, validate_serializer
|
||||
from utils.throttling import TokenBucket, BucketController
|
||||
@@ -20,16 +19,15 @@ def _submit(response, user, problem_id, language, code, contest_id):
|
||||
bucket = TokenBucket(fill_rate=10, capacity=20,
|
||||
last_capacity=controller.last_capacity,
|
||||
last_timestamp=controller.last_timestamp)
|
||||
|
||||
if bucket.consume():
|
||||
controller.last_capacity -= 1
|
||||
else:
|
||||
return response.error("Please wait %d seconds" % int(bucket.expected_time() + 1))
|
||||
|
||||
try:
|
||||
if contest_id:
|
||||
problem = ContestProblem.objects.get(_id=problem_id, visible=True)
|
||||
else:
|
||||
problem = Problem.objects.get(_id=problem_id, visible=True)
|
||||
problem = Problem.objects.get(_id=problem_id,
|
||||
contest_id=contest_id,
|
||||
visible=True)
|
||||
except Problem.DoesNotExist:
|
||||
return response.error("Problem not exist")
|
||||
|
||||
@@ -37,11 +35,11 @@ def _submit(response, user, problem_id, language, code, contest_id):
|
||||
username=user.username,
|
||||
language=language,
|
||||
code=code,
|
||||
problem_id=problem._id,
|
||||
problem_id=problem.id,
|
||||
contest_id=contest_id)
|
||||
# use this for debug
|
||||
# JudgeDispatcher(submission.id, problem._id).judge()
|
||||
judge_task.delay(submission.id, problem._id)
|
||||
# JudgeDispatcher(submission.id, problem.id).judge()
|
||||
judge_task.delay(submission.id, problem.id)
|
||||
return response.success({"submission_id": submission.id})
|
||||
|
||||
|
||||
@@ -65,32 +63,21 @@ class SubmissionAPI(APIView):
|
||||
if not submission_id:
|
||||
return self.error("Parameter id doesn't exist.")
|
||||
try:
|
||||
submission = Submission.objects.get(id=submission_id)
|
||||
submission = Submission.objects.select_related("problem").get(id=submission_id)
|
||||
except Submission.DoesNotExist:
|
||||
return self.error("Submission doesn't exist.")
|
||||
if not submission.check_user_permission(request.user):
|
||||
return self.error("No permission for this submission.")
|
||||
|
||||
if submission.contest_id:
|
||||
# check problem'rule is ACM or IO.
|
||||
if ContestProblem.objects.filter(contest_id=submission.contest_id,
|
||||
_id=submission.problem_id,
|
||||
visible=True,
|
||||
rule_type=ProblemRuleType.ACM
|
||||
).exists():
|
||||
return self.success(SubmissionSafeSerializer(submission).data)
|
||||
return self.success(SubmissionModelSerializer(submission).data)
|
||||
|
||||
if Problem.objects.filter(_id=submission.problem_id,
|
||||
visible=True,
|
||||
rule_type=ProblemRuleType.ACM
|
||||
).exists():
|
||||
if submission.problem.rule_type == ProblemRuleType.ACM:
|
||||
return self.success(SubmissionSafeSerializer(submission).data)
|
||||
return self.success(SubmissionModelSerializer(submission).data)
|
||||
|
||||
|
||||
class SubmissionListAPI(APIView):
|
||||
def get(self, request):
|
||||
if not request.GET.get("limit"):
|
||||
return self.error("Limit is needed")
|
||||
if request.GET.get("contest_id"):
|
||||
return self._get_contest_submission_list(request)
|
||||
|
||||
@@ -107,7 +94,11 @@ class SubmissionListAPI(APIView):
|
||||
myself = request.GET.get("myself")
|
||||
result = request.GET.get("result")
|
||||
if problem_id:
|
||||
submissions = submissions.filter(problem_id=problem_id)
|
||||
try:
|
||||
problem = Problem.objects.get(_id=problem_id, visible=True)
|
||||
except Problem.DoesNotExist:
|
||||
return self.error("Problem doesn't exist")
|
||||
submissions = problem.submission_set.all()
|
||||
if myself and myself == "1":
|
||||
submissions = submissions.filter(user_id=request.user.id)
|
||||
if result:
|
||||
|
||||
Reference in New Issue
Block a user