2020-01-21 06:35:58 +00:00
|
|
|
import logging
|
|
|
|
import os
|
|
|
|
import shutil
|
|
|
|
from datetime import timedelta
|
|
|
|
from operator import itemgetter
|
|
|
|
from random import randrange
|
2020-07-26 05:11:03 +00:00
|
|
|
import zipfile, tempfile
|
2020-01-21 06:35:58 +00:00
|
|
|
|
|
|
|
from django.conf import settings
|
|
|
|
from django.contrib.auth.decorators import login_required
|
|
|
|
from django.contrib.auth.mixins import PermissionRequiredMixin
|
|
|
|
from django.core.exceptions import ObjectDoesNotExist, PermissionDenied
|
|
|
|
from django.db import transaction
|
2020-07-19 21:48:39 +00:00
|
|
|
from django.db.models import Count, F, Prefetch, Q, Sum, Case, When, IntegerField
|
2020-01-21 06:35:58 +00:00
|
|
|
from django.db.utils import ProgrammingError
|
2020-07-19 21:39:28 +00:00
|
|
|
from django.http import Http404, HttpResponse, HttpResponseForbidden, HttpResponseRedirect
|
2020-01-21 06:35:58 +00:00
|
|
|
from django.shortcuts import get_object_or_404, render
|
|
|
|
from django.template.loader import get_template
|
|
|
|
from django.urls import reverse
|
|
|
|
from django.utils import timezone, translation
|
|
|
|
from django.utils.functional import cached_property
|
|
|
|
from django.utils.html import escape, format_html
|
|
|
|
from django.utils.safestring import mark_safe
|
|
|
|
from django.utils.translation import gettext as _, gettext_lazy
|
|
|
|
from django.views.generic import ListView, View
|
|
|
|
from django.views.generic.base import TemplateResponseMixin
|
|
|
|
from django.views.generic.detail import SingleObjectMixin
|
|
|
|
|
|
|
|
from judge.comments import CommentedDetailView
|
|
|
|
from judge.forms import ProblemCloneForm, ProblemSubmitForm
|
|
|
|
from judge.models import ContestProblem, ContestSubmission, Judge, Language, Problem, ProblemGroup, \
|
|
|
|
ProblemTranslation, ProblemType, RuntimeVersion, Solution, Submission, SubmissionSource, \
|
|
|
|
TranslatedProblemForeignKeyQuerySet
|
|
|
|
from judge.pdf_problems import DefaultPdfMaker, HAS_PDF
|
|
|
|
from judge.utils.diggpaginator import DiggPaginator
|
|
|
|
from judge.utils.opengraph import generate_opengraph
|
|
|
|
from judge.utils.problems import contest_attempted_ids, contest_completed_ids, hot_problems, user_attempted_ids, \
|
|
|
|
user_completed_ids
|
|
|
|
from judge.utils.strings import safe_float_or_none, safe_int_or_none
|
|
|
|
from judge.utils.tickets import own_ticket_filter
|
|
|
|
from judge.utils.views import QueryStringSortMixin, SingleObjectFormView, TitleMixin, generic_message
|
|
|
|
|
|
|
|
|
|
|
|
def get_contest_problem(problem, profile):
|
|
|
|
try:
|
|
|
|
return problem.contests.get(contest_id=profile.current_contest.contest_id)
|
|
|
|
except ObjectDoesNotExist:
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
|
|
|
def get_contest_submission_count(problem, profile, virtual):
|
|
|
|
return profile.current_contest.submissions.exclude(submission__status__in=['IE']) \
|
2020-07-19 21:39:28 +00:00
|
|
|
.filter(problem__problem=problem, participation__virtual=virtual).count()
|
2020-01-21 06:35:58 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ProblemMixin(object):
|
|
|
|
model = Problem
|
|
|
|
slug_url_kwarg = 'problem'
|
|
|
|
slug_field = 'code'
|
|
|
|
|
|
|
|
def get_object(self, queryset=None):
|
|
|
|
problem = super(ProblemMixin, self).get_object(queryset)
|
|
|
|
if not problem.is_accessible_by(self.request.user):
|
|
|
|
raise Http404()
|
|
|
|
return problem
|
|
|
|
|
|
|
|
def no_such_problem(self):
|
|
|
|
code = self.kwargs.get(self.slug_url_kwarg, None)
|
|
|
|
return generic_message(self.request, _('No such problem'),
|
|
|
|
_('Could not find a problem with the code "%s".') % code, status=404)
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
try:
|
|
|
|
return super(ProblemMixin, self).get(request, *args, **kwargs)
|
|
|
|
except Http404:
|
|
|
|
return self.no_such_problem()
|
|
|
|
|
|
|
|
|
|
|
|
class SolvedProblemMixin(object):
|
|
|
|
def get_completed_problems(self):
|
|
|
|
if self.in_contest:
|
|
|
|
return contest_completed_ids(self.profile.current_contest)
|
|
|
|
else:
|
|
|
|
return user_completed_ids(self.profile) if self.profile is not None else ()
|
|
|
|
|
|
|
|
def get_attempted_problems(self):
|
|
|
|
if self.in_contest:
|
|
|
|
return contest_attempted_ids(self.profile.current_contest)
|
|
|
|
else:
|
|
|
|
return user_attempted_ids(self.profile) if self.profile is not None else ()
|
|
|
|
|
|
|
|
@cached_property
|
|
|
|
def in_contest(self):
|
|
|
|
return self.profile is not None and self.profile.current_contest is not None
|
|
|
|
|
|
|
|
@cached_property
|
|
|
|
def contest(self):
|
|
|
|
return self.request.profile.current_contest.contest
|
|
|
|
|
|
|
|
@cached_property
|
|
|
|
def profile(self):
|
|
|
|
if not self.request.user.is_authenticated:
|
|
|
|
return None
|
|
|
|
return self.request.profile
|
|
|
|
|
|
|
|
|
|
|
|
class ProblemSolution(SolvedProblemMixin, ProblemMixin, TitleMixin, CommentedDetailView):
|
|
|
|
context_object_name = 'problem'
|
|
|
|
template_name = 'problem/editorial.html'
|
|
|
|
|
|
|
|
def get_title(self):
|
|
|
|
return _('Editorial for {0}').format(self.object.name)
|
|
|
|
|
|
|
|
def get_content_title(self):
|
|
|
|
return format_html(_(u'Editorial for <a href="{1}">{0}</a>'), self.object.name,
|
|
|
|
reverse('problem_detail', args=[self.object.code]))
|
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super(ProblemSolution, self).get_context_data(**kwargs)
|
|
|
|
|
|
|
|
solution = get_object_or_404(Solution, problem=self.object)
|
|
|
|
|
|
|
|
if (not solution.is_public or solution.publish_on > timezone.now()) and \
|
|
|
|
not self.request.user.has_perm('judge.see_private_solution') or \
|
|
|
|
(self.request.user.is_authenticated and
|
|
|
|
self.request.profile.current_contest):
|
|
|
|
raise Http404()
|
|
|
|
context['solution'] = solution
|
|
|
|
context['has_solved_problem'] = self.object.id in self.get_completed_problems()
|
|
|
|
return context
|
|
|
|
|
|
|
|
def get_comment_page(self):
|
|
|
|
return 's:' + self.object.code
|
|
|
|
|
|
|
|
|
|
|
|
class ProblemRaw(ProblemMixin, TitleMixin, TemplateResponseMixin, SingleObjectMixin, View):
|
|
|
|
context_object_name = 'problem'
|
|
|
|
template_name = 'problem/raw.html'
|
|
|
|
|
|
|
|
def get_title(self):
|
|
|
|
return self.object.name
|
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super(ProblemRaw, self).get_context_data(**kwargs)
|
|
|
|
context['problem_name'] = self.object.name
|
|
|
|
context['url'] = self.request.build_absolute_uri()
|
|
|
|
context['description'] = self.object.description
|
|
|
|
return context
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
self.object = self.get_object()
|
|
|
|
with translation.override(settings.LANGUAGE_CODE):
|
|
|
|
return self.render_to_response(self.get_context_data(
|
|
|
|
object=self.object,
|
|
|
|
))
|
|
|
|
|
|
|
|
|
|
|
|
class ProblemDetail(ProblemMixin, SolvedProblemMixin, CommentedDetailView):
|
|
|
|
context_object_name = 'problem'
|
|
|
|
template_name = 'problem/problem.html'
|
|
|
|
|
|
|
|
def get_comment_page(self):
|
|
|
|
return 'p:%s' % self.object.code
|
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super(ProblemDetail, self).get_context_data(**kwargs)
|
|
|
|
user = self.request.user
|
|
|
|
authed = user.is_authenticated
|
|
|
|
context['has_submissions'] = authed and Submission.objects.filter(user=user.profile,
|
|
|
|
problem=self.object).exists()
|
|
|
|
contest_problem = (None if not authed or user.profile.current_contest is None else
|
|
|
|
get_contest_problem(self.object, user.profile))
|
|
|
|
context['contest_problem'] = contest_problem
|
|
|
|
if contest_problem:
|
|
|
|
clarifications = self.object.clarifications
|
|
|
|
context['has_clarifications'] = clarifications.count() > 0
|
|
|
|
context['clarifications'] = clarifications.order_by('-date')
|
|
|
|
context['submission_limit'] = contest_problem.max_submissions
|
|
|
|
if contest_problem.max_submissions:
|
|
|
|
context['submissions_left'] = max(contest_problem.max_submissions -
|
2020-07-19 21:39:28 +00:00
|
|
|
get_contest_submission_count(self.object, user.profile,
|
2020-01-21 06:35:58 +00:00
|
|
|
user.profile.current_contest.virtual), 0)
|
|
|
|
|
|
|
|
context['available_judges'] = Judge.objects.filter(online=True, problems=self.object)
|
|
|
|
context['show_languages'] = self.object.allowed_languages.count() != Language.objects.count()
|
|
|
|
context['has_pdf_render'] = HAS_PDF
|
|
|
|
context['completed_problem_ids'] = self.get_completed_problems()
|
|
|
|
context['attempted_problems'] = self.get_attempted_problems()
|
|
|
|
|
|
|
|
can_edit = self.object.is_editable_by(user)
|
|
|
|
context['can_edit_problem'] = can_edit
|
|
|
|
if user.is_authenticated:
|
|
|
|
tickets = self.object.tickets
|
|
|
|
if not can_edit:
|
|
|
|
tickets = tickets.filter(own_ticket_filter(user.profile.id))
|
|
|
|
context['has_tickets'] = tickets.exists()
|
|
|
|
context['num_open_tickets'] = tickets.filter(is_open=True).values('id').distinct().count()
|
|
|
|
|
|
|
|
try:
|
|
|
|
context['editorial'] = Solution.objects.get(problem=self.object)
|
|
|
|
except ObjectDoesNotExist:
|
|
|
|
pass
|
|
|
|
try:
|
|
|
|
translation = self.object.translations.get(language=self.request.LANGUAGE_CODE)
|
|
|
|
except ProblemTranslation.DoesNotExist:
|
|
|
|
context['title'] = self.object.name
|
|
|
|
context['language'] = settings.LANGUAGE_CODE
|
|
|
|
context['description'] = self.object.description
|
|
|
|
context['translated'] = False
|
|
|
|
else:
|
|
|
|
context['title'] = translation.name
|
|
|
|
context['language'] = self.request.LANGUAGE_CODE
|
|
|
|
context['description'] = translation.description
|
|
|
|
context['translated'] = True
|
|
|
|
|
|
|
|
if not self.object.og_image or not self.object.summary:
|
|
|
|
metadata = generate_opengraph('generated-meta-problem:%s:%d' % (context['language'], self.object.id),
|
|
|
|
context['description'], 'problem')
|
|
|
|
context['meta_description'] = self.object.summary or metadata[0]
|
|
|
|
context['og_image'] = self.object.og_image or metadata[1]
|
|
|
|
return context
|
|
|
|
|
|
|
|
|
|
|
|
class LatexError(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class ProblemPdfView(ProblemMixin, SingleObjectMixin, View):
|
|
|
|
logger = logging.getLogger('judge.problem.pdf')
|
|
|
|
languages = set(map(itemgetter(0), settings.LANGUAGES))
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
if not HAS_PDF:
|
|
|
|
raise Http404()
|
|
|
|
|
|
|
|
language = kwargs.get('language', self.request.LANGUAGE_CODE)
|
|
|
|
if language not in self.languages:
|
|
|
|
raise Http404()
|
|
|
|
|
|
|
|
problem = self.get_object()
|
|
|
|
try:
|
|
|
|
trans = problem.translations.get(language=language)
|
|
|
|
except ProblemTranslation.DoesNotExist:
|
|
|
|
trans = None
|
|
|
|
|
|
|
|
cache = os.path.join(settings.DMOJ_PDF_PROBLEM_CACHE, '%s.%s.pdf' % (problem.code, language))
|
|
|
|
|
|
|
|
if not os.path.exists(cache):
|
|
|
|
self.logger.info('Rendering: %s.%s.pdf', problem.code, language)
|
|
|
|
with DefaultPdfMaker() as maker, translation.override(language):
|
|
|
|
problem_name = problem.name if trans is None else trans.name
|
|
|
|
maker.html = get_template('problem/raw.html').render({
|
|
|
|
'problem': problem,
|
|
|
|
'problem_name': problem_name,
|
|
|
|
'description': problem.description if trans is None else trans.description,
|
|
|
|
'url': request.build_absolute_uri(),
|
|
|
|
'math_engine': maker.math_engine,
|
|
|
|
}).replace('"//', '"https://').replace("'//", "'https://")
|
|
|
|
maker.title = problem_name
|
|
|
|
|
|
|
|
assets = ['style.css', 'pygment-github.css']
|
|
|
|
if maker.math_engine == 'jax':
|
|
|
|
assets.append('mathjax_config.js')
|
|
|
|
for file in assets:
|
|
|
|
maker.load(file, os.path.join(settings.DMOJ_RESOURCES, file))
|
|
|
|
maker.make()
|
|
|
|
if not maker.success:
|
|
|
|
self.logger.error('Failed to render PDF for %s', problem.code)
|
|
|
|
return HttpResponse(maker.log, status=500, content_type='text/plain')
|
|
|
|
shutil.move(maker.pdffile, cache)
|
|
|
|
|
|
|
|
response = HttpResponse()
|
|
|
|
if hasattr(settings, 'DMOJ_PDF_PROBLEM_INTERNAL') and \
|
|
|
|
request.META.get('SERVER_SOFTWARE', '').startswith('nginx/'):
|
|
|
|
response['X-Accel-Redirect'] = '%s/%s.%s.pdf' % (settings.DMOJ_PDF_PROBLEM_INTERNAL, problem.code, language)
|
|
|
|
else:
|
|
|
|
with open(cache, 'rb') as f:
|
|
|
|
response.content = f.read()
|
|
|
|
|
|
|
|
response['Content-Type'] = 'application/pdf'
|
|
|
|
response['Content-Disposition'] = 'inline; filename=%s.%s.pdf' % (problem.code, language)
|
|
|
|
return response
|
|
|
|
|
|
|
|
|
|
|
|
class ProblemList(QueryStringSortMixin, TitleMixin, SolvedProblemMixin, ListView):
|
|
|
|
model = Problem
|
|
|
|
title = gettext_lazy('Problems')
|
|
|
|
context_object_name = 'problems'
|
|
|
|
template_name = 'problem/list.html'
|
|
|
|
paginate_by = 50
|
2020-07-19 21:48:39 +00:00
|
|
|
sql_sort = frozenset(('date', 'points', 'ac_rate', 'user_count', 'code'))
|
2020-01-21 06:35:58 +00:00
|
|
|
manual_sort = frozenset(('name', 'group', 'solved', 'type'))
|
|
|
|
all_sorts = sql_sort | manual_sort
|
2020-07-19 21:48:39 +00:00
|
|
|
default_desc = frozenset(('date', 'points', 'ac_rate', 'user_count'))
|
|
|
|
default_sort = '-date'
|
2020-01-21 06:35:58 +00:00
|
|
|
|
|
|
|
def get_paginator(self, queryset, per_page, orphans=0,
|
|
|
|
allow_empty_first_page=True, **kwargs):
|
|
|
|
paginator = DiggPaginator(queryset, per_page, body=6, padding=2, orphans=orphans,
|
|
|
|
allow_empty_first_page=allow_empty_first_page, **kwargs)
|
|
|
|
if not self.in_contest:
|
|
|
|
# Get the number of pages and then add in this magic.
|
|
|
|
# noinspection PyStatementEffect
|
|
|
|
paginator.num_pages
|
|
|
|
|
|
|
|
queryset = queryset.add_i18n_name(self.request.LANGUAGE_CODE)
|
|
|
|
sort_key = self.order.lstrip('-')
|
|
|
|
if sort_key in self.sql_sort:
|
|
|
|
queryset = queryset.order_by(self.order)
|
|
|
|
elif sort_key == 'name':
|
|
|
|
queryset = queryset.order_by(self.order.replace('name', 'i18n_name'))
|
|
|
|
elif sort_key == 'group':
|
|
|
|
queryset = queryset.order_by(self.order + '__name')
|
|
|
|
elif sort_key == 'solved':
|
|
|
|
if self.request.user.is_authenticated:
|
|
|
|
profile = self.request.profile
|
|
|
|
solved = user_completed_ids(profile)
|
|
|
|
attempted = user_attempted_ids(profile)
|
|
|
|
|
|
|
|
def _solved_sort_order(problem):
|
|
|
|
if problem.id in solved:
|
|
|
|
return 1
|
|
|
|
if problem.id in attempted:
|
|
|
|
return 0
|
|
|
|
return -1
|
|
|
|
|
|
|
|
queryset = list(queryset)
|
|
|
|
queryset.sort(key=_solved_sort_order, reverse=self.order.startswith('-'))
|
|
|
|
elif sort_key == 'type':
|
|
|
|
if self.show_types:
|
|
|
|
queryset = list(queryset)
|
|
|
|
queryset.sort(key=lambda problem: problem.types_list[0] if problem.types_list else '',
|
|
|
|
reverse=self.order.startswith('-'))
|
|
|
|
paginator.object_list = queryset
|
|
|
|
return paginator
|
|
|
|
|
|
|
|
@cached_property
|
|
|
|
def profile(self):
|
|
|
|
if not self.request.user.is_authenticated:
|
|
|
|
return None
|
|
|
|
return self.request.profile
|
|
|
|
|
|
|
|
def get_contest_queryset(self):
|
|
|
|
queryset = self.profile.current_contest.contest.contest_problems.select_related('problem__group') \
|
|
|
|
.defer('problem__description').order_by('problem__code') \
|
|
|
|
.annotate(user_count=Count('submission__participation', distinct=True)) \
|
|
|
|
.order_by('order')
|
|
|
|
queryset = TranslatedProblemForeignKeyQuerySet.add_problem_i18n_name(queryset, 'i18n_name',
|
|
|
|
self.request.LANGUAGE_CODE,
|
|
|
|
'problem__name')
|
|
|
|
return [{
|
|
|
|
'id': p['problem_id'],
|
|
|
|
'code': p['problem__code'],
|
|
|
|
'name': p['problem__name'],
|
|
|
|
'i18n_name': p['i18n_name'],
|
|
|
|
'group': {'full_name': p['problem__group__full_name']},
|
|
|
|
'points': p['points'],
|
|
|
|
'partial': p['partial'],
|
|
|
|
'user_count': p['user_count'],
|
|
|
|
} for p in queryset.values('problem_id', 'problem__code', 'problem__name', 'i18n_name',
|
|
|
|
'problem__group__full_name', 'points', 'partial', 'user_count')]
|
|
|
|
|
|
|
|
def get_normal_queryset(self):
|
|
|
|
filter = Q(is_public=True)
|
|
|
|
if self.profile is not None:
|
|
|
|
filter |= Q(authors=self.profile)
|
|
|
|
filter |= Q(curators=self.profile)
|
|
|
|
filter |= Q(testers=self.profile)
|
|
|
|
queryset = Problem.objects.filter(filter).select_related('group').defer('description')
|
|
|
|
if not self.request.user.has_perm('see_organization_problem'):
|
|
|
|
filter = Q(is_organization_private=False)
|
|
|
|
if self.profile is not None:
|
|
|
|
filter |= Q(organizations__in=self.profile.organizations.all())
|
|
|
|
queryset = queryset.filter(filter)
|
|
|
|
if self.profile is not None and self.hide_solved:
|
|
|
|
queryset = queryset.exclude(id__in=Submission.objects.filter(user=self.profile, points=F('problem__points'))
|
|
|
|
.values_list('problem__id', flat=True))
|
|
|
|
if self.show_types:
|
|
|
|
queryset = queryset.prefetch_related('types')
|
|
|
|
if self.category is not None:
|
|
|
|
queryset = queryset.filter(group__id=self.category)
|
|
|
|
if self.selected_types:
|
|
|
|
queryset = queryset.filter(types__in=self.selected_types)
|
|
|
|
if 'search' in self.request.GET:
|
|
|
|
self.search_query = query = ' '.join(self.request.GET.getlist('search')).strip()
|
|
|
|
if query:
|
|
|
|
if settings.ENABLE_FTS and self.full_text:
|
|
|
|
queryset = queryset.search(query, queryset.BOOLEAN).extra(order_by=['-relevance'])
|
|
|
|
else:
|
|
|
|
queryset = queryset.filter(
|
|
|
|
Q(code__icontains=query) | Q(name__icontains=query) |
|
|
|
|
Q(translations__name__icontains=query, translations__language=self.request.LANGUAGE_CODE))
|
|
|
|
self.prepoint_queryset = queryset
|
|
|
|
if self.point_start is not None:
|
|
|
|
queryset = queryset.filter(points__gte=self.point_start)
|
|
|
|
if self.point_end is not None:
|
|
|
|
queryset = queryset.filter(points__lte=self.point_end)
|
|
|
|
return queryset.distinct()
|
|
|
|
|
|
|
|
def get_queryset(self):
|
|
|
|
if self.in_contest:
|
|
|
|
return self.get_contest_queryset()
|
|
|
|
else:
|
|
|
|
return self.get_normal_queryset()
|
|
|
|
|
|
|
|
def get_context_data(self, **kwargs):
|
|
|
|
context = super(ProblemList, self).get_context_data(**kwargs)
|
|
|
|
context['hide_solved'] = 0 if self.in_contest else int(self.hide_solved)
|
|
|
|
context['show_types'] = 0 if self.in_contest else int(self.show_types)
|
|
|
|
context['full_text'] = 0 if self.in_contest else int(self.full_text)
|
2020-07-19 21:48:39 +00:00
|
|
|
context['show_editorial'] = 0 if self.in_contest else int(self.show_editorial)
|
|
|
|
|
|
|
|
if (context['show_editorial']):
|
|
|
|
context['object_list'] = context['object_list'] \
|
|
|
|
.annotate(has_public_editorial=Sum(Case(When(solution__is_public=True, then=1),
|
|
|
|
default=0, output_field=IntegerField())))
|
|
|
|
|
2020-01-21 06:35:58 +00:00
|
|
|
context['category'] = self.category
|
|
|
|
context['categories'] = ProblemGroup.objects.all()
|
|
|
|
if self.show_types:
|
|
|
|
context['selected_types'] = self.selected_types
|
|
|
|
context['problem_types'] = ProblemType.objects.all()
|
|
|
|
context['has_fts'] = settings.ENABLE_FTS
|
|
|
|
context['search_query'] = self.search_query
|
|
|
|
context['completed_problem_ids'] = self.get_completed_problems()
|
|
|
|
context['attempted_problems'] = self.get_attempted_problems()
|
|
|
|
|
|
|
|
context.update(self.get_sort_paginate_context())
|
|
|
|
if not self.in_contest:
|
|
|
|
context.update(self.get_sort_context())
|
|
|
|
context['hot_problems'] = hot_problems(timedelta(days=1), 7)
|
|
|
|
context['point_start'], context['point_end'], context['point_values'] = self.get_noui_slider_points()
|
|
|
|
else:
|
|
|
|
context['hot_problems'] = None
|
|
|
|
context['point_start'], context['point_end'], context['point_values'] = 0, 0, {}
|
|
|
|
context['hide_contest_scoreboard'] = self.contest.hide_scoreboard
|
|
|
|
return context
|
|
|
|
|
|
|
|
def get_noui_slider_points(self):
|
|
|
|
points = sorted(self.prepoint_queryset.values_list('points', flat=True).distinct())
|
|
|
|
if not points:
|
|
|
|
return 0, 0, {}
|
|
|
|
if len(points) == 1:
|
|
|
|
return points[0], points[0], {
|
|
|
|
'min': points[0] - 1,
|
|
|
|
'max': points[0] + 1,
|
|
|
|
}
|
|
|
|
|
|
|
|
start, end = points[0], points[-1]
|
|
|
|
if self.point_start is not None:
|
|
|
|
start = self.point_start
|
|
|
|
if self.point_end is not None:
|
|
|
|
end = self.point_end
|
|
|
|
points_map = {0.0: 'min', 1.0: 'max'}
|
|
|
|
size = len(points) - 1
|
|
|
|
return start, end, {points_map.get(i / size, '%.2f%%' % (100 * i / size,)): j for i, j in enumerate(points)}
|
|
|
|
|
|
|
|
def GET_with_session(self, request, key):
|
|
|
|
if not request.GET:
|
|
|
|
return request.session.get(key, False)
|
|
|
|
return request.GET.get(key, None) == '1'
|
|
|
|
|
|
|
|
def setup_problem_list(self, request):
|
|
|
|
self.hide_solved = self.GET_with_session(request, 'hide_solved')
|
|
|
|
self.show_types = self.GET_with_session(request, 'show_types')
|
|
|
|
self.full_text = self.GET_with_session(request, 'full_text')
|
2020-07-19 21:48:39 +00:00
|
|
|
self.show_editorial = self.GET_with_session(request, 'show_editorial')
|
|
|
|
|
2020-01-21 06:35:58 +00:00
|
|
|
self.search_query = None
|
|
|
|
self.category = None
|
|
|
|
self.selected_types = []
|
|
|
|
|
|
|
|
# This actually copies into the instance dictionary...
|
|
|
|
self.all_sorts = set(self.all_sorts)
|
|
|
|
if not self.show_types:
|
|
|
|
self.all_sorts.discard('type')
|
|
|
|
|
|
|
|
self.category = safe_int_or_none(request.GET.get('category'))
|
|
|
|
if 'type' in request.GET:
|
|
|
|
try:
|
|
|
|
self.selected_types = list(map(int, request.GET.getlist('type')))
|
|
|
|
except ValueError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
self.point_start = safe_float_or_none(request.GET.get('point_start'))
|
|
|
|
self.point_end = safe_float_or_none(request.GET.get('point_end'))
|
|
|
|
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
self.setup_problem_list(request)
|
|
|
|
|
|
|
|
try:
|
|
|
|
return super(ProblemList, self).get(request, *args, **kwargs)
|
|
|
|
except ProgrammingError as e:
|
|
|
|
return generic_message(request, 'FTS syntax error', e.args[1], status=400)
|
|
|
|
|
|
|
|
def post(self, request, *args, **kwargs):
|
2020-07-19 21:39:28 +00:00
|
|
|
to_update = ('hide_solved', 'show_types', 'full_text')
|
2020-01-21 06:35:58 +00:00
|
|
|
for key in to_update:
|
|
|
|
if key in request.GET:
|
|
|
|
val = request.GET.get(key) == '1'
|
|
|
|
request.session[key] = val
|
|
|
|
else:
|
|
|
|
request.session.pop(key, None)
|
|
|
|
return HttpResponseRedirect(request.get_full_path())
|
|
|
|
|
|
|
|
|
|
|
|
class LanguageTemplateAjax(View):
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
try:
|
|
|
|
language = get_object_or_404(Language, id=int(request.GET.get('id', 0)))
|
|
|
|
except ValueError:
|
|
|
|
raise Http404()
|
|
|
|
return HttpResponse(language.template, content_type='text/plain')
|
|
|
|
|
|
|
|
|
|
|
|
class RandomProblem(ProblemList):
|
|
|
|
def get(self, request, *args, **kwargs):
|
|
|
|
self.setup_problem_list(request)
|
|
|
|
if self.in_contest:
|
|
|
|
raise Http404()
|
|
|
|
|
|
|
|
queryset = self.get_normal_queryset()
|
|
|
|
count = queryset.count()
|
|
|
|
if not count:
|
|
|
|
return HttpResponseRedirect('%s%s%s' % (reverse('problem_list'), request.META['QUERY_STRING'] and '?',
|
|
|
|
request.META['QUERY_STRING']))
|
|
|
|
return HttpResponseRedirect(queryset[randrange(count)].get_absolute_url())
|
|
|
|
|
|
|
|
|
|
|
|
user_logger = logging.getLogger('judge.user')
|
|
|
|
|
|
|
|
|
|
|
|
@login_required
|
2020-07-19 21:39:28 +00:00
|
|
|
def problem_submit(request, problem, submission=None):
|
2020-01-21 06:35:58 +00:00
|
|
|
if submission is not None and not request.user.has_perm('judge.resubmit_other') and \
|
|
|
|
get_object_or_404(Submission, id=int(submission)).user.user != request.user:
|
|
|
|
raise PermissionDenied()
|
|
|
|
|
|
|
|
profile = request.profile
|
2020-07-19 21:39:28 +00:00
|
|
|
problem = get_object_or_404(Problem, code=problem)
|
|
|
|
if not problem.is_accessible_by(request.user):
|
|
|
|
if request.method == 'POST':
|
|
|
|
user_logger.info('Naughty user %s wants to submit to %s without permission',
|
|
|
|
request.user.username, problem.code)
|
|
|
|
return HttpResponseForbidden('<h1>Do you want me to ban you?</h1>')
|
|
|
|
raise Http404()
|
2020-07-19 21:27:14 +00:00
|
|
|
|
|
|
|
if problem.is_editable_by(request.user):
|
|
|
|
judge_choices = tuple(Judge.objects.filter(online=True, problems=problem).values_list('name', 'name'))
|
|
|
|
else:
|
|
|
|
judge_choices = ()
|
|
|
|
|
2020-01-21 06:35:58 +00:00
|
|
|
if request.method == 'POST':
|
2020-07-19 21:27:14 +00:00
|
|
|
form = ProblemSubmitForm(request.POST, judge_choices=judge_choices,
|
|
|
|
instance=Submission(user=profile, problem=problem))
|
2020-01-21 06:35:58 +00:00
|
|
|
if form.is_valid():
|
|
|
|
if (not request.user.has_perm('judge.spam_submission') and
|
|
|
|
Submission.objects.filter(user=profile, was_rejudged=False)
|
|
|
|
.exclude(status__in=['D', 'IE', 'CE', 'AB']).count() >= settings.DMOJ_SUBMISSION_LIMIT):
|
|
|
|
return HttpResponse('<h1>You submitted too many submissions.</h1>', status=429)
|
2020-07-19 21:39:28 +00:00
|
|
|
if not problem.allowed_languages.filter(id=form.cleaned_data['language'].id).exists():
|
2020-01-21 06:35:58 +00:00
|
|
|
raise PermissionDenied()
|
2020-07-19 21:39:28 +00:00
|
|
|
if not request.user.is_superuser and problem.banned_users.filter(id=profile.id).exists():
|
2020-01-21 06:35:58 +00:00
|
|
|
return generic_message(request, _('Banned from submitting'),
|
|
|
|
_('You have been declared persona non grata for this problem. '
|
|
|
|
'You are permanently barred from submitting this problem.'))
|
|
|
|
|
|
|
|
with transaction.atomic():
|
|
|
|
if profile.current_contest is not None:
|
|
|
|
contest_id = profile.current_contest.contest_id
|
|
|
|
try:
|
2020-07-19 21:39:28 +00:00
|
|
|
contest_problem = problem.contests.get(contest_id=contest_id)
|
2020-01-21 06:35:58 +00:00
|
|
|
except ContestProblem.DoesNotExist:
|
|
|
|
model = form.save()
|
|
|
|
else:
|
|
|
|
max_subs = contest_problem.max_submissions
|
|
|
|
if max_subs and get_contest_submission_count(problem, profile,
|
|
|
|
profile.current_contest.virtual) >= max_subs:
|
|
|
|
return generic_message(request, _('Too many submissions'),
|
|
|
|
_('You have exceeded the submission limit for this problem.'))
|
|
|
|
model = form.save()
|
|
|
|
model.contest_object_id = contest_id
|
|
|
|
|
|
|
|
contest = ContestSubmission(submission=model, problem=contest_problem,
|
|
|
|
participation=profile.current_contest)
|
|
|
|
contest.save()
|
|
|
|
else:
|
|
|
|
model = form.save()
|
|
|
|
|
|
|
|
# Create the SubmissionSource object
|
|
|
|
source = SubmissionSource(submission=model, source=form.cleaned_data['source'])
|
|
|
|
source.save()
|
|
|
|
profile.update_contest()
|
|
|
|
|
|
|
|
# Save a query
|
|
|
|
model.source = source
|
2020-07-19 21:27:14 +00:00
|
|
|
model.judge(rejudge=False, judge_id=form.cleaned_data['judge'])
|
2020-01-21 06:35:58 +00:00
|
|
|
|
|
|
|
return HttpResponseRedirect(reverse('submission_status', args=[str(model.id)]))
|
|
|
|
else:
|
|
|
|
form_data = form.cleaned_data
|
|
|
|
if submission is not None:
|
|
|
|
sub = get_object_or_404(Submission, id=int(submission))
|
|
|
|
else:
|
|
|
|
initial = {'language': profile.language}
|
|
|
|
if submission is not None:
|
|
|
|
try:
|
|
|
|
sub = get_object_or_404(Submission.objects.select_related('source', 'language'), id=int(submission))
|
|
|
|
initial['source'] = sub.source.source
|
|
|
|
initial['language'] = sub.language
|
|
|
|
except ValueError:
|
|
|
|
raise Http404()
|
2020-07-19 21:27:14 +00:00
|
|
|
form = ProblemSubmitForm(judge_choices=judge_choices, initial=initial)
|
2020-01-21 06:35:58 +00:00
|
|
|
form_data = initial
|
2020-07-19 21:39:28 +00:00
|
|
|
form.fields['language'].queryset = (
|
|
|
|
problem.usable_languages.order_by('name', 'key')
|
|
|
|
.prefetch_related(Prefetch('runtimeversion_set', RuntimeVersion.objects.order_by('priority')))
|
|
|
|
)
|
2020-01-21 06:35:58 +00:00
|
|
|
if 'language' in form_data:
|
|
|
|
form.fields['source'].widget.mode = form_data['language'].ace
|
|
|
|
form.fields['source'].widget.theme = profile.ace_theme
|
|
|
|
|
|
|
|
if submission is not None:
|
|
|
|
default_lang = sub.language
|
|
|
|
else:
|
|
|
|
default_lang = request.profile.language
|
|
|
|
|
|
|
|
submission_limit = submissions_left = None
|
|
|
|
if profile.current_contest is not None:
|
|
|
|
try:
|
2020-07-19 21:39:28 +00:00
|
|
|
submission_limit = problem.contests.get(contest=profile.current_contest.contest).max_submissions
|
2020-01-21 06:35:58 +00:00
|
|
|
except ContestProblem.DoesNotExist:
|
|
|
|
pass
|
|
|
|
else:
|
|
|
|
if submission_limit:
|
|
|
|
submissions_left = submission_limit - get_contest_submission_count(problem, profile,
|
|
|
|
profile.current_contest.virtual)
|
|
|
|
return render(request, 'problem/submit.html', {
|
|
|
|
'form': form,
|
|
|
|
'title': _('Submit to %(problem)s') % {
|
2020-07-19 21:39:28 +00:00
|
|
|
'problem': problem.translated_name(request.LANGUAGE_CODE),
|
2020-01-21 06:35:58 +00:00
|
|
|
},
|
|
|
|
'content_title': mark_safe(escape(_('Submit to %(problem)s')) % {
|
|
|
|
'problem': format_html('<a href="{0}">{1}</a>',
|
2020-07-19 21:39:28 +00:00
|
|
|
reverse('problem_detail', args=[problem.code]),
|
|
|
|
problem.translated_name(request.LANGUAGE_CODE)),
|
2020-01-21 06:35:58 +00:00
|
|
|
}),
|
|
|
|
'langs': Language.objects.all(),
|
|
|
|
'no_judges': not form.fields['language'].queryset,
|
|
|
|
'submission_limit': submission_limit,
|
|
|
|
'submissions_left': submissions_left,
|
|
|
|
'ACE_URL': settings.ACE_URL,
|
|
|
|
'default_lang': default_lang,
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
class ProblemClone(ProblemMixin, PermissionRequiredMixin, TitleMixin, SingleObjectFormView):
|
|
|
|
title = _('Clone Problem')
|
|
|
|
template_name = 'problem/clone.html'
|
|
|
|
form_class = ProblemCloneForm
|
|
|
|
permission_required = 'judge.clone_problem'
|
|
|
|
|
|
|
|
def form_valid(self, form):
|
|
|
|
problem = self.object
|
|
|
|
|
|
|
|
languages = problem.allowed_languages.all()
|
|
|
|
language_limits = problem.language_limits.all()
|
|
|
|
types = problem.types.all()
|
|
|
|
problem.pk = None
|
|
|
|
problem.is_public = False
|
|
|
|
problem.ac_rate = 0
|
|
|
|
problem.user_count = 0
|
|
|
|
problem.code = form.cleaned_data['code']
|
|
|
|
problem.save()
|
|
|
|
problem.authors.add(self.request.profile)
|
|
|
|
problem.allowed_languages.set(languages)
|
|
|
|
problem.language_limits.set(language_limits)
|
|
|
|
problem.types.set(types)
|
|
|
|
|
2020-07-26 05:11:03 +00:00
|
|
|
return HttpResponseRedirect(reverse('admin:judge_problem_change', args=(problem.id,)))
|
|
|
|
|
|
|
|
|
|
|
|
def download_submissions(request, problem):
|
|
|
|
submissions = Submission.objects.filter(problem__code=problem, result='AC')
|
|
|
|
|
|
|
|
with tempfile.SpooledTemporaryFile() as tmp:
|
|
|
|
with zipfile.ZipFile(tmp, 'w', zipfile.ZIP_DEFLATED) as archive:
|
|
|
|
for submission in submissions:
|
|
|
|
file_name = str(submission.id) + '.' + str(submission.language.key)
|
|
|
|
archive.writestr(file_name, submission.source.source)
|
|
|
|
|
|
|
|
# Reset file pointer
|
|
|
|
tmp.seek(0)
|
|
|
|
|
|
|
|
# Write file data to response
|
|
|
|
response = HttpResponse(tmp.read(), content_type='application/x-zip-compressed')
|
|
|
|
response['Content-Disposition'] = 'attachment; filename="%s"' % (str(problem) + '_submissions.zip')
|
|
|
|
return response
|