remove rank

This commit is contained in:
2026-03-29 10:07:22 -06:00
parent 504c2bb70b
commit 8456b76586
4 changed files with 20 additions and 74 deletions

View File

@@ -17,7 +17,6 @@ from .schemas import (
SubmissionOut, SubmissionOut,
RatingScoreIn, RatingScoreIn,
TaskStatsOut, TaskStatsOut,
TopSubmission,
UserTag, UserTag,
) )
@@ -44,10 +43,6 @@ def create_submission(request, payload: SubmissionIn):
) )
conversation.is_active = False conversation.is_active = False
conversation.save(update_fields=["is_active"]) conversation.save(update_fields=["is_active"])
# 如果用户之前已参与排名,自动转移提名到新提交
had_nomination = Submission.objects.filter(
user=request.user, task=task, nominated=True
).update(nominated=False) > 0
Submission.objects.create( Submission.objects.create(
user=request.user, user=request.user,
@@ -56,7 +51,6 @@ def create_submission(request, payload: SubmissionIn):
css=payload.css, css=payload.css,
js=payload.js, js=payload.js,
conversation=conversation, conversation=conversation,
nominated=had_nomination,
) )
@@ -67,8 +61,10 @@ def list_submissions(request, filters: SubmissionFilter = Query(...)):
""" """
获取提交列表,支持按任务和用户过滤 获取提交列表,支持按任务和用户过滤
""" """
submissions = Submission.objects.select_related("task", "user").defer( submissions = (
"html", "css", "js" Submission.objects.select_related("task", "user")
.defer("html", "css", "js")
.exclude(conversation__isnull=False, html__isnull=True, css__isnull=True, js__isnull=True)
) )
if filters.task_id: if filters.task_id:
@@ -86,8 +82,6 @@ def list_submissions(request, filters: SubmissionFilter = Query(...)):
else: else:
submissions = submissions.filter(flag=filters.flag) submissions = submissions.filter(flag=filters.flag)
if filters.nominated is not None:
submissions = submissions.filter(nominated=filters.nominated)
if filters.score_lt_threshold is not None: if filters.score_lt_threshold is not None:
submissions = submissions.filter(score__lt=filters.score_lt_threshold) submissions = submissions.filter(score__lt=filters.score_lt_threshold)
else: else:
@@ -142,6 +136,7 @@ def list_by_user_task(request, user_id: int, task_id: int):
) )
return ( return (
Submission.objects.filter(user_id=user_id, task_id=task_id) Submission.objects.filter(user_id=user_id, task_id=task_id)
.exclude(conversation__isnull=False, html__isnull=True, css__isnull=True, js__isnull=True)
.select_related("task", "user") .select_related("task", "user")
.defer("html", "css", "js") .defer("html", "css", "js")
.annotate(my_score=user_rating_subquery) .annotate(my_score=user_rating_subquery)
@@ -261,14 +256,6 @@ def get_task_stats(request, task_id: int, classname: Optional[str] = None):
for u in students.filter(id__in=unrated_ids).order_by("classname", "username") for u in students.filter(id__in=unrated_ids).order_by("classname", "username")
] ]
# Nominated count: distinct users with nominated=True (task-wide, not class-filtered)
nominated_count = (
Submission.objects.filter(task=task, nominated=True)
.values("user_id")
.distinct()
.count()
)
# Submission count distribution # Submission count distribution
sub_counts = dict( sub_counts = dict(
Submission.objects.filter(task=task, user_id__in=submitted_ids) Submission.objects.filter(task=task, user_id__in=submitted_ids)
@@ -287,24 +274,6 @@ def get_task_stats(request, task_id: int, classname: Optional[str] = None):
else: else:
dist["count_4_plus"] += 1 dist["count_4_plus"] += 1
# Top 5 submissions by rating count
top_subs_qs = (
Submission.objects.filter(task=task, user_id__in=student_ids)
.select_related("user")
.annotate(rating_count=Count("ratings"))
.order_by("-rating_count")[:5]
)
top_submissions = [
TopSubmission(
submission_id=str(s.id),
username=s.user.username,
classname=s.user.classname,
score=s.score,
rating_count=s.rating_count,
)
for s in top_subs_qs
]
# Flag stats (all submissions for this task, not grouped by user) # Flag stats (all submissions for this task, not grouped by user)
flag_counts = dict( flag_counts = dict(
Submission.objects.filter(task=task, flag__isnull=False) Submission.objects.filter(task=task, flag__isnull=False)
@@ -324,11 +293,9 @@ def get_task_stats(request, task_id: int, classname: Optional[str] = None):
unsubmitted_count=unsubmitted_count, unsubmitted_count=unsubmitted_count,
average_score=average_score, average_score=average_score,
unrated_count=unrated_count, unrated_count=unrated_count,
nominated_count=nominated_count,
unsubmitted_users=unsubmitted_users, unsubmitted_users=unsubmitted_users,
unrated_users=unrated_users, unrated_users=unrated_users,
submission_count_distribution=SubmissionCountBucket(**dist), submission_count_distribution=SubmissionCountBucket(**dist),
top_submissions=top_submissions,
flag_stats=flag_stats, flag_stats=flag_stats,
classes=all_classes, classes=all_classes,
) )
@@ -393,25 +360,3 @@ def update_flag(request, submission_id: UUID, payload: FlagIn):
return {"flag": submission.flag} return {"flag": submission.flag}
@router.put("/{submission_id}/nominate")
@login_required
def nominate_submission(request, submission_id: UUID):
"""
学生将某条提交标记为"参与排名"
同一用户同一题目只能有一条参与排名,旧的自动取消。
"""
submission = get_object_or_404(Submission, id=submission_id)
if submission.user != request.user:
raise HttpError(403, "只能提名自己的提交")
Submission.objects.filter(
user=request.user,
task=submission.task,
nominated=True,
).exclude(pk=submission.pk).update(nominated=False)
submission.nominated = True
submission.save(update_fields=["nominated"])
return {"nominated": True}

View File

@@ -0,0 +1,15 @@
from django.db import migrations
class Migration(migrations.Migration):
dependencies = [
("submission", "0006_add_raw_score_nominated"),
]
operations = [
migrations.RemoveField(
model_name="submission",
name="nominated",
),
]

View File

@@ -49,7 +49,6 @@ class Submission(TimeStampedModel):
verbose_name="标记", verbose_name="标记",
) )
raw_score = models.FloatField(default=0.0, verbose_name="原始加权分") raw_score = models.FloatField(default=0.0, verbose_name="原始加权分")
nominated = models.BooleanField(default=False, db_index=True, verbose_name="参与排名")
class Meta: class Meta:
ordering = ("-created",) ordering = ("-created",)

View File

@@ -26,7 +26,6 @@ class SubmissionOut(Schema):
js: Optional[str] = None js: Optional[str] = None
conversation_id: Optional[UUID] = None conversation_id: Optional[UUID] = None
flag: Optional[str] = None flag: Optional[str] = None
nominated: bool = False
submit_count: int = 0 submit_count: int = 0
created: str created: str
modified: str modified: str
@@ -88,7 +87,6 @@ class SubmissionOut(Schema):
"js": submission.js, "js": submission.js,
"conversation_id": submission.conversation_id, "conversation_id": submission.conversation_id,
"flag": submission.flag, "flag": submission.flag,
"nominated": submission.nominated,
"created": submission.created.isoformat(), "created": submission.created.isoformat(),
"modified": submission.modified.isoformat(), "modified": submission.modified.isoformat(),
} }
@@ -112,7 +110,6 @@ class SubmissionFilter(Schema):
score_min: Optional[float] = None score_min: Optional[float] = None
score_max_exclusive: Optional[float] = None score_max_exclusive: Optional[float] = None
score_lt_threshold: Optional[float] = None score_lt_threshold: Optional[float] = None
nominated: Optional[bool] = None
ordering: Optional[str] = None ordering: Optional[str] = None
grouped: Optional[bool] = True grouped: Optional[bool] = True
@@ -126,14 +123,6 @@ class UserTag(Schema):
classname: str classname: str
class TopSubmission(Schema):
submission_id: str # UUID as string
username: str
classname: str
score: float
rating_count: int
class SubmissionCountBucket(Schema): class SubmissionCountBucket(Schema):
count_1: int # users with exactly 1 submission count_1: int # users with exactly 1 submission
count_2: int # users with exactly 2 submissions count_2: int # users with exactly 2 submissions
@@ -153,11 +142,9 @@ class TaskStatsOut(Schema):
unsubmitted_count: int unsubmitted_count: int
average_score: Optional[float] average_score: Optional[float]
unrated_count: int unrated_count: int
nominated_count: int
unsubmitted_users: list[UserTag] unsubmitted_users: list[UserTag]
unrated_users: list[UserTag] unrated_users: list[UserTag]
submission_count_distribution: SubmissionCountBucket submission_count_distribution: SubmissionCountBucket
top_submissions: list[TopSubmission]
flag_stats: FlagStats flag_stats: FlagStats
classes: list[str] classes: list[str]