NDOJ/judge/utils/problems.py

322 lines
9.8 KiB
Python
Raw Normal View History

2020-01-21 06:35:58 +00:00
from collections import defaultdict
from math import e
2023-11-09 08:43:11 +00:00
from datetime import datetime, timedelta
2023-01-28 01:15:37 +00:00
import random
2023-11-09 08:43:11 +00:00
from enum import Enum
2020-01-21 06:35:58 +00:00
2021-05-25 04:22:56 +00:00
from django.conf import settings
2020-01-21 06:35:58 +00:00
from django.core.cache import cache
from django.db.models import Case, Count, ExpressionWrapper, F, Max, Q, When
from django.db.models.fields import FloatField
from django.utils import timezone
from django.utils.translation import gettext as _, gettext_noop
2023-11-09 08:43:11 +00:00
from django.http import Http404
2020-01-21 06:35:58 +00:00
from judge.models import Problem, Submission
2023-01-28 01:15:37 +00:00
from judge.ml.collab_filter import CollabFilter
2023-04-05 17:49:23 +00:00
from judge.caching import cache_wrapper
2020-01-21 06:35:58 +00:00
2022-05-14 17:57:27 +00:00
__all__ = [
"contest_completed_ids",
"get_result_data",
"user_completed_ids",
"user_editable_ids",
2023-02-18 22:38:47 +00:00
"user_tester_ids",
2022-05-14 17:57:27 +00:00
]
2020-01-21 06:35:58 +00:00
2023-10-12 01:33:48 +00:00
@cache_wrapper(prefix="user_tester")
2023-02-18 22:38:47 +00:00
def user_tester_ids(profile):
return set(
2023-10-12 13:56:53 +00:00
Problem.testers.through.objects.filter(profile=profile)
.values_list("problem_id", flat=True)
.distinct()
2023-02-18 22:38:47 +00:00
)
2020-01-21 06:35:58 +00:00
2023-10-12 01:33:48 +00:00
@cache_wrapper(prefix="user_editable")
2020-01-21 06:35:58 +00:00
def user_editable_ids(profile):
2022-05-14 17:57:27 +00:00
result = set(
(
Problem.objects.filter(authors=profile)
| Problem.objects.filter(curators=profile)
2023-10-12 13:56:53 +00:00
)
.values_list("id", flat=True)
.distinct()
2022-05-14 17:57:27 +00:00
)
2020-01-21 06:35:58 +00:00
return result
2023-10-12 01:33:48 +00:00
@cache_wrapper(prefix="contest_complete")
2020-01-21 06:35:58 +00:00
def contest_completed_ids(participation):
2023-10-12 01:33:48 +00:00
result = set(
participation.submissions.filter(
submission__result="AC", points=F("problem__points")
2022-05-14 17:57:27 +00:00
)
2023-10-12 01:33:48 +00:00
.values_list("problem__problem__id", flat=True)
.distinct()
)
2020-01-21 06:35:58 +00:00
return result
2023-10-12 01:33:48 +00:00
@cache_wrapper(prefix="user_complete")
2020-01-21 06:35:58 +00:00
def user_completed_ids(profile):
2023-04-05 17:49:23 +00:00
result = set(
Submission.objects.filter(
user=profile, result="AC", points=F("problem__points")
2022-05-14 17:57:27 +00:00
)
2023-04-05 17:49:23 +00:00
.values_list("problem_id", flat=True)
.distinct()
)
2020-01-21 06:35:58 +00:00
return result
2023-10-12 01:33:48 +00:00
@cache_wrapper(prefix="contest_attempted")
2020-01-21 06:35:58 +00:00
def contest_attempted_ids(participation):
2023-04-05 17:49:23 +00:00
result = {
id: {"achieved_points": points, "max_points": max_points}
for id, max_points, points in (
participation.submissions.values_list(
"problem__problem__id", "problem__points"
2022-05-14 17:57:27 +00:00
)
2023-04-05 17:49:23 +00:00
.annotate(points=Max("points"))
.filter(points__lt=F("problem__points"))
)
}
2020-01-21 06:35:58 +00:00
return result
2023-10-12 01:33:48 +00:00
@cache_wrapper(prefix="user_attempted")
2020-01-21 06:35:58 +00:00
def user_attempted_ids(profile):
2023-04-05 17:49:23 +00:00
result = {
id: {
"achieved_points": points,
"max_points": max_points,
"last_submission": last_submission,
"code": problem_code,
"name": problem_name,
2022-05-14 17:57:27 +00:00
}
2023-04-05 17:49:23 +00:00
for id, max_points, problem_code, problem_name, points, last_submission in (
Submission.objects.filter(user=profile)
.values_list(
"problem__id", "problem__points", "problem__code", "problem__name"
)
.annotate(points=Max("points"), last_submission=Max("id"))
.filter(points__lt=F("problem__points"))
)
}
2020-01-21 06:35:58 +00:00
return result
def _get_result_data(results):
return {
2022-05-14 17:57:27 +00:00
"categories": [
2020-01-21 06:35:58 +00:00
# Using gettext_noop here since this will be tacked into the cache, so it must be language neutral.
# The caller, SubmissionList.get_result_data will run ugettext on the name.
2022-05-14 17:57:27 +00:00
{"code": "AC", "name": gettext_noop("Accepted"), "count": results["AC"]},
{"code": "WA", "name": gettext_noop("Wrong"), "count": results["WA"]},
{
"code": "CE",
"name": gettext_noop("Compile Error"),
"count": results["CE"],
},
{"code": "TLE", "name": gettext_noop("Timeout"), "count": results["TLE"]},
{
"code": "ERR",
"name": gettext_noop("Error"),
"count": results["MLE"]
+ results["OLE"]
+ results["IR"]
+ results["RTE"]
+ results["AB"]
+ results["IE"],
},
2020-01-21 06:35:58 +00:00
],
2022-05-14 17:57:27 +00:00
"total": sum(results.values()),
2020-01-21 06:35:58 +00:00
}
def get_result_data(*args, **kwargs):
if args:
submissions = args[0]
if kwargs:
raise ValueError(_("Can't pass both queryset and keyword filters"))
else:
2022-05-14 17:57:27 +00:00
submissions = (
Submission.objects.filter(**kwargs)
if kwargs is not None
else Submission.objects
)
raw = (
submissions.values("result")
.annotate(count=Count("result"))
.values_list("result", "count")
)
2020-01-21 06:35:58 +00:00
return _get_result_data(defaultdict(int, raw))
def editable_problems(user, profile=None):
subquery = Problem.objects.all()
if profile is None:
profile = user.profile
2022-05-14 17:57:27 +00:00
if not user.has_perm("judge.edit_all_problem"):
2020-01-21 06:35:58 +00:00
subfilter = Q(authors__id=profile.id) | Q(curators__id=profile.id)
2022-05-14 17:57:27 +00:00
if user.has_perm("judge.edit_public_problem"):
2020-01-21 06:35:58 +00:00
subfilter |= Q(is_public=True)
subquery = subquery.filter(subfilter)
return subquery
2023-04-05 17:49:23 +00:00
@cache_wrapper(prefix="hp", timeout=900)
2020-01-21 06:35:58 +00:00
def hot_problems(duration, limit):
2023-04-05 17:49:23 +00:00
qs = Problem.get_public_problems().filter(
submission__date__gt=timezone.now() - duration
)
qs0 = (
qs.annotate(k=Count("submission__user", distinct=True))
.order_by("-k")
.values_list("k", flat=True)
)
2020-01-21 06:35:58 +00:00
2023-04-05 17:49:23 +00:00
if not qs0:
return []
# make this an aggregate
mx = float(qs0[0])
qs = qs.annotate(unique_user_count=Count("submission__user", distinct=True))
# fix braindamage in excluding CE
qs = qs.annotate(
submission_volume=Count(
Case(
When(submission__result="AC", then=1),
When(submission__result="WA", then=1),
When(submission__result="IR", then=1),
When(submission__result="RTE", then=1),
When(submission__result="TLE", then=1),
When(submission__result="OLE", then=1),
output_field=FloatField(),
2022-05-14 17:57:27 +00:00
)
)
2023-04-05 17:49:23 +00:00
)
qs = qs.annotate(
ac_volume=Count(
Case(
When(submission__result="AC", then=1),
output_field=FloatField(),
2022-05-14 17:57:27 +00:00
)
)
2023-04-05 17:49:23 +00:00
)
qs = qs.filter(unique_user_count__gt=max(mx / 3.0, 1))
qs = (
qs.annotate(
ordering=ExpressionWrapper(
0.02
* F("points")
* (0.4 * F("ac_volume") / F("submission_volume") + 0.6 * F("ac_rate"))
+ 100 * e ** (F("unique_user_count") / mx),
output_field=FloatField(),
2022-05-14 17:57:27 +00:00
)
)
2023-04-05 17:49:23 +00:00
.order_by("-ordering")
.defer("description")[:limit]
)
2022-05-14 17:57:27 +00:00
return qs
2023-01-28 01:15:37 +00:00
2023-04-05 17:49:23 +00:00
@cache_wrapper(prefix="grp", timeout=26400)
2023-01-28 01:15:37 +00:00
def get_related_problems(profile, problem, limit=8):
2023-01-28 01:16:48 +00:00
if not profile or not settings.ML_OUTPUT_PATH:
2023-01-28 01:15:37 +00:00
return None
problemset = Problem.get_visible_problems(profile.user).values_list("id", flat=True)
problemset = problemset.exclude(id__in=user_completed_ids(profile))
problemset = problemset.exclude(id=problem.id)
cf_model = CollabFilter("collab_filter")
results = cf_model.problem_neighbors(
problem, problemset, CollabFilter.DOT, limit
) + cf_model.problem_neighbors(problem, problemset, CollabFilter.COSINE, limit)
results = list(set([i[1] for i in results]))
seed = datetime.now().strftime("%d%m%Y")
2023-04-05 17:49:23 +00:00
random.shuffle(results)
2023-01-28 01:15:37 +00:00
results = results[:limit]
2023-01-28 16:46:17 +00:00
results = [Problem.objects.get(id=i) for i in results]
return results
2023-04-05 17:49:23 +00:00
def finished_submission(sub):
keys = ["user_complete:%d" % sub.user_id, "user_attempted:%s" % sub.user_id]
if hasattr(sub, "contest"):
participation = sub.contest.participation
keys += ["contest_complete:%d" % participation.id]
keys += ["contest_attempted:%d" % participation.id]
cache.delete_many(keys)
2023-11-09 08:43:11 +00:00
class RecommendationType(Enum):
HOT_PROBLEM = 1
CF_DOT = 2
CF_COSINE = 3
CF_TIME_DOT = 4
CF_TIME_COSINE = 5
# Return a list of list. Each inner list correspond to each type in types
def get_user_recommended_problems(
user_id,
problem_ids,
recommendation_types,
limits,
shuffle=False,
):
cf_model = CollabFilter("collab_filter")
cf_time_model = CollabFilter("collab_filter_time")
def get_problem_ids_from_type(rec_type, limit):
if type(rec_type) == int:
try:
rec_type = RecommendationType(rec_type)
except ValueError:
raise Http404()
if rec_type == RecommendationType.HOT_PROBLEM:
return [
problem.id
for problem in hot_problems(timedelta(days=7), limit)
if problem.id in set(problem_ids)
]
if rec_type == RecommendationType.CF_DOT:
return cf_model.user_recommendations(
user_id, problem_ids, cf_model.DOT, limit
)
if rec_type == RecommendationType.CF_COSINE:
return cf_model.user_recommendations(
user_id, problem_ids, cf_model.COSINE, limit
)
if rec_type == RecommendationType.CF_TIME_DOT:
return cf_time_model.user_recommendations(
user_id, problem_ids, cf_model.DOT, limit
)
if rec_type == RecommendationType.CF_TIME_COSINE:
return cf_time_model.user_recommendations(
user_id, problem_ids, cf_model.COSINE, limit
)
return []
all_problems = []
for rec_type, limit in zip(recommendation_types, limits):
all_problems += get_problem_ids_from_type(rec_type, limit)
if shuffle:
seed = datetime.now().strftime("%d%m%Y")
random.Random(seed).shuffle(all_problems)
# deduplicate problems
res = []
used_pid = set()
for obj in all_problems:
if type(obj) == tuple:
obj = obj[1]
if obj not in used_pid:
res.append(obj)
used_pid.add(obj)
return res