71 lines
2.5 KiB
Python
71 lines
2.5 KiB
Python
from django.core.management.base import BaseCommand
|
||
|
||
from account.models import UserProfile
|
||
from problem.models import Problem
|
||
from submission.models import JudgeStatus
|
||
|
||
ACCEPTED_STATUSES = {JudgeStatus.ACCEPTED, JudgeStatus.AST_CHECK_FAILED}
|
||
|
||
|
||
class Command(BaseCommand):
|
||
help = "从用户 Profile 中移除已被删除的题目记录,并同步修正 accepted_number"
|
||
|
||
def add_arguments(self, parser):
|
||
parser.add_argument("--dry-run", action="store_true", help="只检查,不写入数据库")
|
||
|
||
def handle(self, *args, **options):
|
||
dry_run = options["dry_run"]
|
||
|
||
# 所有现存非比赛题目的 PK 集合
|
||
existing_ids = set(
|
||
Problem.objects.filter(contest__isnull=True).values_list("id", flat=True)
|
||
)
|
||
self.stdout.write(f"现存题库题目数: {len(existing_ids)}")
|
||
|
||
profiles = UserProfile.objects.select_related("user").exclude(
|
||
acm_problems_status={}
|
||
)
|
||
total = profiles.count()
|
||
self.stdout.write(f"检查用户数: {total}{'(dry-run 模式)' if dry_run else ''}")
|
||
|
||
fixed_count = 0
|
||
for profile in profiles:
|
||
problems = profile.acm_problems_status.get("problems", {})
|
||
if not problems:
|
||
continue
|
||
|
||
stale_keys = [k for k in problems if int(k) not in existing_ids]
|
||
if not stale_keys:
|
||
continue
|
||
|
||
removed_accepted = sum(
|
||
1
|
||
for k in stale_keys
|
||
if problems[k].get("status") in ACCEPTED_STATUSES
|
||
)
|
||
|
||
stale_display = [problems[k].get("_id", k) for k in stale_keys]
|
||
self.stdout.write(
|
||
f" 用户 {profile.user.username}"
|
||
f" | 删除 {len(stale_keys)} 题: {', '.join(stale_display)}"
|
||
f"{f' | 其中已AC {removed_accepted} 题' if removed_accepted else ''}"
|
||
)
|
||
|
||
if dry_run:
|
||
continue
|
||
|
||
for k in stale_keys:
|
||
del profile.acm_problems_status["problems"][k]
|
||
|
||
if removed_accepted:
|
||
# 防止 accepted_number 变为负数
|
||
profile.accepted_number = max(0, profile.accepted_number - removed_accepted)
|
||
|
||
profile.save(update_fields=["acm_problems_status", "accepted_number"])
|
||
fixed_count += 1
|
||
|
||
if dry_run:
|
||
self.stdout.write(self.style.WARNING("dry-run 完成,未写入任何数据"))
|
||
else:
|
||
self.stdout.write(self.style.SUCCESS(f"完成,共修复 {fixed_count} 个用户 Profile"))
|