From ce9fbed01fa692d2478639f4fe8b24cb8e90d532 Mon Sep 17 00:00:00 2001 From: virusdefender <1670873886@qq.com> Date: Mon, 24 Aug 2015 03:34:42 +0800 Subject: [PATCH 01/16] =?UTF-8?q?=E6=88=91=E7=9A=84=E6=8F=90=E4=BA=A4?= =?UTF-8?q?=E9=A1=B5=E9=9D=A2=E5=88=A4=E6=96=AD=E9=A2=98=E7=9B=AE=E7=9A=84?= =?UTF-8?q?=E7=B1=BB=E5=9E=8B=EF=BC=8C=E7=94=9F=E6=88=90=E4=B8=8D=E5=90=8C?= =?UTF-8?q?=E7=9A=84=E9=93=BE=E6=8E=A5?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- template/oj/problem/my_submission.html | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/template/oj/problem/my_submission.html b/template/oj/problem/my_submission.html index 1aca690..f8927ba 100644 --- a/template/oj/problem/my_submission.html +++ b/template/oj/problem/my_submission.html @@ -11,7 +11,12 @@
{% include "oj/problem/_problem_header.html" %} From 1c37c4bfb6722f61ebf38ce4928adda567ab3f53 Mon Sep 17 00:00:00 2001 From: virusdefender <1670873886@qq.com> Date: Mon, 24 Aug 2015 03:35:26 +0800 Subject: [PATCH 02/16] =?UTF-8?q?=E6=AF=94=E8=B5=9B=E9=A2=98=E7=9B=AE?= =?UTF-8?q?=E8=AF=A6=E6=83=85=E9=A1=B5=EF=BC=8C=E6=8F=90=E7=A4=BA=E6=AF=94?= =?UTF-8?q?=E8=B5=9B=E7=8A=B6=E6=80=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- template/oj/contest/contest_problem.html | 164 ++++++++++++----------- 1 file changed, 84 insertions(+), 80 deletions(-) diff --git a/template/oj/contest/contest_problem.html b/template/oj/contest/contest_problem.html index 32b0617..907d81e 100644 --- a/template/oj/contest/contest_problem.html +++ b/template/oj/contest/contest_problem.html @@ -1,104 +1,108 @@ - {% extends 'oj_base.html' %} {% block body %} -
- -

{{ contest_problem.title }}

+
+ +

{{ contest_problem.title }}

-

发布时间 : {{ contest_problem.create_time }}   - 时间限制 : {{ contest_problem.time_limit }}ms   - 内存限制 : {{ contest_problem.memory_limit }}M -

+

发布时间 : {{ contest_problem.create_time }}   + 时间限制 : {{ contest_problem.time_limit }}ms   + 内存限制 : {{ contest_problem.memory_limit }}M +

-
-
- +
+
+ -

{{ contest_problem.description|safe }}

-
-
- +

{{ contest_problem.description|safe }}

+
+
+ -

{{ contest_problem.input_description }}

-
-
- +

{{ contest_problem.input_description }}

+
+
+ -

{{ contest_problem.output_description }}k

-
- {% for item in samples %} -
- +

{{ contest_problem.output_description }}k

+
+ {% for item in samples %} +
+
 {{ item.input }}
-
-
+
+
- +
 {{ item.output }}
-
- {% endfor %} - {% if problem.hint %} -
- +
+ {% endfor %} + {% if problem.hint %} +
+ -

{{ contest_problem.hint|safe }}

-
- {% endif %} - {% if contest_problem.hint %} -
- +

{{ contest_problem.hint|safe }}

+
+ {% endif %} + {% if contest_problem.hint %} +
+ -

{{ contest_problem.hint|safe }}

-
- {% endif %} +

{{ contest_problem.hint|safe }}

+
+ {% endif %} -
- + {% ifequal contest.status 0 %} +
+ -
- - - +
+ + + +
+
+ +
+ + +
+
+ +
+ + + +
+ {% endifequal %} + + {% if show_warning %} + + {% endif %} + +
+
- -
- - -
-
- -
- - - -
- {% if show_warning %} - - {% endif %} - -
-
-
-
{% endblock %} {% block js_block %} - + {% endblock %} \ No newline at end of file From 76b28f2da2a67797f49e5d589793d0e6bb5562e7 Mon Sep 17 00:00:00 2001 From: virusdefender <1670873886@qq.com> Date: Mon, 24 Aug 2015 03:36:17 +0800 Subject: [PATCH 03/16] =?UTF-8?q?=E4=BD=BF=E7=94=A8=20model=20=E7=9A=84=20?= =?UTF-8?q?instance=20method=20=E8=BF=9B=E8=A1=8C=E6=AF=94=E8=B5=9B?= =?UTF-8?q?=E7=8A=B6=E6=80=81=E7=9A=84=E5=88=A4=E6=96=AD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- contest/decorators.py | 2 +- contest/models.py | 13 +++++++++++++ utils/templatetags/contest.py | 16 ++++++++++------ 3 files changed, 24 insertions(+), 7 deletions(-) diff --git a/contest/decorators.py b/contest/decorators.py index 0b25759..df73274 100644 --- a/contest/decorators.py +++ b/contest/decorators.py @@ -74,7 +74,7 @@ def check_user_contest_permission(func): {"reason": "group_limited", "show_tab": False, "contest": contest}) # 比赛没有开始 - if contest.start_time > now(): + if contest.status == 1: if request.is_ajax(): return error_response(u"比赛还没有开始") else: diff --git a/contest/models.py b/contest/models.py index 524f2ed..de9f2e1 100644 --- a/contest/models.py +++ b/contest/models.py @@ -1,5 +1,6 @@ # coding=utf-8 from django.db import models +from django.utils.timezone import now from account.models import User from problem.models import AbstractProblem @@ -34,6 +35,18 @@ class Contest(models.Model): # 是否可见 false的话相当于删除 visible = models.BooleanField(default=True) + @property + def status(self): + if self.start_time > now(): + # 没有开始 返回1 + return 1 + elif self.end_time < now(): + # 已经结束 返回0 + return -1 + else: + # 正在进行 返回0 + return 0 + class Meta: db_table = "contest" diff --git a/utils/templatetags/contest.py b/utils/templatetags/contest.py index 04b144f..76bf768 100644 --- a/utils/templatetags/contest.py +++ b/utils/templatetags/contest.py @@ -4,19 +4,23 @@ from django.utils.timezone import now def get_contest_status(contest): - if contest.start_time > now(): + status = contest.status + if status == 1: return "没有开始" - if contest.end_time < now(): + elif status == -1: return "已经结束" - return "正在进行" + else: + return "正在进行" def get_contest_status_color(contest): - if contest.start_time > now(): + status = contest.status + if status == 1: return "info" - if contest.end_time < now(): + elif status == -1: return "warning" - return "success" + else: + return "success" from django import template From c06626cbc1c73987a74d6bbc848626eb5ec6afa0 Mon Sep 17 00:00:00 2001 From: virusdefender <1670873886@qq.com> Date: Mon, 24 Aug 2015 03:37:25 +0800 Subject: [PATCH 04/16] =?UTF-8?q?=E6=8F=90=E4=BA=A4=E9=A2=98=E7=9B=AE?= =?UTF-8?q?=E7=9A=84=E6=97=B6=E5=80=99=E5=A2=9E=E5=8A=A0=E5=AF=B9=E6=AF=94?= =?UTF-8?q?=E8=B5=9B=E7=8A=B6=E6=80=81=E7=9A=84=E5=88=A4=E6=96=AD?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- submission/views.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/submission/views.py b/submission/views.py index 6399e22..daac6f6 100644 --- a/submission/views.py +++ b/submission/views.py @@ -173,6 +173,8 @@ class ContestSubmissionAPIView(APIView): contest = Contest.objects.get(id=data["contest_id"]) except Contest.DoesNotExist: return error_response(u"比赛不存在") + if contest.status != 0: + return error_response(u"比赛已经结束或者还没有开始") try: problem = ContestProblem.objects.get(contest=contest, id=data["problem_id"]) # 更新题目提交计数器 From d0742960876b80b879136e007e7c5c262459919b Mon Sep 17 00:00:00 2001 From: virusdefender <1670873886@qq.com> Date: Mon, 24 Aug 2015 03:37:46 +0800 Subject: [PATCH 05/16] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=E6=AF=94=E8=B5=9B?= =?UTF-8?q?=E7=BB=9F=E8=AE=A1=E6=97=B6=E9=97=B4=E7=9A=84=20bug?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- mq/scripts/info.py | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/mq/scripts/info.py b/mq/scripts/info.py index 70843d0..661c3ff 100644 --- a/mq/scripts/info.py +++ b/mq/scripts/info.py @@ -75,10 +75,8 @@ class MessageQueue(object): except ContestSubmission.DoesNotExist: # 第一次提交 is_ac = submission.result == result["accepted"] - # 增加题目总提交数计数器 - contest_problem.total_submit_number += 1 if is_ac: - total_time = 0 + total_time = int((submission.create_time - contest.start_time).total_seconds() / 60) # 增加题目总的ac数计数器 contest_problem.total_accepted_number += 1 contest_problem.save() From fca5abaf6de6af2d5641a1c785ba4f2d29c930f2 Mon Sep 17 00:00:00 2001 From: virusdefender <1670873886@qq.com> Date: Mon, 24 Aug 2015 03:56:12 +0800 Subject: [PATCH 06/16] =?UTF-8?q?=E4=BF=AE=E5=A4=8D=20account=20=E9=87=8C?= =?UTF-8?q?=E9=9D=A2=E7=9A=84=E5=8D=95=E5=85=83=E6=B5=8B=E8=AF=95=E9=94=99?= =?UTF-8?q?=E8=AF=AF?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- account/tests.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/account/tests.py b/account/tests.py index 0b7a8bb..a75d2c5 100644 --- a/account/tests.py +++ b/account/tests.py @@ -280,7 +280,7 @@ class LoginRequiredDecoratorTest(TestCase): def test_cbv_without_args(self): # 没登陆 response = self.client.get("/login_required_test/cbv/1/") - self.assertTemplateUsed(response, "utils/error.html") + self.assertRedirects(response, "/login/") # 登陆后 self.client.login(username="test", password="test") @@ -353,7 +353,7 @@ class AdminRequiredDecoratorTest(TestCase): def test_cbv_without_args(self): # 没登陆 response = self.client.get("/admin_required_test/cbv/1/") - self.assertRedirects(response, "/login/") + self.assertTemplateUsed(response, "utils/error.html") # 登陆后 self.client.login(username="test", password="test") From 230610dc23faedad958164d54b353b892afa9ec6 Mon Sep 17 00:00:00 2001 From: virusdefender <1670873886@qq.com> Date: Mon, 24 Aug 2015 04:37:31 +0800 Subject: [PATCH 07/16] =?UTF-8?q?=E5=88=A0=E9=99=A4=E6=97=A0=E7=94=A8?= =?UTF-8?q?=E6=96=87=E4=BB=B6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- account/admin.py | 3 --- admin/admin.py | 3 --- announcement/admin.py | 3 --- contest/admin.py | 3 --- group/admin.py | 3 --- install/admin.py | 3 --- submission/admin.py | 3 --- 7 files changed, 21 deletions(-) delete mode 100644 account/admin.py delete mode 100644 admin/admin.py delete mode 100644 announcement/admin.py delete mode 100644 contest/admin.py delete mode 100644 group/admin.py delete mode 100644 install/admin.py delete mode 100644 submission/admin.py diff --git a/account/admin.py b/account/admin.py deleted file mode 100644 index 8c38f3f..0000000 --- a/account/admin.py +++ /dev/null @@ -1,3 +0,0 @@ -from django.contrib import admin - -# Register your models here. diff --git a/admin/admin.py b/admin/admin.py deleted file mode 100644 index 8c38f3f..0000000 --- a/admin/admin.py +++ /dev/null @@ -1,3 +0,0 @@ -from django.contrib import admin - -# Register your models here. diff --git a/announcement/admin.py b/announcement/admin.py deleted file mode 100644 index 8c38f3f..0000000 --- a/announcement/admin.py +++ /dev/null @@ -1,3 +0,0 @@ -from django.contrib import admin - -# Register your models here. diff --git a/contest/admin.py b/contest/admin.py deleted file mode 100644 index 8c38f3f..0000000 --- a/contest/admin.py +++ /dev/null @@ -1,3 +0,0 @@ -from django.contrib import admin - -# Register your models here. diff --git a/group/admin.py b/group/admin.py deleted file mode 100644 index 8c38f3f..0000000 --- a/group/admin.py +++ /dev/null @@ -1,3 +0,0 @@ -from django.contrib import admin - -# Register your models here. diff --git a/install/admin.py b/install/admin.py deleted file mode 100644 index 8c38f3f..0000000 --- a/install/admin.py +++ /dev/null @@ -1,3 +0,0 @@ -from django.contrib import admin - -# Register your models here. diff --git a/submission/admin.py b/submission/admin.py deleted file mode 100644 index 8c38f3f..0000000 --- a/submission/admin.py +++ /dev/null @@ -1,3 +0,0 @@ -from django.contrib import admin - -# Register your models here. From 5f6228cf6524b8df6213902d7122c867b1983a86 Mon Sep 17 00:00:00 2001 From: virusdefender <1670873886@qq.com> Date: Mon, 24 Aug 2015 04:37:55 +0800 Subject: [PATCH 08/16] =?UTF-8?q?=E5=A2=9E=E5=8A=A0=20admin=20=E7=9A=84?= =?UTF-8?q?=E6=B5=8B=E8=AF=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- admin/tests.py | 21 ++++++++++++++++++++- 1 file changed, 20 insertions(+), 1 deletion(-) diff --git a/admin/tests.py b/admin/tests.py index 8c88c3b..522e34c 100644 --- a/admin/tests.py +++ b/admin/tests.py @@ -1,5 +1,6 @@ # coding=utf-8 import json + from django.test import TestCase, Client from django.core.urlresolvers import reverse from django.http import HttpResponse @@ -11,7 +12,7 @@ def middleware_test_func(request): return HttpResponse(json.dumps({"code": 0})) -class AdminRequiredMidlewareTest(TestCase): +class AdminRequiredMiddlewareTest(TestCase): urls = "admin.test_urls" def setUp(self): @@ -66,3 +67,21 @@ class AdminRequiredMidlewareTest(TestCase): response = self.client.get(url, HTTP_X_REQUESTED_WITH='XMLHttpRequest') self.assertEqual(json.loads(response.content)["code"], 0) + +class AdminTemplateViewTest(TestCase): + def setUp(self): + super_admin_user = User.objects.create(username="test", admin_type=2) + super_admin_user.set_password("test") + super_admin_user.save() + + self.client = Client() + self.client.login(username="test", password="test") + + def test_file_exists(self): + response = self.client.get("/admin/template/index/index.html") + self.assertEqual(response.status_code, 200) + + def test_file_does_not_exist(self): + response = self.client.get("/admin/template/index/index123.html") + self.assertEqual(response.status_code, 200) + self.assertHTMLEqual(response.content, u"模板不存在") From 457845a41b10197e5ba03c4980c6cad4caed0eb8 Mon Sep 17 00:00:00 2001 From: virusdefender <1670873886@qq.com> Date: Mon, 24 Aug 2015 19:46:09 +0800 Subject: [PATCH 09/16] fiix typo --- admin/tests.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/admin/tests.py b/admin/tests.py index 522e34c..3023834 100644 --- a/admin/tests.py +++ b/admin/tests.py @@ -54,7 +54,7 @@ class AdminRequiredMiddlewareTest(TestCase): self.assertEqual(json.loads(response.content), {"code": 1, "data": u"请先登录"}) self.client.login(username="test", password="test") - rresponse = self.client.get(url, HTTP_X_REQUESTED_WITH='XMLHttpRequest') + response = self.client.get(url, HTTP_X_REQUESTED_WITH='XMLHttpRequest') self.assertEqual(json.loads(response.content), {"code": 1, "data": u"请先登录"}) self.client.logout() From 3d1f12f2a23632c5aa802e279ca535430fd78e57 Mon Sep 17 00:00:00 2001 From: virusdefender <1670873886@qq.com> Date: Mon, 24 Aug 2015 19:47:23 +0800 Subject: [PATCH 10/16] =?UTF-8?q?=E5=A2=9E=E5=8A=A0=E6=AF=94=E8=B5=9B?= =?UTF-8?q?=E6=8E=92=E9=A1=B5=E9=9D=A2=E7=9A=84=20url?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- oj/urls.py | 2 ++ 1 file changed, 2 insertions(+) diff --git a/oj/urls.py b/oj/urls.py index c1b3a57..45dc789 100644 --- a/oj/urls.py +++ b/oj/urls.py @@ -84,4 +84,6 @@ urlpatterns = [ url(r'^api/contest/password/$', ContestPasswordVerifyAPIView.as_view(), name="contest_password_verify_api"), url(r'^api/contest/submission/$', ContestSubmissionAPIView.as_view(), name="contest_submission_api"), + url(r'^contest/(?P\d+)/rank/$', "contest.views.contest_rank_page", name="contest_rank_page") + ] From 7b7e7131b834544a82c96aaea348be4b6986abce Mon Sep 17 00:00:00 2001 From: virusdefender <1670873886@qq.com> Date: Mon, 24 Aug 2015 19:48:00 +0800 Subject: [PATCH 11/16] =?UTF-8?q?=E5=8E=BB=E9=99=A4=E5=A4=9A=E4=BD=99?= =?UTF-8?q?=E7=9A=84=E5=BC=95=E7=94=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- static/src/js/app/admin/problem/submission_list.js | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/static/src/js/app/admin/problem/submission_list.js b/static/src/js/app/admin/problem/submission_list.js index 21c451b..8593557 100644 --- a/static/src/js/app/admin/problem/submission_list.js +++ b/static/src/js/app/admin/problem/submission_list.js @@ -1,4 +1,4 @@ -require(["jquery", "avalon", "csrfToken", "bsAlert", "formValidation"], function ($, avalon, csrfTokenHeader, bsAlert) { +require(["jquery", "avalon", "csrfToken", "bsAlert"], function ($, avalon, csrfTokenHeader, bsAlert) { avalon.ready(function () { avalon.vmodels.submissionList = null; @@ -40,10 +40,6 @@ require(["jquery", "avalon", "csrfToken", "bsAlert", "formValidation"], function }, getPage: function (page_index) { getPageData(page_index); - }, - - showSubmissionDetailPage: function (submissionId) { - } }); @@ -70,7 +66,9 @@ require(["jquery", "avalon", "csrfToken", "bsAlert", "formValidation"], function }); } + avalon.scan(); + }); - avalon.scan(); + }); \ No newline at end of file From 48eb2d1e9bfe8d6b9f00b2fdbfcd671aadf028ca Mon Sep 17 00:00:00 2001 From: hohoTT <609029365@qq.com> Date: Mon, 24 Aug 2015 21:03:30 +0800 Subject: [PATCH 12/16] =?UTF-8?q?=E5=89=8D=E5=8F=B0=E6=AF=94=E8=B5=9B?= =?UTF-8?q?=E3=80=81=E6=AF=94=E8=B5=9B=E9=A2=98=E7=9B=AE=E4=B8=8E=E6=99=AE?= =?UTF-8?q?=E9=80=9A=E9=A2=98=E7=9B=AE=E7=9A=84=E4=B8=80=E7=B3=BB=E5=88=97?= =?UTF-8?q?=E7=95=8C=E9=9D=A2?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- contest/views.py | 12 ++ contest_submission/__init__.py | 0 contest_submission/admin.py | 3 + contest_submission/migrations/__init__.py | 0 contest_submission/models.py | 3 + contest_submission/serializers.py | 11 ++ contest_submission/tests.py | 3 + contest_submission/views.py | 117 +++++++++++++++++ oj/settings.py | 1 + oj/urls.py | 81 +++++++----- static/src/css/oj.css | 2 +- submission/serializers.py | 5 - submission/tests.py | 104 ++++++++++++++- submission/views.py | 82 ++++-------- template/oj/contest/contest_problem.html | 2 +- .../oj/contest/contest_problems_list.html | 122 ++++++++---------- template/oj/contest/my_submission.html | 71 ++++++++++ template/oj/contest/my_submissions_list.html | 12 +- template/oj/contest/submissions_list.html | 80 ++++++++++++ template/oj/problem/my_submission.html | 15 ++- template/oj/problem/my_submissions_list.html | 10 +- template/oj/problem/problem.html | 2 +- .../oj/submission/my_submissions_list.html | 4 +- template/oj_base.html | 2 +- 24 files changed, 554 insertions(+), 190 deletions(-) create mode 100644 contest_submission/__init__.py create mode 100644 contest_submission/admin.py create mode 100644 contest_submission/migrations/__init__.py create mode 100644 contest_submission/models.py create mode 100644 contest_submission/serializers.py create mode 100644 contest_submission/tests.py create mode 100644 contest_submission/views.py create mode 100644 template/oj/contest/my_submission.html create mode 100644 template/oj/contest/submissions_list.html diff --git a/contest/views.py b/contest/views.py index 42ecd62..8134dca 100644 --- a/contest/views.py +++ b/contest/views.py @@ -255,6 +255,9 @@ class ContestPasswordVerifyAPIView(APIView): @check_user_contest_permission def contest_page(request, contest_id): + """ + 单个比赛的详情页 + """ try: contest = Contest.objects.get(id=contest_id) except Contest.DoesNotExist: @@ -265,6 +268,9 @@ def contest_page(request, contest_id): @check_user_contest_permission def contest_problem_page(request, contest_id, contest_problem_id): + """ + 单个比赛题目的详情页 + """ try: contest = Contest.objects.get(id=contest_id) except Contest.DoesNotExist: @@ -286,6 +292,9 @@ def contest_problem_page(request, contest_id, contest_problem_id): @check_user_contest_permission def contest_problems_list_page(request, contest_id): + """ + 比赛所有题目的列表页 + """ try: contest_problems = ContestProblem.objects.filter(contest=Contest.objects.get(id=contest_id)).order_by("sort_index") except Contest.DoesNotExist: @@ -298,6 +307,9 @@ def contest_problems_list_page(request, contest_id): def contest_list_page(request, page=1): + """ + 所有比赛的列表页 + """ # 正常情况 contests = Contest.objects.filter(visible=True) diff --git a/contest_submission/__init__.py b/contest_submission/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/contest_submission/admin.py b/contest_submission/admin.py new file mode 100644 index 0000000..8c38f3f --- /dev/null +++ b/contest_submission/admin.py @@ -0,0 +1,3 @@ +from django.contrib import admin + +# Register your models here. diff --git a/contest_submission/migrations/__init__.py b/contest_submission/migrations/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/contest_submission/models.py b/contest_submission/models.py new file mode 100644 index 0000000..71a8362 --- /dev/null +++ b/contest_submission/models.py @@ -0,0 +1,3 @@ +from django.db import models + +# Create your models here. diff --git a/contest_submission/serializers.py b/contest_submission/serializers.py new file mode 100644 index 0000000..eb9a860 --- /dev/null +++ b/contest_submission/serializers.py @@ -0,0 +1,11 @@ +# coding=utf-8 +from rest_framework import serializers + +from account.models import User + + +class CreateContestSubmissionSerializer(serializers.Serializer): + contest_id = serializers.IntegerField() + problem_id = serializers.IntegerField() + language = serializers.IntegerField() + code = serializers.CharField(max_length=3000) diff --git a/contest_submission/tests.py b/contest_submission/tests.py new file mode 100644 index 0000000..7ce503c --- /dev/null +++ b/contest_submission/tests.py @@ -0,0 +1,3 @@ +from django.test import TestCase + +# Create your tests here. diff --git a/contest_submission/views.py b/contest_submission/views.py new file mode 100644 index 0000000..192e331 --- /dev/null +++ b/contest_submission/views.py @@ -0,0 +1,117 @@ +# coding=utf-8 +import json + +import redis +from django.shortcuts import render +from django.core.paginator import Paginator + +from rest_framework.views import APIView + +from judge.judger_controller.tasks import judge +from judge.judger_controller.settings import redis_config +from account.decorators import login_required +from account.models import SUPER_ADMIN + +from contest.decorators import check_user_contest_permission + +from problem.models import Problem +from contest.models import Contest, ContestProblem + +from utils.shortcuts import serializer_invalid_response, error_response, success_response, error_page, paginate + +from submission.models import Submission +from .serializers import CreateContestSubmissionSerializer + + +class ContestSubmissionAPIView(APIView): + @check_user_contest_permission + def post(self, request): + """ + 创建比赛的提交 + --- + request_serializer: CreateContestSubmissionSerializer + """ + serializer = CreateContestSubmissionSerializer(data=request.data) + if serializer.is_valid(): + data = serializer.data + try: + contest = Contest.objects.get(id=data["contest_id"]) + except Contest.DoesNotExist: + return error_response(u"比赛不存在") + try: + problem = ContestProblem.objects.get(contest=contest, id=data["problem_id"]) + # 更新题目提交计数器 + problem.total_submit_number += 1 + problem.save() + except Problem.DoesNotExist: + return error_response(u"题目不存在") + + submission = Submission.objects.create(user_id=request.user.id, language=int(data["language"]), + contest_id=contest.id, code=data["code"], problem_id=problem.id) + try: + judge.delay(submission.id, problem.time_limit, problem.memory_limit, problem.test_case_id) + except Exception: + return error_response(u"提交判题任务失败") + + # 增加redis 中判题队列长度的计数器 + r = redis.Redis(host=redis_config["host"], port=redis_config["port"], db=redis_config["db"]) + r.incr("judge_queue_length") + + return success_response({"submission_id": submission.id}) + + else: + return serializer_invalid_response(serializer) + + +@login_required +def contest_problem_my_submissions_list_page(request, contest_id, contest_problem_id): + """ + 我比赛单个题目的所有提交列表 + """ + try: + Contest.objects.get(id=contest_id) + except Contest.DoesNotExist: + return error_page(request, u"比赛不存在") + try: + contest_problem = ContestProblem.objects.get(id=contest_problem_id, visible=True) + except Problem.DoesNotExist: + return error_page(request, u"比赛问题不存在") + submissions = Submission.objects.filter(user_id=request.user.id, problem_id=contest_problem.id).order_by("-create_time"). \ + values("id", "result", "create_time", "accepted_answer_time", "language") + return render(request, "oj/contest/my_submissions_list.html", + {"submissions": submissions, "problem": contest_problem}) + + +@login_required +def contest_problem_submissions_list_page(request, contest_id, page=1): + """ + 单个比赛中的所有提交(包含自己和别人,自己可查提交结果,其他人不可查) + """ + try: + contest = Contest.objects.get(id=contest_id) + except Contest.DoesNotExist: + return error_page(request, u"比赛不存在") + # 以下是本场比赛中所有的提交 + submissions = Submission.objects.filter(contest_id=contest_id). \ + values("id", "result", "create_time", "accepted_answer_time", "language", "user_id").order_by("-create_time") + paginator = Paginator(submissions, 20) + try: + current_page = paginator.page(int(page)) + except Exception: + return error_page(request, u"不存在的页码") + previous_page = next_page = None + try: + previous_page = current_page.previous_page_number() + except Exception: + pass + try: + next_page = current_page.next_page_number() + except Exception: + pass + + print current_page + + return render(request, "oj/contest/submissions_list.html", + {"submissions": current_page, "page": int(page), + "previous_page": previous_page, "next_page": next_page, "start_id": int(page) * 20 - 20, + "contest": contest}) \ No newline at end of file diff --git a/oj/settings.py b/oj/settings.py index 5703566..c3883ae 100644 --- a/oj/settings.py +++ b/oj/settings.py @@ -54,6 +54,7 @@ INSTALLED_APPS = ( 'submission', 'mq', 'contest', + 'contest_submission', 'django_extensions', 'rest_framework', diff --git a/oj/urls.py b/oj/urls.py index c1b3a57..465e1a6 100644 --- a/oj/urls.py +++ b/oj/urls.py @@ -16,48 +16,44 @@ from group.views import (GroupAdminAPIView, GroupMemberAdminAPIView, from admin.views import AdminTemplateView from problem.views import TestCaseUploadAPIView, ProblemTagAdminAPIView, ProblemAdminAPIView -from submission.views import SubmissionAPIView, SubmissionAdminAPIView, ContestSubmissionAPIView +from submission.views import SubmissionAPIView, SubmissionAdminAPIView +from contest_submission.views import ContestSubmissionAPIView from monitor.views import QueueLengthMonitorAPIView +from contest_submission.views import contest_problem_my_submissions_list_page + urlpatterns = [ url(r'^install/$', "install.views.install"), url("^$", TemplateView.as_view(template_name="oj/index.html"), name="index_page"), url(r'^docs/', include('rest_framework_swagger.urls')), url(r'^admin/$', TemplateView.as_view(template_name="admin/admin.html"), name="admin_spa_page"), + url(r'^admin/contest/$', TemplateView.as_view(template_name="admin/contest/add_contest.html"), + name="add_contest_page"), + url(r'^admin/template/(?P\w+)/(?P\w+).html$', AdminTemplateView.as_view(), + name="admin_template"), + url(r'^login/$', TemplateView.as_view(template_name="oj/account/login.html"), name="user_login_page"), url(r'^register/$', TemplateView.as_view(template_name="oj/account/register.html"), name="user_register_page"), url(r'^change_password/$', TemplateView.as_view(template_name="oj/account/change_password.html"), name="user_change_password_page"), + url(r'^announcement/(?P\d+)/$', "announcement.views.announcement_page", + name="announcement_page"), + url(r'^api/user/$', UserInfoAPIView.as_view(), name="user_info_api"), url(r'^api/login/$', UserLoginAPIView.as_view(), name="user_login_api"), url(r'^api/register/$', UserRegisterAPIView.as_view(), name="user_register_api"), url(r'^api/change_password/$', UserChangePasswordAPIView.as_view(), name="user_change_password_api"), url(r'^api/username_check/$', UsernameCheckAPIView.as_view(), name="username_check_api"), url(r'^api/email_check/$', EmailCheckAPIView.as_view(), name="email_check_api"), + url(r'^api/contest/password/$', ContestPasswordVerifyAPIView.as_view(), name="contest_password_verify_api"), + url(r'^api/contest/submission/$', ContestSubmissionAPIView.as_view(), name="contest_submission_api"), + url(r'^api/submission/$', SubmissionAPIView.as_view(), name="submission_api"), + url(r'^api/admin/announcement/$', AnnouncementAdminAPIView.as_view(), name="announcement_admin_api"), url(r'^api/admin/contest/$', ContestAdminAPIView.as_view(), name="contest_admin_api"), url(r'^api/admin/user/$', UserAdminAPIView.as_view(), name="user_admin_api"), - url(r'^problem/(?P\d+)/$', "problem.views.problem_page", name="problem_page"), - url(r'^problem/(?P\d+)/$', "problem.views.problem_page", name="problem_page"), - - url(r'^contest/(?P\d+)/problem/(?P\d+)/$', "contest.views.contest_problem_page", - name="contest_problem_page"), - - url(r'^contest/(?P\d+)/$', "contest.views.contest_page", name="contest_page"), - url(r'^contest/(?P\d+)/problems/$', "contest.views.contest_problems_list_page", - name="contest_problems_list_page"), - url(r'^announcement/(?P\d+)/$', "announcement.views.announcement_page", - name="announcement_page"), - url(r'^admin/contest/$', TemplateView.as_view(template_name="admin/contest/add_contest.html"), - name="add_contest_page"), - url(r'^problems/$', "problem.views.problem_list_page", name="problem_list_page"), - url(r'^problems/(?P\d+)/$', "problem.views.problem_list_page", name="problem_list_page"), - url(r'^contests/$', "contest.views.contest_list_page", name="contest_list_page"), - url(r'^contests/(?P\d+)/$', "contest.views.contest_list_page", name="contest_list_page"), - url(r'^admin/template/(?P\w+)/(?P\w+).html$', AdminTemplateView.as_view(), - name="admin_template"), url(r'^api/admin/group/$', GroupAdminAPIView.as_view(), name="group_admin_api"), url(r'^api/admin/group_member/$', GroupMemberAdminAPIView.as_view(), name="group_member_admin_api"), url(r'^api/admin/group_join/$', JoinGroupAPIView.as_view(), name="group_join_admin_api"), @@ -65,23 +61,42 @@ urlpatterns = [ url(r'^api/admin/contest_problem/$', ContestProblemAdminAPIView.as_view(), name="contest_problem_admin_api"), url(r'^api/admin/test_case_upload/$', TestCaseUploadAPIView.as_view(), name="test_case_upload_api"), url(r'^api/admin/tag/$', ProblemTagAdminAPIView.as_view(), name="problem_tag_admin_api"), - url(r'^problem/(?P\d+)/my_submissions/$', "submission.views.problem_my_submissions_list_page", - name="problem_my_submissions_page"), - - url(r'^contest/(?P\d+)/problem/(?P\d+)/my_submissions/$', - "submission.views.contest_problem_my_submissions_list_page", name="contest_problem_my_submissions_list_page"), - - url(r'^my_submission/(?P\w+)/$', "submission.views.my_submission", name="my_submission_page"), - url(r'^api/admin/join_group_request/$', JoinGroupRequestAdminAPIView.as_view(), name="join_group_request_admin_api"), - url(r'^api/submission/$', SubmissionAPIView.as_view(), name="submission_api"), url(r'^api/admin/submission/$', SubmissionAdminAPIView.as_view(), name="submission_admin_api_view"), - url(r'^my_submissions/$', "submission.views.my_submission_list_page", name="my_submission_list_page"), - url(r'^my_submissions/(?P\d+)/$', "submission.views.my_submission_list_page", name="my_submission_list_page"), url(r'^api/admin/monitor/$', QueueLengthMonitorAPIView.as_view(), name="queue_length_monitor_api"), + + + + url(r'^contest/(?P\d+)/problem/(?P\d+)/$', "contest.views.contest_problem_page", + name="contest_problem_page"), + url(r'^contest/(?P\d+)/problem/(?P\d+)/submissions/$', + "contest_submission.views.contest_problem_my_submissions_list_page", + name="contest_problem_my_submissions_list_page"), + url(r'^contest/(?P\d+)/$', "contest.views.contest_page", name="contest_page"), - url(r'^api/contest/password/$', ContestPasswordVerifyAPIView.as_view(), name="contest_password_verify_api"), - url(r'^api/contest/submission/$', ContestSubmissionAPIView.as_view(), name="contest_submission_api"), + url(r'^contest/(?P\d+)/problems/$', "contest.views.contest_problems_list_page", + name="contest_problems_list_page"), + url(r'^contest/(?P\d+)/submissions/$', "contest_submission.views.contest_problem_submissions_list_page", + name="contest_problem_submissions_list_page"), + url(r'^contest/(?P\d+)/submissions/(?P\d+)/$', + "contest_submission.views.contest_problem_submissions_list_page", name="contest_problem_submissions_list_page"), + + url(r'^contests/$', "contest.views.contest_list_page", name="contest_list_page"), + url(r'^contests/(?P\d+)/$', "contest.views.contest_list_page", name="contest_list_page"), + url(r'^contest/(?P\d+)/$', "contest.views.contest_page", name="contest_page"), + + + url(r'^problem/(?P\d+)/$', "problem.views.problem_page", name="problem_page"), + url(r'^problem/(?P\d+)/$', "problem.views.problem_page", name="problem_page"), + url(r'^problems/$', "problem.views.problem_list_page", name="problem_list_page"), + url(r'^problems/(?P\d+)/$', "problem.views.problem_list_page", name="problem_list_page"), + url(r'^problem/(?P\d+)/submissions/$', "submission.views.problem_my_submissions_list_page", + name="problem_my_submissions_page"), + + + url(r'^submission/(?P\w+)/$', "submission.views.my_submission", name="my_submission_page"), + url(r'^submissions/$', "submission.views.my_submission_list_page", name="my_submission_list_page"), + url(r'^submissions/(?P\d+)/$', "submission.views.my_submission_list_page", name="my_submission_list_page"), ] diff --git a/static/src/css/oj.css b/static/src/css/oj.css index ce1bc0e..e099835 100644 --- a/static/src/css/oj.css +++ b/static/src/css/oj.css @@ -90,5 +90,5 @@ li.list-group-item { } .contest-tab{ - margin-bottom: 5px; + margin-bottom: 15px; } \ No newline at end of file diff --git a/submission/serializers.py b/submission/serializers.py index 545c13a..8170486 100644 --- a/submission/serializers.py +++ b/submission/serializers.py @@ -22,8 +22,3 @@ class SubmissionSerializer(serializers.ModelSerializer): return User.objects.get(id=obj.user_id).username -class CreateContestSubmissionSerializer(serializers.Serializer): - contest_id = serializers.IntegerField() - problem_id = serializers.IntegerField() - language = serializers.IntegerField() - code = serializers.CharField(max_length=3000) \ No newline at end of file diff --git a/submission/tests.py b/submission/tests.py index c31741b..a9b6de1 100644 --- a/submission/tests.py +++ b/submission/tests.py @@ -1,5 +1,10 @@ +# coding=utf-8 +import json from django.test import TestCase -from account.models import User, REGULAR_USER +from django.core.urlresolvers import reverse +from account.models import User, REGULAR_USER, ADMIN, SUPER_ADMIN +from problem.models import Problem +from contest.models import Contest from submission.models import Submission from rest_framework.test import APITestCase, APIClient @@ -12,6 +17,7 @@ class SubmissionsListPageTest(TestCase): self.user2.set_password("666666") self.user.set_password("666666") self.user.save() + self.user2.save() # self.client.login(username="gogoing", password="666666") self.submission = Submission.objects.create(user_id=self.user.id, language=1, @@ -42,3 +48,99 @@ class SubmissionsListPageTest(TestCase): self.client.login(username="cool", password="666666") response = self.client.get('/my_submissions/') self.assertEqual(response.status_code, 200) + + +class SubmissionAPITest(APITestCase): + def setUp(self): + self.client = APIClient() + self.url = reverse('submission_api') + self.user1 = User.objects.create(username="test1", admin_type=REGULAR_USER) + self.user1.set_password("testaa") + self.user1.save() + self.user2 = User.objects.create(username="test2", admin_type=SUPER_ADMIN) + self.user2.set_password("testbb") + self.user2.save() + self.problem = Problem.objects.create(title="title1", + description="description1", + input_description="input1_description", + output_description="output1_description", + test_case_id="1", + source="source1", + samples=json.dumps([{"input": "1 1", "output": "2"}]), + time_limit=100, + memory_limit=1000, + difficulty=1, + hint="hint1", + created_by=User.objects.get(username="test2")) + self.global_contest = Contest.objects.create(title="titlex", description="descriptionx", mode=1, + contest_type=2, show_rank=True, show_user_submission=True, + start_time="2015-08-15T10:00:00.000Z", + end_time="2015-08-15T12:00:00.000Z", + password="aacc", created_by=User.objects.get(username="test2")) + + self.submission = Submission.objects.create(user_id=self.user1.id, + language=1, + code='#include "stdio.h"\nint main(){\n\treturn 0;\n}', + problem_id=self.problem.id) + + def test_invalid_format(self): + self.client.login(username="test1", password="testaa") + data = {"language": 1} + response = self.client.post(self.url, data=data) + self.assertEqual(response.data["code"], 1) + + def test_problem_does_not_exist(self): + self.client.login(username="test1", password="testaa") + data = {"language": 1, "code": '#include "stdio.h"\nint main(){\n\treturn 0;\n}', + "problem_id": self.problem.id + 10} + response = self.client.post(self.url, data=data) + self.assertEqual(response.data, {"code": 1, "data": u"题目不存在"}) + + def test_submission_successfully(self): + self.client.login(username="test1", password="testaa") + data = {"language": 1, "code": '#include "stdio.h"\nint main(){\n\treturn 0;\n}', + "problem_id": self.problem.id} + response = self.client.post(self.url, data=data) + self.assertEqual(response.data["code"], 0) + + def test_submission_does_not_exist(self): + self.client.login(username="test1", password="testaa") + data = {"submission_id": self.submission.id + "111"} + response = self.client.get(self.url, data=data) + self.assertEqual(response.data, {"code": 1, "data": u"提交不存在"}) + + def test_get_submission_successfully(self): + self.client.login(username="test1", password="testaa") + data = {"submission_id": self.submission.id} + response = self.client.get(self.url, data=data) + self.assertEqual(response.data["code"], 0) + + +class ContestSubmissionAPITest(APITestCase): + def setUp(self): + self.client = APIClient() + self.url = reverse('contest_submission_api') + self.user1 = User.objects.create(username="test1", admin_type=REGULAR_USER) + self.user1.set_password("testaa") + self.user1.save() + self.user2 = User.objects.create(username="test2", admin_type=SUPER_ADMIN) + self.user2.set_password("testbb") + self.user2.save() + self.global_contest = Contest.objects.create(title="titlex", description="descriptionx", mode=1, + contest_type=2, show_rank=True, show_user_submission=True, + start_time="2015-08-15T10:00:00.000Z", + end_time="2015-08-15T12:00:00.000Z", + password="aacc", created_by=User.objects.get(username="test2")) + + def test_invalid_format(self): + self.client.login(username="test1", password="testaa") + data = {"language": 1} + response = self.client.post(self.url, data=data) + pass + + + + + + + diff --git a/submission/views.py b/submission/views.py index 6399e22..918ad58 100644 --- a/submission/views.py +++ b/submission/views.py @@ -19,7 +19,7 @@ from contest.models import Contest, ContestProblem from utils.shortcuts import serializer_invalid_response, error_response, success_response, error_page, paginate from .models import Submission -from .serializers import CreateSubmissionSerializer, SubmissionSerializer, CreateContestSubmissionSerializer +from .serializers import CreateSubmissionSerializer, SubmissionSerializer class SubmissionAPIView(APIView): @@ -73,34 +73,24 @@ class SubmissionAPIView(APIView): @login_required def problem_my_submissions_list_page(request, problem_id): + """ + 我单个题目所有提交的列表页 + """ try: problem = Problem.objects.get(id=problem_id, visible=True) except Problem.DoesNotExist: return error_page(request, u"问题不存在") - submissions = Submission.objects.filter(user_id=request.user.id, problem_id=problem.id).order_by("-create_time"). \ + submissions = Submission.objects.filter(user_id=request.user.id, problem_id=problem.id, contest_id__isnull=True).order_by("-create_time"). \ values("id", "result", "create_time", "accepted_answer_time", "language") return render(request, "oj/problem/my_submissions_list.html", {"submissions": submissions, "problem": problem}) -@login_required -def contest_problem_my_submissions_list_page(request, contest_id, contest_problem_id): - try: - Contest.objects.get(id=contest_id) - except Contest.DoesNotExist: - return error_page(request, u"比赛不存在") - try: - contest_problem = ContestProblem.objects.get(id=contest_problem_id, visible=True) - except Problem.DoesNotExist: - return error_page(request, u"比赛问题不存在") - submissions = Submission.objects.filter(user_id=request.user.id, problem_id=contest_problem.id).order_by("-create_time"). \ - values("id", "result", "create_time", "accepted_answer_time", "language") - return render(request, "oj/contest/my_submissions_list.html", - {"submissions": submissions, "contest_problem": contest_problem}) - - @login_required def my_submission(request, submission_id): + """ + 单个题目的提交详情页 + """ try: # 超级管理员可以查看所有的提交 if request.user.admin_type != SUPER_ADMIN: @@ -110,10 +100,18 @@ def my_submission(request, submission_id): except Submission.DoesNotExist: return error_page(request, u"提交不存在") - try: - problem = Problem.objects.get(id=submission.problem_id, visible=True) - except Problem.DoesNotExist: - return error_page(request, u"提交不存在") + if submission.contest_id: + try: + problem = ContestProblem.objects.get(id=submission.problem_id, + visible=True) + except ContestProblem.DoesNotExist: + return error_page(request, u"提交不存在") + else: + try: + problem = Problem.objects.get(id=submission.problem_id, visible=True) + except Problem.DoesNotExist: + return error_page(request, u"提交不存在") + if submission.info: try: info = json.loads(submission.info) @@ -136,6 +134,9 @@ class SubmissionAdminAPIView(APIView): @login_required def my_submission_list_page(request, page=1): + """ + 我的所有提交的列表页 + """ submissions = Submission.objects.filter(user_id=request.user.id). \ values("id", "result", "create_time", "accepted_answer_time", "language").order_by("-create_time") paginator = Paginator(submissions, 20) @@ -158,41 +159,4 @@ def my_submission_list_page(request, page=1): "previous_page": previous_page, "next_page": next_page, "start_id": int(page) * 20 - 20}) -class ContestSubmissionAPIView(APIView): - @check_user_contest_permission - def post(self, request): - """ - 创建比赛的提交 - --- - request_serializer: ConestSubmissionSerializer - """ - serializer = CreateContestSubmissionSerializer(data=request.data) - if serializer.is_valid(): - data = serializer.data - try: - contest = Contest.objects.get(id=data["contest_id"]) - except Contest.DoesNotExist: - return error_response(u"比赛不存在") - try: - problem = ContestProblem.objects.get(contest=contest, id=data["problem_id"]) - # 更新题目提交计数器 - problem.total_submit_number += 1 - problem.save() - except Problem.DoesNotExist: - return error_response(u"题目不存在") - submission = Submission.objects.create(user_id=request.user.id, language=int(data["language"]), - contest_id=contest.id, code=data["code"], problem_id=problem.id) - try: - judge.delay(submission.id, problem.time_limit, problem.memory_limit, problem.test_case_id) - except Exception: - return error_response(u"提交判题任务失败") - - # 增加redis 中判题队列长度的计数器 - r = redis.Redis(host=redis_config["host"], port=redis_config["port"], db=redis_config["db"]) - r.incr("judge_queue_length") - - return success_response({"submission_id": submission.id}) - - else: - return serializer_invalid_response(serializer) diff --git a/template/oj/contest/contest_problem.html b/template/oj/contest/contest_problem.html index 32b0617..cbea994 100644 --- a/template/oj/contest/contest_problem.html +++ b/template/oj/contest/contest_problem.html @@ -6,7 +6,7 @@

{{ contest_problem.title }}

diff --git a/template/oj/contest/contest_problems_list.html b/template/oj/contest/contest_problems_list.html index 23cefd7..4dd68c5 100644 --- a/template/oj/contest/contest_problems_list.html +++ b/template/oj/contest/contest_problems_list.html @@ -1,82 +1,66 @@ {% extends "oj_base.html" %} {% block body %} - {% load problem %} -
+{% load problem %} +
-
-
+
+
- -
- -
- -
- - - - - - - - - - - {% for item in contest_problems %} - - - - - - - {% endfor %} - -
#题目通过率
- - - {{ item.sort_index }} - - {{ item.title }} - {{ item|accepted_radio }}
- -
-
-
- {% include "oj/announcement/_announcement_panel.html" %} -
+
+ + +
+
+ + + + + + + + + + + {% for item in contest_problems %} + + + + + + + {% endfor %} + +
#题目通过率
{{ item.sort_index }} + {{ item.title }} + {{ item|accepted_radio }}
+ +
+
+
+ {% include "oj/announcement/_announcement_panel.html" %} +
+
{% endblock %} {% block js_block %} - + {% endblock %} \ No newline at end of file diff --git a/template/oj/contest/my_submission.html b/template/oj/contest/my_submission.html new file mode 100644 index 0000000..ed26b9e --- /dev/null +++ b/template/oj/contest/my_submission.html @@ -0,0 +1,71 @@ +{% extends 'oj_base.html' %} +{% block css_block %} + +{% endblock %} +{% block body %} + {% load submission %} +
+ + {% include "oj/problem/_problem_header.html" %} +
+
+

运行结果 : + {{ submission.result|translate_result }} + +

+ {% ifequal submission.result 0 %} +

时间 : {{ submission.accepted_answer_time }}ms 语言 : + {{ submission.language|translate_language }} +

+ {% endifequal %} + {% ifequal submission.result 4 %} +

{{ submission.info }}

+ {% endifequal %} +

提交时间 : {{ submission.create_time }}

+
+
+ {% ifequal request.user.admin_type 2 %} +

本调试信息仅超级管理员可见

+ {% ifequal submission.result 7 %} +
System Error: {{ submission.info }}
+ {% else %} +
{{ info }}
+ {% endifequal %} + + {% endifequal %} + + +
+ +
+
+ +{% endblock %} +{% block js_block %} + +{% endblock %} \ No newline at end of file diff --git a/template/oj/contest/my_submissions_list.html b/template/oj/contest/my_submissions_list.html index dc4f5f5..fb24bef 100644 --- a/template/oj/contest/my_submissions_list.html +++ b/template/oj/contest/my_submissions_list.html @@ -6,17 +6,13 @@
-

{{ contest_problem.title }}

- -

发布时间: {{ contest_problem.create_time }}   - 时间限制: {{ contest_problem.time_limit }}ms   - 内存限制: {{ contest_problem.memory_limit }}M

+ {% include "oj/problem/_problem_header.html" %} @@ -30,7 +26,7 @@ {% for item in submissions %} - + - - - {% endfor %} - - -
{{ forloop.counter }}{{ forloop.counter }} {{ item.create_time }} {{ item.result|translate_result }} diff --git a/template/oj/contest/submissions_list.html b/template/oj/contest/submissions_list.html new file mode 100644 index 0000000..6635219 --- /dev/null +++ b/template/oj/contest/submissions_list.html @@ -0,0 +1,80 @@ +{% extends 'oj_base.html' %} + +{% block body %} + +{% load submission %} +
+
+ +
+ {% if submissions %} + + + + + + + + + + + + {% for item in submissions %} + + + {% ifequal item.user_id request.user.id %} + + {% else %} + + {% endifequal %} + + + + + + + {% endfor %} + + +
#提交时间结果运行时间语言
+ {{ forloop.counter |add:start_id }}{{ forloop.counter |add:start_id }}{{ item.create_time }}{{ item.result|translate_result }} + {% if item.accepted_answer_time %} + {{ item.accepted_answer_time }}ms + {% else %} + -- + {% endif %} + + {{ item.language|translate_language }} +
+ {% else %} +

你还没有提交记录!

+ {% endif %} + +
+{% endblock %} diff --git a/template/oj/problem/my_submission.html b/template/oj/problem/my_submission.html index 1aca690..2e361f5 100644 --- a/template/oj/problem/my_submission.html +++ b/template/oj/problem/my_submission.html @@ -11,8 +11,19 @@
{% include "oj/problem/_problem_header.html" %}
diff --git a/template/oj/problem/my_submissions_list.html b/template/oj/problem/my_submissions_list.html index 8a9a235..fb07a4c 100644 --- a/template/oj/problem/my_submissions_list.html +++ b/template/oj/problem/my_submissions_list.html @@ -8,13 +8,9 @@
  • 题目
  • + 我的提交 -

    {{ problem.title }}

    - -

    发布时间: {{ problem.create_time }}   - 时间限制: {{ problem.time_limit }}ms   - 内存限制: {{ problem.memory_limit }}M

    + {% include "oj/problem/_problem_header.html" %} @@ -28,7 +24,7 @@ {% for item in submissions %} - + - - diff --git a/template/oj/contest/submissions_list.html b/template/oj/contest/submissions_list.html index 6635219..336e9c3 100644 --- a/template/oj/contest/submissions_list.html +++ b/template/oj/contest/submissions_list.html @@ -2,79 +2,83 @@ {% block body %} -{% load submission %} -
    -
    - -
    - {% if submissions %} -
    {{ forloop.counter }}{{ forloop.counter }} {{ item.create_time }} {{ item.result|translate_result }} diff --git a/template/oj/problem/problem.html b/template/oj/problem/problem.html index 5a7d847..2efd86a 100644 --- a/template/oj/problem/problem.html +++ b/template/oj/problem/problem.html @@ -5,7 +5,7 @@ {% include "oj/problem/_problem_header.html" %} diff --git a/template/oj/submission/my_submissions_list.html b/template/oj/submission/my_submissions_list.html index 4b939ec..279bee6 100644 --- a/template/oj/submission/my_submissions_list.html +++ b/template/oj/submission/my_submissions_list.html @@ -44,12 +44,12 @@ diff --git a/template/oj_base.html b/template/oj_base.html index 61b887a..082cad5 100644 --- a/template/oj_base.html +++ b/template/oj_base.html @@ -43,7 +43,7 @@
    {{ item.sort_index }} + + {{ item.sort_index }} {{ item.title }} + + {{ item.title }} {{ item|accepted_radio }}
    - - - - - - - - - - - {% for item in submissions %} - + {% load submission %} + {% load user %} +
    +
    + +
    + {% if submissions %} +
    #提交时间结果运行时间语言
    + + + + + + + + + + + + {% for item in submissions %} + - {% ifequal item.user_id request.user.id %} - - {% else %} - - {% endifequal %} + {% ifequal item.user_id request.user.id %} + + {% else %} + + {% endifequal %} - - - + + + + + + + {% endfor %} + + +
    #用户提交时间结果运行时间语言
    - {{ forloop.counter |add:start_id }}{{ forloop.counter |add:start_id }} + {{ forloop.counter |add:start_id }}{{ forloop.counter |add:start_id }}{{ item.create_time }}{{ item.result|translate_result }} - {% if item.accepted_answer_time %} - {{ item.accepted_answer_time }}ms - {% else %} - -- + {{ item.user_id|get_username }}{{ item.create_time }}{{ item.result|translate_result }} + {% if item.accepted_answer_time %} + {{ item.accepted_answer_time }}ms + {% else %} + -- + {% endif %} + + {{ item.language|translate_language }} +
    + {% else %} +

    你还没有提交记录!

    + {% endif %} +
    - {{ item.language|translate_language }} -
    - {% else %} -

    你还没有提交记录!

    - {% endif %} - -
    + {% if next_page %} + + {% endif %} + + +
    {% endblock %} diff --git a/utils/templatetags/user.py b/utils/templatetags/user.py new file mode 100644 index 0000000..4a81de5 --- /dev/null +++ b/utils/templatetags/user.py @@ -0,0 +1,17 @@ +# coding=utf-8 +import datetime +from account.models import User + + +def get_username(user_id): + try: + return User.objects.get(id=user_id).username + except User.DoesNotExist: + return "" + + +from django import template + +register = template.Library() +register.filter("get_username", get_username) +