mamweb/vysledkovky/utils.py

490 lines
16 KiB
Python
Raw Normal View History

2022-10-01 21:47:15 +02:00
import abc
from functools import cached_property
2023-01-02 21:40:07 +01:00
from typing import Union, Iterable # TODO: s pythonem 3.10 přepsat na '|'
2022-10-01 21:47:15 +02:00
import seminar.models as m
2022-10-01 21:47:15 +02:00
from django.db.models import Q, Sum
from seminar.utils import resi_v_rocniku
ROCNIK_ZRUSENI_TEMAT = 25
2021-09-06 02:26:05 +02:00
class FixedIterator:
def next(self):
return self.niter.__next__()
def __init__(self, niter):
self.niter = niter
2022-10-01 21:47:15 +02:00
def body_resitelu(
2022-10-10 22:30:33 +02:00
za: Union[m.Cislo, m.Rocnik, None] = None,
2022-10-01 21:47:15 +02:00
do: m.Deadline = None,
od: m.Deadline = None,
jen_verejne: bool = True,
2023-01-02 21:40:07 +01:00
resitele: Iterable[m.Resitel] = None,
2022-10-10 22:33:32 +02:00
null=0 # Výchozí hodnota, pokud pro daného řešitele nejsou body
2022-10-01 21:47:15 +02:00
) -> dict[int, int]:
filtr = Q()
2022-10-01 21:47:15 +02:00
if jen_verejne:
filtr &= Q(reseni__hodnoceni__deadline_body__verejna_vysledkovka=True)
2022-10-01 21:47:15 +02:00
# Zjistíme, typ objektu v parametru "za"
if isinstance(za, m.Rocnik):
filtr &= Q(reseni__hodnoceni__deadline_body__cislo__rocnik=za)
elif isinstance(za, m.Cislo):
filtr &= Q(reseni__hodnoceni__deadline_body__cislo=za)
if do:
filtr &= Q(reseni__hodnoceni__deadline_body__deadline__lte=do.deadline)
2022-10-01 21:47:15 +02:00
if od:
filtr &= Q(reseni__hodnoceni__deadline_body__deadline__gte=od.deadline)
2022-10-01 21:47:15 +02:00
resiteleQuery = m.Resitel.objects.all()
if resitele is not None:
resitele_id = [r.id for r in resitele]
resiteleQuery = resiteleQuery.filter(id__in=resitele_id)
2022-10-01 21:47:15 +02:00
# Přidáme ke každému řešiteli údaj ".body" se součtem jejich bodů
resitele_s_body = resiteleQuery.annotate(
body=Sum('reseni__hodnoceni__body', filter=filtr))
2022-10-01 21:47:15 +02:00
# Teď jen z QuerySetu řešitelů anotovaných body vygenerujeme slovník
# indexovaný řešitelským id obsahující body.
# Pokud jsou body None, nahradíme za 0.
slovnik = {
int(res.id): (res.body if res.body else null) for res in resitele_s_body
}
return slovnik
2022-10-01 21:47:15 +02:00
class Vysledkovka(abc.ABC):
jen_verejne: bool
rocnik: m.Rocnik
do_deadlinu: m.Deadline
@property
@abc.abstractmethod
def aktivni_resitele(self) -> list[m.Resitel]:
...
@cached_property
def resitele_s_body_za_rocnik_setrizeny_seznam(self) -> list[tuple[int, int]]:
# spočítáme všem řešitelům jejich body za ročník
resitel_body_za_rocnik_slovnik = body_resitelu(
resitele=self.aktivni_resitele,
za=self.rocnik,
jen_verejne=self.jen_verejne,
do=self.do_deadlinu
)
# zeptáme se na dvojice (řešitel, body) za ročník a setřídíme sestupně
resitele_s_body_za_rocnik_setrizeny_seznam = sorted(
resitel_body_za_rocnik_slovnik.items(),
key=lambda x: x[1], reverse=True
)
return resitele_s_body_za_rocnik_setrizeny_seznam
@cached_property
def body_za_rocnik_seznamy(self) -> tuple[list[int], list[int]]:
if len(self.resitele_s_body_za_rocnik_setrizeny_seznam) == 0:
return [], []
return tuple(zip(*self.resitele_s_body_za_rocnik_setrizeny_seznam))
@property
2022-10-01 21:47:15 +02:00
def setrizeni_resitele_id(self) -> list[int]:
return self.body_za_rocnik_seznamy[0]
@property
2022-10-01 21:47:15 +02:00
def setrizene_body(self) -> list[int]:
return self.body_za_rocnik_seznamy[1]
@cached_property
def resitel_body_odjakziva_slovnik(self) -> dict[int, int]:
return body_resitelu(jen_verejne=self.jen_verejne, do=self.do_deadlinu)
@cached_property
def poradi(self):
# ze seznamu obsahujícího setřízené body spočítáme sloupec s pořadím
aktualni_poradi = 1
sloupec_s_poradim = []
# seskupíme seznam všech bodů podle hodnot
for index in range(0, len(self.setrizene_body)):
# pokud je pořadí větší než číslo řádku, tak jsme vypsali větší rozsah
# a chceme vypsat už jen prázdné místo, než dojdeme na správný řádek
if (index + 1) < aktualni_poradi:
sloupec_s_poradim.append("")
continue
velikost_skupiny = 0
# zjistíme počet po sobě jdoucích stejných hodnot
while self.setrizene_body[index] == self.setrizene_body[
index + velikost_skupiny]:
velikost_skupiny += 1
# na konci musíme ošetřit přetečení seznamu
if (index + velikost_skupiny) > len(self.setrizene_body) - 1:
break
# pokud je velikost skupiny 1, vypíšu pořadí
if velikost_skupiny == 1:
sloupec_s_poradim.append(f"{aktualni_poradi}.")
# pokud je skupina větší, vypíšu rozsah
else:
sloupec_s_poradim.append(
f"{aktualni_poradi}.{aktualni_poradi + velikost_skupiny - 1}."
)
# zvětšíme aktuální pořadí o tolik, kolik pozic bylo přeskočeno
aktualni_poradi += velikost_skupiny
return sloupec_s_poradim
class VysledkovkaRocniku(Vysledkovka):
def __init__(self, rocnik: m.Rocnik, jen_verejne: bool = True):
self.rocnik = rocnik
self.jen_verejne = jen_verejne
deadliny = m.Deadline.objects.filter(cislo__rocnik=rocnik)
if jen_verejne:
deadliny = deadliny.filter(verejna_vysledkovka=True)
self.do_deadlinu = deadliny.order_by("deadline").last()
2022-10-01 21:47:15 +02:00
@cached_property
def aktivni_resitele(self) -> list[m.Resitel]:
return list(resi_v_rocniku(self.rocnik))
@cached_property
def cisla_rocniku(self) -> list[m.Cislo]:
""" Vrátí všechna čísla daného ročníku. """
if self.jen_verejne:
return self.rocnik.verejne_vysledkovky_cisla()
else:
return self.rocnik.cisla.all().order_by('poradi')
2022-10-01 21:47:15 +02:00
@cached_property
2022-10-10 23:29:26 +02:00
def body_za_cisla_slovnik(self) -> dict[int, dict[int, int]]: # Výstup: m.Cislo.id → ( m.Resitel.id → body )
# TODO: Body jsou decimal!
2022-10-01 21:47:15 +02:00
body_cisla_slovnik = dict()
for cislo in self.cisla_rocniku:
# získáme body za číslo
body_za_cislo = body_resitelu(
za=cislo,
resitele=self.aktivni_resitele,
jen_verejne=self.jen_verejne,
null=""
)
body_cisla_slovnik[cislo.id] = body_za_cislo
return body_cisla_slovnik
class RadekVysledkovkyRocniku:
2022-10-10 23:30:07 +02:00
# TODO: přepsat na dataclass
2022-10-01 21:47:15 +02:00
""" Obsahuje věci, které se hodí vědět při konstruování výsledkovky.
Umožňuje snazší práci v templatu (lepší, než seznam)."""
def __init__(self, poradi, resitel, body_cisla_seznam, body_rocnik, body_odjakziva, rok):
self.poradi = poradi
self.resitel = resitel
self.rocnik_resitele = resitel.rocnik(rok)
self.body_rocnik = body_rocnik
self.body_celkem_odjakziva = body_odjakziva
self.body_cisla_seznam = body_cisla_seznam
self.titul = resitel.get_titul(body_odjakziva)
@cached_property
def radky_vysledkovky(self) -> list[RadekVysledkovkyRocniku]:
radky_vysledkovky = []
setrizeni_resitele_dict = dict()
for r in m.Resitel.objects.filter(
id__in=self.setrizeni_resitele_id
).select_related('osoba'):
setrizeni_resitele_dict[r.id] = r
2022-10-10 23:02:42 +02:00
for i, ar_id in enumerate(self.setrizeni_resitele_id):
2022-10-01 21:47:15 +02:00
if self.setrizene_body[i] > 0:
# seznam počtu bodů daného řešitele pro jednotlivá čísla
body_cisla_seznam = []
for cislo in self.cisla_rocniku:
body_cisla_seznam.append(self.body_za_cisla_slovnik[cislo.id][ar_id])
# Pokud řešitel dostal nějaké body
if self.resitele_s_body_za_rocnik_setrizeny_seznam[i] != 0:
# vytáhneme informace pro daného řešitele
radek = self.RadekVysledkovkyRocniku(
poradi=self.poradi[i],
resitel=setrizeni_resitele_dict[ar_id],
body_cisla_seznam=body_cisla_seznam,
body_rocnik=self.setrizene_body[i],
body_odjakziva=self.resitel_body_odjakziva_slovnik[ar_id],
rok=self.rocnik) # ročník semináře pro získání ročníku řešitele
radky_vysledkovky.append(radek)
return radky_vysledkovky
class VysledkovkaCisla(Vysledkovka):
def __init__(
self,
cislo: m.Cislo,
jen_verejne: bool = True,
do_deadlinu: m.Deadline = None
):
self.cislo = cislo
self.rocnik = cislo.rocnik
self.jen_verejne = jen_verejne
if do_deadlinu is None:
do_deadlinu = m.Deadline.objects.filter(cislo=cislo).last()
self.do_deadlinu = do_deadlinu
@cached_property
def aktivni_resitele(self) -> list[m.Resitel]:
# TODO možná chytřeji vybírat aktivní řešitele
return list(resi_v_rocniku(self.rocnik))
@cached_property
def problemy(self) -> list[m.Problem]:
""" Vrátí seznam všech problémů s body v daném čísle. """
return m.Problem.objects.filter(
hodnoceni__in=m.Hodnoceni.objects.filter(deadline_body__cislo=self.cislo)
).distinct().non_polymorphic().select_related('nadproblem').select_related('nadproblem__nadproblem')
2022-10-01 21:47:15 +02:00
@cached_property
def hlavni_problemy(self) -> list[m.Problem]:
""" Vrátí seznam všech problémů, které již nemají nadproblém. """
# hlavní problémy čísla
# (mají vlastní sloupeček ve výsledkovce, nemají nadproblém)
hlavni_problemy = set()
for p in self.problemy:
hlavni_problemy.add(p.hlavni_problem) # FIXME: proč tohle nemůže obsahovat reálné instance? Ve výsledkovce by se pak zobrazovaly správné kódy…
# zunikátnění
hlavni_problemy = list(hlavni_problemy)
hlavni_problemy.sort(
key=lambda k: k.kod_v_rocniku) # setřídit podle t1, t2, c3, ...
return hlavni_problemy
2022-10-01 21:47:15 +02:00
# Není cached, protože si myslím, že queryset lze použít ve for jen jednou.
@property
def hodnoceni_do_cisla(self):
hodnoceni = m.Hodnoceni.objects.prefetch_related('reseni__resitele').select_related('problem', 'reseni')
if self.jen_verejne:
hodnoceni = hodnoceni.filter(deadline_body__verejna_vysledkovka=True)
return hodnoceni.filter(
deadline_body__cislo=self.cislo,
2022-10-05 20:14:53 +02:00
deadline_body__deadline__lte=self.do_deadlinu.deadline,
body__isnull=False,
)
2022-10-01 21:47:15 +02:00
@cached_property
2022-10-05 21:02:18 +02:00
def sectene_body(self):
"""
2022-10-11 01:36:48 +02:00
Sečte body za číslo, hlavní problémy a podproblémy.
Problém s ID '-1' znamená problémy bez nadproblémů, jež nejsou témata, tj. články, úlohy, konfery,
"""
2022-10-01 21:47:15 +02:00
2022-10-05 21:02:18 +02:00
# Body za číslo
body_za_cislo = {ar.id: "" for ar in self.aktivni_resitele}
# Body za hlavní problémy
body_za_temata = {
hp.id: {ar.id: "" for ar in self.aktivni_resitele}
for hp in self.temata_a_spol
}
2022-10-11 01:36:48 +02:00
# Ostatní body
2022-10-05 21:02:18 +02:00
body_za_temata[-1] = {ar.id: "" for ar in self.aktivni_resitele}
# Body za podproblémy
body_za_problemy = {
tema.id: {
problem.id: {ar.id: "" for ar in self.aktivni_resitele}
for problem in self.podproblemy[tema.id]
}
for tema in self.temata_a_spol
}
2022-10-11 01:36:48 +02:00
# Ostatní body
2022-10-05 21:02:18 +02:00
body_za_problemy[-1] = {
problem.id: {ar.id: "" for ar in self.aktivni_resitele}
for problem in self.podproblemy[-1]
}
# Sečteme hodnocení
for hodnoceni in self.hodnoceni_do_cisla:
prob = hodnoceni.problem.get_real_instance()
2022-10-05 21:08:43 +02:00
nadproblem = prob.hlavni_problem.id
2022-10-01 21:47:15 +02:00
2022-10-05 21:14:27 +02:00
# Když nadproblém není "téma", pak je "Ostatní"
if nadproblem not in body_za_temata:
nadproblem = -1
2022-10-05 21:02:18 +02:00
problem_slovnik = body_za_problemy[nadproblem][prob.id]
nadproblem_slovnik = body_za_temata[nadproblem]
2022-10-01 21:47:15 +02:00
2022-10-05 21:02:18 +02:00
body = hodnoceni.body
2022-10-05 21:02:18 +02:00
# Může mít více řešitelů
2022-10-01 21:47:15 +02:00
for resitel in hodnoceni.reseni.resitele.all():
if resitel not in self.aktivni_resitele:
continue
2022-10-05 21:02:18 +02:00
self.pricti_body(body_za_cislo, resitel, body)
2022-10-01 21:47:15 +02:00
self.pricti_body(nadproblem_slovnik, resitel, body)
self.pricti_body(problem_slovnik, resitel, body)
2022-10-05 21:02:18 +02:00
return body_za_cislo, body_za_temata, body_za_problemy
2022-10-01 21:47:15 +02:00
@cached_property
def body_za_temata(self) -> dict[int, dict[int, str]]:
2022-10-05 21:02:18 +02:00
return self.sectene_body[1]
2022-10-01 21:47:15 +02:00
@cached_property
def body_za_cislo(self) -> dict[int, str]:
2022-10-05 21:02:18 +02:00
return self.sectene_body[0]
2022-10-01 21:47:15 +02:00
@cached_property
def problemy_slovnik(self):
2022-10-05 21:02:18 +02:00
return self.sectene_body[2]
2022-10-01 21:47:15 +02:00
@cached_property
def temata_a_spol(self) -> list[m.Problem]:
2022-10-02 20:53:08 +02:00
if self.rocnik.rocnik < ROCNIK_ZRUSENI_TEMAT:
2022-10-01 21:47:15 +02:00
return self.hlavni_problemy
else:
return list(filter(self.ne_clanek_ne_konfera, self.hlavni_problemy))
@cached_property
def je_nejake_ostatni(self):
return len(self.hlavni_problemy) - len(self.temata_a_spol) > 0
@cached_property
def podproblemy(self) -> dict[int, list[m.Problem]]:
podproblemy = {hp.id: [] for hp in self.temata_a_spol}
temata_a_spol = set(self.temata_a_spol)
podproblemy[-1] = []
for problem in self.problemy:
h_problem = problem.hlavni_problem
if h_problem in temata_a_spol:
podproblemy[h_problem.id].append(problem.get_real_instance())
else:
podproblemy[-1].append(problem.get_real_instance())
for podproblem in podproblemy.keys():
podproblemy[podproblem] = sorted(podproblemy[podproblem], key=lambda p: p.kod_v_rocniku)
return podproblemy
2022-10-01 21:47:15 +02:00
@cached_property
def podproblemy_seznam(self) -> list[list[m.Problem]]:
return [self.podproblemy[it.id] for it in self.temata_a_spol] + [self.podproblemy[-1]]
@cached_property
def podproblemy_iter(self) -> FixedIterator:
return FixedIterator(self.podproblemy_seznam.__iter__())
class RadekVysledkovkyCisla(object):
2022-10-10 23:30:07 +02:00
# TODO: Přepsat na dataclass
2022-10-01 21:47:15 +02:00
"""Obsahuje věci, které se hodí vědět při konstruování výsledkovky.
Umožňuje snazší práci v templatu (lepší, než seznam)."""
def __init__(self, poradi, resitel, temata_seznamk, body_cislo, body_rocnik, body_odjakziva, rok, body_podproblemy, body_podproblemy_iter):
2022-10-01 21:47:15 +02:00
self.resitel = resitel
self.rocnik_resitele = resitel.rocnik(rok)
self.body_cislo = body_cislo
self.body_rocnik = body_rocnik
self.body_celkem_odjakziva = body_odjakziva
self.poradi = poradi
self.body_za_temata_seznam = temata_seznamk
2022-10-01 21:47:15 +02:00
self.titul = resitel.get_titul(body_odjakziva)
self.body_podproblemy = body_podproblemy
self.body_podproblemy_iter = body_podproblemy_iter
@cached_property
def radky_vysledkovky(self) -> list[RadekVysledkovkyCisla]:
# vytvoříme jednotlivé sloupce výsledkovky
radky_vysledkovky = []
setrizeni_resitele_slovnik = {}
setrizeni_resitele = m.Resitel.objects.filter(id__in=self.setrizeni_resitele_id).select_related('osoba')
for r in setrizeni_resitele:
setrizeni_resitele_slovnik[r.id] = r
2022-10-10 23:24:32 +02:00
for i, ar_id in enumerate(self.setrizeni_resitele_id):
2022-10-01 21:47:15 +02:00
if self.setrizene_body[i] > 0:
# získáme seznam bodů za problémy pro daného řešitele
body_problemy = []
body_podproblemy = []
for hp in self.temata_a_spol:
body_problemy.append(self.body_za_temata[hp.id][ar_id])
2022-10-01 21:47:15 +02:00
body_podproblemy.append([
self.problemy_slovnik[hp.id][it.id][ar_id]
for it in self.podproblemy[hp.id]
])
if self.je_nejake_ostatni:
body_problemy.append(self.body_za_temata[-1][ar_id])
2022-10-01 21:47:15 +02:00
body_podproblemy.append(
[self.problemy_slovnik[-1][it.id][ar_id] for it in self.podproblemy[-1]])
# vytáhneme informace pro daného řešitele
radek = self.RadekVysledkovkyCisla(
poradi=self.poradi[i],
resitel=setrizeni_resitele_slovnik[ar_id],
temata_seznamk=body_problemy,
2022-10-01 21:47:15 +02:00
body_cislo=self.body_za_cislo[ar_id],
body_rocnik=self.setrizene_body[i],
body_odjakziva=self.resitel_body_odjakziva_slovnik[ar_id],
rok=self.rocnik,
body_podproblemy=body_podproblemy, # body všech podproblémů
body_podproblemy_iter=FixedIterator(body_podproblemy.__iter__())
) # ročník semináře pro zjištění ročníku řešitele
radky_vysledkovky.append(radek)
return radky_vysledkovky
@staticmethod
def pricti_body(slovnik, resitel, body):
""" Přiřazuje danému řešiteli body do slovníku. """
# testujeme na None (""), pokud je to první řešení
# daného řešitele, předěláme na 0
# (v dalším kroku přičteme reálný počet bodů),
# rozlišujeme tím mezi 0 a neodevzdaným řešením
if slovnik[resitel.id] == "":
slovnik[resitel.id] = 0
slovnik[resitel.id] += body
@staticmethod
def ne_clanek_ne_konfera(problem):
2022-10-01 21:47:15 +02:00
inst = problem.get_real_instance()
return not (isinstance(inst, m.Clanek) or isinstance(inst, m.Konfera))
class VysledkovkaDoTeXu(VysledkovkaCisla):
def __init__(
self,
nejake_cislo: m.Cislo,
od_vyjma: m.Deadline,
do_vcetne: m.Deadline
):
super().__init__(nejake_cislo, False, do_vcetne)
self.od_deadlinu = od_vyjma
@cached_property
def problemy(self) -> list[m.Problem]:
return m.Problem.objects.filter(hodnoceni__in=m.Hodnoceni.objects.filter(
deadline_body__deadline__gt=self.od_deadlinu.deadline,
deadline_body__deadline__lte=self.do_deadlinu.deadline,
)).distinct().non_polymorphic().select_related('nadproblem').select_related('nadproblem__nadproblem')
@property
def hodnoceni_do_cisla(self):
hodnoceni = m.Hodnoceni.objects.prefetch_related(
'problem', 'reseni', 'reseni__resitele')
if self.jen_verejne:
hodnoceni = hodnoceni.filter(deadline_body__verejna_vysledkovka=True)
return hodnoceni.filter(
deadline_body__deadline__gt=self.od_deadlinu.deadline,
2022-10-05 20:14:53 +02:00
deadline_body__deadline__lte=self.do_deadlinu.deadline,
body__isnull=False,
2022-10-10 22:30:33 +02:00
)