|
@ -2,22 +2,26 @@ |
|
|
|
|
|
|
|
|
import datetime |
|
|
import datetime |
|
|
from django.contrib.auth.decorators import user_passes_test |
|
|
from django.contrib.auth.decorators import user_passes_test |
|
|
from html.parser import HTMLParser |
|
|
from html.parser import HTMLParser |
|
|
from django.contrib.contenttypes.models import ContentType |
|
|
from django.contrib.contenttypes.models import ContentType |
|
|
|
|
|
from django.core.exceptions import ObjectDoesNotExist |
|
|
|
|
|
|
|
|
import seminar.models as m |
|
|
import seminar.models as m |
|
|
import seminar.treelib as t |
|
|
import seminar.treelib as t |
|
|
|
|
|
|
|
|
staff_member_required = user_passes_test(lambda u: u.is_staff) |
|
|
staff_member_required = user_passes_test(lambda u: u.is_staff) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class FirstTagParser(HTMLParser): |
|
|
class FirstTagParser(HTMLParser): |
|
|
def __init__(self, *args, **kwargs): |
|
|
def __init__(self, *args, **kwargs): |
|
|
self.firstTag = None |
|
|
self.firstTag = None |
|
|
super().__init__(*args, **kwargs) |
|
|
super().__init__(*args, **kwargs) |
|
|
|
|
|
|
|
|
def handle_data(self, data): |
|
|
def handle_data(self, data): |
|
|
if self.firstTag == None: |
|
|
if self.firstTag == None: |
|
|
self.firstTag = data |
|
|
self.firstTag = data |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def histogram(seznam): |
|
|
def histogram(seznam): |
|
|
d = {} |
|
|
d = {} |
|
|
for i in seznam: |
|
|
for i in seznam: |
|
@ -27,8 +31,9 @@ def histogram(seznam): |
|
|
return d |
|
|
return d |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
roman_numerals = zip((1000, 900, 500, 400, 100, 90, 50, 40, 10, 9, 5, 4, 1), |
|
|
roman_numerals = zip((1000, 900, 500, 400, 100, 90, 50, 40, 10, 9, 5, 4, 1), |
|
|
('M', 'CM', 'D', 'CD','C', 'XC','L','XL','X','IX','V','IV','I')) |
|
|
('M', 'CM', 'D', 'CD', 'C', 'XC', 'L', 'XL', 'X', 'IX', 'V', 'IV', 'I')) |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def roman(num): |
|
|
def roman(num): |
|
|
res = "" |
|
|
res = "" |
|
@ -37,6 +42,7 @@ def roman(num): |
|
|
num %= i |
|
|
num %= i |
|
|
return res |
|
|
return res |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def from_roman(rom): |
|
|
def from_roman(rom): |
|
|
if not rom: |
|
|
if not rom: |
|
|
return 0 |
|
|
return 0 |
|
@ -60,9 +66,9 @@ def seznam_problemu(): |
|
|
except: |
|
|
except: |
|
|
url = None |
|
|
url = None |
|
|
if url: |
|
|
if url: |
|
|
s += '<a href="%s">%s</a>, ' % (url, o.pk, ) |
|
|
s += '<a href="%s">%s</a>, ' % (url, o.pk,) |
|
|
else: |
|
|
else: |
|
|
s += '%s, ' % (o.pk, ) |
|
|
s += '%s, ' % (o.pk,) |
|
|
s = s[:-2] + ']' |
|
|
s = s[:-2] + ']' |
|
|
problemy.append(s) |
|
|
problemy.append(s) |
|
|
|
|
|
|
|
@ -75,7 +81,7 @@ def seznam_problemu(): |
|
|
jmena[j].append(r) |
|
|
jmena[j].append(r) |
|
|
for j in jmena: |
|
|
for j in jmena: |
|
|
if len(jmena[j]) > 1: |
|
|
if len(jmena[j]) > 1: |
|
|
prb(m.Resitel, 'Duplicitní jméno "%s"' % (j, ), jmena[j]) |
|
|
prb(m.Resitel, 'Duplicitní jméno "%s"' % (j,), jmena[j]) |
|
|
|
|
|
|
|
|
# Data maturity a narození |
|
|
# Data maturity a narození |
|
|
for r in m.Resitel.objects.all(): |
|
|
for r in m.Resitel.objects.all(): |
|
@ -83,13 +89,14 @@ def seznam_problemu(): |
|
|
prb(m.Resitel, 'Neznámý rok maturity', [r]) |
|
|
prb(m.Resitel, 'Neznámý rok maturity', [r]) |
|
|
if r.rok_maturity and (r.rok_maturity < 1990 or r.rok_maturity > datetime.date.today().year + 10): |
|
|
if r.rok_maturity and (r.rok_maturity < 1990 or r.rok_maturity > datetime.date.today().year + 10): |
|
|
prb(m.Resitel, 'Podezřelé datum maturity', [r]) |
|
|
prb(m.Resitel, 'Podezřelé datum maturity', [r]) |
|
|
if r.osoba.datum_narozeni and (r.osoba.datum_narozeni.year < 1970 or r.osoba.datum_narozeni.year > datetime.date.today().year - 12): |
|
|
if r.osoba.datum_narozeni and ( |
|
|
|
|
|
r.osoba.datum_narozeni.year < 1970 or r.osoba.datum_narozeni.year > datetime.date.today().year - 12): |
|
|
prb(m.Resitel, 'Podezřelé datum narození', [r]) |
|
|
prb(m.Resitel, 'Podezřelé datum narození', [r]) |
|
|
# if not r.email: |
|
|
# if not r.email: |
|
|
# prb(Resitel, u'Neznámý email', [r]) |
|
|
# prb(Resitel, u'Neznámý email', [r]) |
|
|
|
|
|
|
|
|
## Kontroly konzistence databáze a TreeNodů |
|
|
## Kontroly konzistence databáze a TreeNodů |
|
|
|
|
|
|
|
|
# Články |
|
|
# Články |
|
|
for clanek in m.Clanek.objects.all(): |
|
|
for clanek in m.Clanek.objects.all(): |
|
|
# získáme řešení svázané se článkem a z něj node ve stromě |
|
|
# získáme řešení svázané se článkem a z něj node ve stromě |
|
@ -97,7 +104,7 @@ def seznam_problemu(): |
|
|
if (reseni.count() != 1): |
|
|
if (reseni.count() != 1): |
|
|
raise ValueError("Článek k sobě má nejedno řešení!") |
|
|
raise ValueError("Článek k sobě má nejedno řešení!") |
|
|
r = reseni.first() |
|
|
r = reseni.first() |
|
|
clanek_node = r.text_cely # vazba na ReseniNode z Reseni |
|
|
clanek_node = r.text_cely # vazba na ReseniNode z Reseni |
|
|
# content type je věc pomáhající rozeznávat různé typy objektů v django-polymorphic |
|
|
# content type je věc pomáhající rozeznávat různé typy objektů v django-polymorphic |
|
|
# protože isinstance vrátí vždy jen TreeNode |
|
|
# protože isinstance vrátí vždy jen TreeNode |
|
|
# https://django-polymorphic.readthedocs.io/en/stable/migrating.html |
|
|
# https://django-polymorphic.readthedocs.io/en/stable/migrating.html |
|
@ -105,14 +112,77 @@ def seznam_problemu(): |
|
|
node = clanek_node |
|
|
node = clanek_node |
|
|
while node is not None: |
|
|
while node is not None: |
|
|
node_ct = node.polymorphic_ctype |
|
|
node_ct = node.polymorphic_ctype |
|
|
if node_ct == cislonode_ct: # dostali jsme se k CisloNode |
|
|
if node_ct == cislonode_ct: # dostali jsme se k CisloNode |
|
|
# zkontrolujeme, že stromové číslo odpovídá atributu |
|
|
# zkontrolujeme, že stromové číslo odpovídá atributu |
|
|
# .cislonode je opačná vazba k treenode_ptr, abychom z TreeNode dostali |
|
|
# .cislonode je opačná vazba k treenode_ptr, abychom z TreeNode dostali |
|
|
# CisloNode |
|
|
# CisloNode |
|
|
if clanek.cislo != node.cislonode.cislo: |
|
|
if clanek.cislo != node.cislonode.cislo: |
|
|
prb(m.Clanek, "Číslo otištění uložené u článku nesedí s " |
|
|
prb(m.Clanek, "Číslo otištění uložené u článku nesedí s " |
|
|
"číslem otištění podle struktury treenodů.", [clanek]) |
|
|
"číslem otištění podle struktury treenodů.", [clanek]) |
|
|
break |
|
|
break |
|
|
node = t.get_parent(node) |
|
|
node = t.get_parent(node) |
|
|
|
|
|
|
|
|
return problemy |
|
|
return problemy |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
### Generovani obalek |
|
|
|
|
|
def resi_v_rocniku(rocnik, cislo=None): |
|
|
|
|
|
""" Vrátí seznam řešitelů, co vyřešili nějaký problém v daném ročníku, do daného čísla. |
|
|
|
|
|
Parametry: |
|
|
|
|
|
rocnik (typu Rocnik) ročník, ze kterého chci řešitele, co něco odevzdali |
|
|
|
|
|
cislo (typu Cislo) číslo, do kterého včetně se počítá, že v daném |
|
|
|
|
|
ročníku řešitel něco poslal. |
|
|
|
|
|
Pokud není zadané, počítají se všechna řešení z daného ročníku. |
|
|
|
|
|
Výstup: |
|
|
|
|
|
QuerySet objektů typu Resitel """ |
|
|
|
|
|
|
|
|
|
|
|
if cislo is None: |
|
|
|
|
|
# filtrujeme pouze podle ročníku |
|
|
|
|
|
letosni_reseni = m.Reseni.objects.filter(hodnoceni__cislo_body__rocnik=rocnik) |
|
|
|
|
|
else: # filtrujeme podle ročníku i čísla |
|
|
|
|
|
letosni_reseni = m.Reseni.objects.filter(hodnoceni__cislo_body__rocnik=rocnik, |
|
|
|
|
|
hodnoceni__cislo_body__poradi__lte=cislo.poradi) |
|
|
|
|
|
|
|
|
|
|
|
# vygenerujeme queryset řešitelů, co letos něco poslali |
|
|
|
|
|
letosni_resitele = m.Resitel.objects.none() |
|
|
|
|
|
for reseni in letosni_reseni: |
|
|
|
|
|
letosni_resitele = letosni_resitele | reseni.resitele.filter(rok_maturity__gte=rocnik.druhy_rok()) |
|
|
|
|
|
return letosni_resitele.distinct() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def aktivniResitele(cislo, pouze_letosni=False): |
|
|
|
|
|
""" Vrací QuerySet aktivních řešitelů, což jsou ti, co ještě neodmaturovali |
|
|
|
|
|
a letos něco poslali (anebo loni něco poslali, pokud jde o první tři čísla). |
|
|
|
|
|
Parametry: |
|
|
|
|
|
cislo (typu Cislo) číslo, o které se jedná |
|
|
|
|
|
pouze_letosni jen řešitelé, kteří tento rok něco poslali |
|
|
|
|
|
|
|
|
|
|
|
""" |
|
|
|
|
|
letos = cislo.rocnik |
|
|
|
|
|
|
|
|
|
|
|
# detekujeme, zda jde o první tři čísla či nikoli (tj. zda spamovat řešitele z minulého roku) |
|
|
|
|
|
zacatek_rocniku = True |
|
|
|
|
|
try: |
|
|
|
|
|
if int(cislo.poradi) > 3: |
|
|
|
|
|
zacatek_rocniku = False |
|
|
|
|
|
except ValueError: |
|
|
|
|
|
if cislo.poradi != '7-8': |
|
|
|
|
|
raise ValueError(f'{cislo} je neplatné číslo čísla (není int a není 7-8)') |
|
|
|
|
|
zacatek_rocniku = False |
|
|
|
|
|
|
|
|
|
|
|
# nehledě na číslo chceme jen řešitele, kteří letos něco odevzdali |
|
|
|
|
|
if pouze_letosni: |
|
|
|
|
|
zacatek_rocniku = False |
|
|
|
|
|
|
|
|
|
|
|
try: |
|
|
|
|
|
loni = m.Rocnik.objects.get(rocnik=letos.rocnik - 1) |
|
|
|
|
|
except ObjectDoesNotExist: |
|
|
|
|
|
# Pro první ročník neexistuje ročník předchozí |
|
|
|
|
|
zacatek_rocniku = False |
|
|
|
|
|
|
|
|
|
|
|
if not zacatek_rocniku: |
|
|
|
|
|
return resi_v_rocniku(letos, cislo) |
|
|
|
|
|
else: |
|
|
|
|
|
# spojíme querysety s řešiteli loni a letos do daného čísla |
|
|
|
|
|
|
|
|
|
|
|
return (resi_v_rocniku(loni) | resi_v_rocniku(letos, cislo)).distinct() |
|
|