24 Commits

Author SHA1 Message Date
8fb2a22c2a another one 2025-06-13 00:32:13 +02:00
f96a3ed1f1 fix 2025-06-12 23:42:55 +02:00
00bbe05b1c feat: include club's full name in matches view 2025-06-12 22:35:04 +02:00
e373c6b274 hotfix: fixes /mecze endpoint not working (cause of a wrong API call) 2025-06-12 20:43:22 +02:00
e80ee6bf8f feat: return matches with respect to received sportsman id
also include a comment in the database model to include a relation
2025-06-12 20:36:10 +02:00
a25be482b0 matches from DB 2025-06-12 16:23:14 +02:00
4e8cb48c82 fix: safe_traverse renaming issue 2025-06-11 00:38:55 +02:00
bc82fcd6b8 chore: depend on safe_traverse from lewy_globals, add scraper cron job 2025-06-11 00:35:48 +02:00
5d238cb6b8 fix: general fixes to scraper, and db model
Co-authored-by: Noicei <kuba.baj@interia.pl>
2025-06-11 00:15:30 +02:00
fe4c9aab3d Merge branch 'ScraperV2' 2025-06-06 00:50:21 +02:00
9e8a60dfa1 chore: simplify expression, add an important todo note 2025-06-06 00:50:05 +02:00
7fe4e03945 fix: matches deafult data set to 2025 2025-06-05 22:12:48 +02:00
ffcfa12d4c static data for page 2025-06-05 22:07:29 +02:00
Pc
f7e80811a0 Updated Scraper 2025-06-05 21:41:31 +02:00
c86a93c153 fix: i forgor to add some waiting time to avoid rate limiting 2025-06-05 19:22:05 +02:00
43eb1d421d Merge branch 'ScraperTest' 2025-06-05 19:18:49 +02:00
c2568c86ef fix: scraper fixes, list matches 2025-06-05 19:18:31 +02:00
cf46fdfbb7 data for stats 2025-06-05 18:47:30 +02:00
791d20139d chore: offload some shared functionality to other methods
should also fix unexpected db behavior (hangs and disconnects)
2025-06-05 16:31:45 +02:00
c3a6626d6f fix: display commit info properly 2025-06-05 11:40:56 +02:00
5960e44b17 Merge pull request 'merge new front-end' (#3) from frontend into master
Reviewed-on: #3
2025-06-05 10:30:28 +02:00
Pc
13f6e2e3b9 Added rest of player updating code 2025-06-04 20:34:26 +02:00
Pc
9a007f504c Wins update 2025-06-04 19:56:32 +02:00
Pc
b5fdbb3230 Updating matches
Updating players stats(not finished)
Not tested
2025-06-04 19:50:18 +02:00
16 changed files with 12678 additions and 211 deletions

View File

@@ -1,20 +1,12 @@
from flask import session
from lewy_db import baza as ldb
from lewy_globals import colors as c
from lewy_globals import safeTraverse as safe_traverse
import json
import lewy_globals
import requests
import time
def safe_traverse(obj: dict, path: list, default=None):
result = obj
try:
for x in path:
result = result[x]
except KeyError:
result = default
# print(f"error reading: {' -> '.join(path)} - returning: {default}")
finally:
return result
from sqlalchemy import func
class scraper:
@@ -45,22 +37,26 @@ class scraper:
else:
return False
def to_iso_compatible_date(self, dmy_date: str):
"""
Zamienia datę z formatu DD.MM.YY na YYYY-MM-DD
:param dmy_date: Data w formacie DD.MM.YY
:type dmy_date: str
"""
day, month, year = dmy_date.split(".")
return f"{2000 + int(year)}-{month}-{day}"
def czy_mecz_istnieje(self, zewnetrzne_id_meczu: str):
# mecz = db.simple_select_all(ldb.mecze, zewnetrzne_id_meczu=zewnetrzne_id_meczu)
# if mecz is not None and mecz != []:
# return True
# else:
# return False
return self.__czy_x_istnieje("mecze", zewnetrzne_id_meczu=zewnetrzne_id_meczu)
def czy_klub_istnieje(self, id_klubu: str):
# mecz = db.simple_select_all(ldb.mecze, zewnetrzne_id_meczu=zewnetrzne_id_meczu)
# if mecz is not None and mecz != []:
# return True
# else:
# return False
return self.__czy_x_istnieje("kluby", id_klubu=id_klubu)
def czy_nie_trzeba_pobierac_nowych_meczy_zawodnika(self, id_zawodnika: int, zewnetrzne_id_meczu: str):
return self.__czy_x_istnieje("sportowcy_w_meczach", id_zawodnika=id_zawodnika, zewnetrzne_id_meczu=zewnetrzne_id_meczu)
def id_na_imie_nazwisko_urodziny(self, zewnetrzne_id_sportowca: str = "MVC8zHZD"):
"""
Scraper z dykty xD
@@ -96,6 +92,10 @@ class scraper:
# TODO: Sprawdź, czy sportowiec istnieje w bazie.
# Jeśli nie, dodaj go w podobny sposób, jak
# w sample_data_init() (w lewy_db.py).
# -- Jednak tego tak nie robimy, ponieważ nie uzyskamy wielu informacji.
id_zawodnika = self.db.get_id_zawodnika_by_zewnetrzne_id(zewnetrzne_id_sportowca)
zawodnik = self.db.simple_select_all("sportowcy", zewnetrzne_id_zawodnika=zewnetrzne_id_sportowca)[0]
page = 0
match_num = 0
@@ -115,11 +115,29 @@ class scraper:
home_club_id = safe_traverse(match, ["homeParticipantUrl"], default="non-existent-club-id")
away_club_id = safe_traverse(match, ["awayParticipantUrl"], default="non-existent-club-id")
# Sprawdź, czy mecz nie znajduje się już w bazie
if self.czy_mecz_istnieje(zewnetrzne_id_meczu=match_id):
# ~Sprawdź, czy mecz nie znajduje się już w bazie~
#
# UWAGA! Nie powinniśmy tego sprawdzać jak w komentarzu poniżej!
# To sprawdzenie powinno jedynie służyć zapobieganiu dodania istniejących meczy,
# natomiast istniejący mecz nie oznacza, że sportowiec ma już statystykę z niego!
# Przerwać scrapowanie należy wtedy, gdy znajdzie się statystykę sportowca
# z bieżącego meczu, a nie kiedy znajdzie się bieżący mecz w bazie!
#
# if self.czy_mecz_istnieje(zewnetrzne_id_meczu=match_id):
# stop_scraping = True
# break
#
# Rozwiązanie jest mocno nieefektywne przy scrapowaniu całej bazy od zera,
# ale rozwiąże przypadki, w których zawodnicy, których śledzimy, grali przeciwko sobie.
if self.czy_nie_trzeba_pobierac_nowych_meczy_zawodnika(id_zawodnika=id_zawodnika, zewnetrzne_id_meczu=match_id):
stop_scraping = True
break
if self.czy_mecz_istnieje(zewnetrzne_id_meczu=match_id):
# Nie scrapuj istniejących meczy.
# Naturalnie, istniejący mecz nie musi oznaczać potrzeby zakończenia scrapowania.
continue
# Sprawdź, czy klub znajduje się już w bazie. Jeśli nie,
# trzeba go dodać przed meczem.
if not self.czy_klub_istnieje(id_klubu=home_club_id):
@@ -135,14 +153,91 @@ class scraper:
pelna_nazwa=safe_traverse(match, ["awayParticipantName"]),
skrocona_nazwa=safe_traverse(match, ["awayParticipant3CharName"]))
# TODO: (opcjonalnie) zamień *słownik match* na *obiekt mecz*
# TODO: dodaj obiekt mecz do bazy (simple_insert_one(), simple_insert_many())
print(f"{c.OKCYAN}Nowy mecz ({match_num}){c.ENDC}: {match}")
iso_converted_date = self.to_iso_compatible_date(safe_traverse(match, ["eventStartTime"], default="1970-01-01"))
self.db.simple_insert_one("mecze",
zewnetrzne_id_meczu = safe_traverse(match, ["eventEncodedId"], default=""),
data = iso_converted_date,
gospodarze_id = home_club_id,
gospodarze = self.db.simple_select_all("kluby", id_klubu=home_club_id)[0],
goscie_id = away_club_id,
goscie = self.db.simple_select_all("kluby", id_klubu=away_club_id)[0],
gosp_wynik = safe_traverse(match, ["homeScore"], default=0),
gosc_wynik = safe_traverse(match, ["awayScore"], default=0),
sezon = safe_traverse(match, ["tournamentSeason"], default=""),
nazwa_turnieju = safe_traverse(match, ["tournamentTitle"], default=""),
skrocona_nazwa_turnieju = safe_traverse(match, ["tournamentTemplateShortCode"], default=""),
flaga = safe_traverse(match, ["flagId"], default=0),
)
match_num += 1
stats = safe_traverse(match, ["stats"], default="")
zewnetrzne_id_meczu = safe_traverse(match, ["eventEncodedId"], default="")
if stats != False: # gdy sportowiec był aktywny w meczu
# print("todo :)")
self.db.simple_insert_one("sportowcy_w_meczach",
id_zawodnika = id_zawodnika,
zawodnik = zawodnik,
zewnetrzne_id_meczu = zewnetrzne_id_meczu,
# Uwaga! Czasami przygłupy z flashscore zwracają puste pole '' zamiast zera, np. do liczby strzelonych goli.
# Dlatego int("0" + "") = int("0"), co zapobiegnie wysypaniu się przy int("").
czas_gry = int("0" + safe_traverse(stats, ["595", "value"], default="0").rstrip("'?")),
goli = int("0" + safe_traverse(stats, ["596", "value"], default="0")),
asyst = int("0" + safe_traverse(stats, ["541", "value"], default="0")),
interwencje_bramkarza = 0,
suma_interwencji_na_bramke = 0,
zolte_kartki = int("0" + safe_traverse(stats, ["599", "value"], default="0")),
czerwone_kartki = int("0" + safe_traverse(stats, ["600", "value"], default="0")),
wygrana = {"Z": 1, "R": 0, "P": -1}.get(safe_traverse(match, ["winLoseShort"], default=""), 0),
wynik = safe_traverse(match, ["rating"], default=0) or 0
)
# # analogicznie zinkrementuj statystyki_sportowcow:
# # aby to zrobić, najpierw pobierz najnowszą statystykę sportowca
# # ...
#
#
# # a następnie użyj funkcji w lewy_db do inkrementowania danych
# # (póki co jeszcze takiej nie ma)
# self.db.simple_increment_data("statystyki_sportowcow",
# id = ... # pewnie id się przyda
# sportowiec = zawodnik,
# ostatni_mecz = self.db.get_id_meczu_by_zewnetrzne_id(zewnetrzne_id_meczu),
# ilosc_wystapien = 1 if int(safe_traverse(stats, ["595", "value"], default="0").rstrip("'")) > 0 else 0,
# minut_gry = int(safe_traverse(stats, ["595", "value"], default="0").rstrip("'")),
# gier_sum = 1 if int(safe_traverse(stats, ["595", "value"], default="0").rstrip("'")) > 0 else 0,
# goli_sum = int(safe_traverse(stats, ["596", "value"], default="0")),
# asyst_sum = int(safe_traverse(stats, ["541", "value"], default="0")),
# interwencji_sum = 0,
# nieobronionych_interwencji_sum = 0,
# zoltych_kartek_sum = int(safe_traverse(stats, ["599", "value"], default="0")),
# czerwonych_kartek_sum = int(safe_traverse(stats, ["600", "value"], default="0")),
# wygranych_sum = 1 if safe_traverse(match, ["winloseshort"], default="") == "z" else 0,
# wynik_sum = safe_traverse(match, ["rating"], default=0),
# meczow_do_wynikow_sum = 1 if safe_traverse(match, ["rating"], default=0) not in (0, none) else none
# )
else:
# print("też todo :)")
# # TODO: TU TEŻ TRZEBA POPRAWIĆ ANALOGICZNIE DO TEGO, CO JEST WEWNĄTRZ IF'A
self.db.simple_insert_one("sportowcy_w_meczach",
id_zawodnika = id_zawodnika,
zawodnik = zawodnik,
zewnetrzne_id_meczu = zewnetrzne_id_meczu,
czas_gry = 0, #(lambda v: int(str(v).rstrip("'")) if isinstance(v, (str, int, float)) and str(v).rstrip("'").lstrip("-").isdigit() else 0)(safe_traverse(stats if isinstance(stats, dict) else {}, ["595", "value"], default="0")),
goli = int("0" + safe_traverse(stats, ["596", "value"], default="0")),
asyst = int("0" + safe_traverse(stats, ["541", "value"], default="0")),
interwencje_bramkarza = 0,
suma_interwencji_na_bramke = 0,
zolte_kartki = int("0" + safe_traverse(stats, ["599", "value"], default="0")),
czerwone_kartki = int("0" + safe_traverse(stats, ["600", "value"], default="0")),
wygrana = {"Z": 1, "R": 0, "P": -1}.get(safe_traverse(match, ["winLoseShort"], default=""), 0),
wynik = safe_traverse(match, ["rating"], default=0) or 0
)
# TODO: Zaktualizuj statystyki sportowca
@@ -155,7 +250,7 @@ class scraper:
# rate limita. - sherl
page += 1
time.sleep(15)
#time.sleep(5)
def aktualizuj_dane(self):
@@ -163,8 +258,12 @@ class scraper:
Pobiera mecze dla każdego sportowca wymienionego
w pliku konfiguracyjnym.
"""
start_time = time.time()
for id_sportowca in lewy_globals.config['sportsmen']['tracked_ids']:
self.aktualizuj_dane_sportowca(zewnetrzne_id_sportowca=id_sportowca)
time.sleep(15)
time.sleep(5)
end_time = time.time()
print(f"Scrapowanie trwało {end_time - start_time}s.")

View File

@@ -1,7 +1,7 @@
from argparse import ArgumentParser
from flask import Flask, Response, render_template
from flask_apscheduler import APScheduler
from fs_scraper import scraper as scr
from fs_scraper import scraper
from lewy_globals import colors as c
import lewy_api
import lewy_db
@@ -13,7 +13,7 @@ import time
app = Flask(__name__)
app_host = "None"
app_port = "None"
scrape = None
scr = None
def setup():
# sanity check: make sure config is set
@@ -58,6 +58,8 @@ def setup():
sanity_string += f" If you're running a reverse proxy, set {c.OKCYAN}is_proxied{c.ENDC} to true to silence this message.\n"
print(sanity_string)
# Should fix disconnects: https://stackoverflow.com/a/61739721
app.config['SQLALCHEMY_ENGINE_OPTIONS'] = {"pool_pre_ping": True}
app.config['SQLALCHEMY_DATABASE_URI'] = f"{config['general']['db_path_url']}"
app.config['SQLALCHEMY_TRACK_MODIFICATIONS'] = False
@@ -77,7 +79,7 @@ def setup():
app.add_url_rule('/api/<path:received_request>', view_func=lewy_api.api_global_catchall)
db = lewy_globals.setupDb(app, config)
scraper = scr()
scr = scraper()
with app.app_context():
db.create_all()
@@ -95,8 +97,7 @@ def every5seconds():
def every2hours():
# zaktualizuj bazę danych scrapując FS
# ...
# scraper.aktualizuj_dane()
scr.aktualizuj_dane()
return
@app.route('/<string:val>', methods=['GET'])

View File

@@ -50,6 +50,16 @@ def not_implemented(data):
# TODO: change list to string -> data, not data[0]
return 501, f"not recognised/implemented: {data[0]}", []
def __czy_x_istnieje(typ, **id):
rekord = getDb().simple_select_all(typ, **id)
if rekord is not None and rekord != []:
return True
else:
return False
def czy_sportowiec_istnieje(id_zawodnika: str):
return __czy_x_istnieje("sportowcy", id_zawodnika=id_zawodnika)
# GET /api/v1
def stub_hello():
"""
@@ -85,11 +95,37 @@ def stats():
return 200, "ok", data_to_send
# GET /api/v1/matches
def get_matches(r):
def get_matches(r = None, id_zawodnika: str | None = None):
"""
TODO: Zwraca mecze.
Zwraca mecze.
Przykład wywołania:
get_matches(r, id_zawodnika=1), tożsame z GET /api/v1/matches?id_zawodnika=1
get_matches(r), tożsame z GET /api/v1/matches
"""
pass
response_json = []
mecze = None
if id_zawodnika is None:
# Gdy nie podano id wprost, sprawdź, czy podano je przez parametr.
id_zawodnika = r.args.get('id_zawodnika', -1)
# Sprawdź, czy podano jakiekolwiek ID sportowca. Jeżeli nie, wypisz wszystkie mecze.
if id_zawodnika == -1:
mecze = getDb().simple_select_all("mecze")
# Sprawdź, czy sportowiec o podanym (lub niepodanym) id istnieje.
# Jeśli nie istnieje, wypisz wszystkie mecze.
elif not czy_sportowiec_istnieje(id_zawodnika=id_zawodnika):
return 404, "error", {"error_msg": "This sportsman has not been found in the database. Try: id_zawodnika=1"}
# Gdy sportowiec istnieje, wypisz jego mecze.
else:
mecze = getDb().get_sportsman_matches(id_zawodnika=id_zawodnika)
for mecz in mecze:
response_json.append(mecz.jsonify())
# print(f"zwracam mecze: {response_json}")
return 200, "ok", response_json
# GET /api/v1/debugger_halt?token=XXX...
@require_authentication
@@ -142,7 +178,7 @@ def lookup(data, request):
case 'halt':
return debugger_halt(r = request)
case 'matches':
get_matches(r = request)
return get_matches(r = request)
case _:
increment_bad_requests()
return not_implemented(data)

View File

@@ -36,6 +36,9 @@ class baza():
self.db = self.initDB(self.app, config)
self.refresh_session()
def __repr__(self):
return f"<Obiekt bazodanowy baza ({len(self.entities)} encji)>"
def initDB(self, app, config):
global sportowcy, trofea, sportowcy_w_meczach, statystyki_sportowcow, kluby, mecze
tnp = config['general']['db_prefix'] + "_lewangoalski_"
@@ -67,6 +70,7 @@ class baza():
statystyki_id: Mapped[ List[int]] = mapped_column(ForeignKey(f"{tnp}statystyki_sportowcow.id_statystyki"), nullable=True)
statystyki: Mapped[List["statystyki_sportowcow"]] = relationship(back_populates="sportowiec")
trofea: Mapped[ List["trofea"]] = relationship(back_populates="zawodnik", foreign_keys="[trofea.id_zawodnika]")
mecze_zawodnika: Mapped[ List["sportowcy_w_meczach"]] = relationship(back_populates="zawodnik")
def __repr__(self):
return f"<Sportowiec #{self.id_zawodnika} ({self.imie} {self.nazwisko})>"
@@ -89,8 +93,9 @@ class baza():
__tablename__ = tnp + "sportowcy_w_meczach"
id_rekordu: Mapped[ int] = mapped_column(primary_key=True, autoincrement=True)
id_zawodnika: Mapped[ int] = mapped_column(ForeignKey(f"{tnp}sportowcy.id_zawodnika"))
zawodnik: Mapped[ "sportowcy"] = relationship()
zawodnik: Mapped[ "sportowcy"] = relationship(back_populates="mecze_zawodnika")
zewnetrzne_id_meczu: Mapped[ str] = mapped_column(ForeignKey(f"{tnp}mecze.zewnetrzne_id_meczu"))
mecz: Mapped[ "mecze"] = relationship() # back_populates="zawodnicy_w_meczu", foreign_keys=[zewnetrzne_id_meczu])
czas_gry: Mapped[ int] = mapped_column()
goli: Mapped[ int] = mapped_column()
asyst: Mapped[ int] = mapped_column()
@@ -102,7 +107,7 @@ class baza():
wynik: Mapped[ float] = mapped_column()
def __repr__(self):
return f"<Sportowiec #{self.id_zawodnika} ({self.imie} {self.nazwisko})>"
return f"<{self.zawodnik.imie} {self.zawodnik.nazwisko} w meczu {self.mecz.gospodarze.skrocona_nazwa} vs. {self.mecz.goscie.skrocona_nazwa}>"
class statystyki_sportowcow(Base):
__tablename__ = tnp + "statystyki_sportowcow"
@@ -119,6 +124,8 @@ class baza():
zoltych_kartek_sum: Mapped[ int] = mapped_column()
czerwonych_kartek_sum: Mapped[ int] = mapped_column()
wygranych_sum: Mapped[ int] = mapped_column()
przegranych_sum: Mapped[ int] = mapped_column()
remisow_sum: Mapped[ int] = mapped_column()
wynik_sum: Mapped[ int] = mapped_column()
meczow_do_wynikow_sum: Mapped[ int] = mapped_column()
@@ -152,9 +159,30 @@ class baza():
skrocona_nazwa_turnieju: Mapped[ str] = mapped_column()
flaga: Mapped[ int] = mapped_column()
# zawodnicy_w_meczu: Mapped[ List["sportowcy_w_meczach"]] = relationship(back_populates="mecz")
def __repr__(self):
return f"<Mecz #{self.id_meczu} ({self.zewnetrzne_id_meczu}, {self.gospodarze.skrocona_nazwa} vs. {self.goscie.skrocona_nazwa})>"
def jsonify(self):
return {
"id_meczu": self.id_meczu,
"zewnetrzne_id_meczu": self.zewnetrzne_id_meczu,
"data": self.data.strftime("%Y-%m-%d"),
"gospodarze_id": self.gospodarze_id,
"gospodarze": self.gospodarze.skrocona_nazwa,
"gospodarze_pelna_nazwa": self.gospodarze.pelna_nazwa,
"goscie_id": self.goscie_id,
"goscie": self.goscie.skrocona_nazwa,
"goscie_pelna_nazwa": self.goscie.pelna_nazwa,
"gosp_wynik": self.gosp_wynik,
"gosc_wynik": self.gosc_wynik,
"sezon": self.sezon,
"nazwa_turnieju": self.nazwa_turnieju,
"skrocona_nazwa_turnieju": self.skrocona_nazwa_turnieju,
"flaga": self.flaga
}
self.entities = {
'sportowcy': sportowcy,
'trofea': trofea,
@@ -201,8 +229,10 @@ class baza():
:param string: Zapis tekstowy
:type string: str
"""
table_str = string[:string.find('.')]
column_str = string[string.find('.') + 1:]
try:
table_str, column_str = string.split('.')
except:
raise ValueError("Nieprawidłowe dane - podaj zarówno tabelę, jak i kolumnę, np.: \"kluby.id_klubu\".")
if hasattr(self.entities[table_str], column_str):
return getattr(self.entities[table_str], column_str)
return None
@@ -222,27 +252,60 @@ class baza():
if not isinstance(entity_type, str):
entity_type = entity_type.__name__
query_params, special_args = self.extract_special_args(kwargs)
print(f"[{round(time.time())}] SELECT {entity_type}")
results = (
self.session.
query(self.entities[entity_type]).
filter_by(**query_params)
)
# Obsługuje "ORDER_BY", "LIMIT", itp. itd.
results = self.manipulate_results(results, special_args)
results_objs = results.all()
print(f"[{round(time.time())}] SELECT RESULTS: {results_objs}")
return results_objs
def extract_special_args(self, dictionary: dict):
"""
Zwraca krotkę składającą się ze słowników z:
- parametrami wyszukiwania
- specjalnymi argumentami
:param dictionary: Słownik wejściowy
:type dictionary: dict
"""
# Save special arguments received with kwargs,
# that are meant for SQL operations to special_args,
# and delete from the rest, that will be passed
# directly to filter_by().
# They will not be passed as search query, but serve
# as an additional search parameter.
dictionary = dictionary.copy()
special_keywords = ("ORDER_BY", "ORDER_BY_DESC", "LIMIT")
special_args = {}
for arg in special_keywords:
if arg in kwargs:
special_args[arg] = kwargs[arg]
del kwargs[arg]
if arg in dictionary:
special_args[arg] = dictionary[arg]
del dictionary[arg]
print(f"[{round(time.time())}] SELECT")
return dictionary, special_args
results = (
self.session.
query(self.entities[entity_type]).
filter_by(**kwargs)
)
def manipulate_results(self, results, special_args: dict):
"""
Wykonuje specjalne operacje na rezultatach wyszukiwania.
:param results: Wyniki wyszukiwania (ORM)
:param special_args: Specjalne operacje
:type special_args: dict
"""
if "ORDER_BY" in special_args:
column = self.str_to_column(special_args["ORDER_BY"])
@@ -257,10 +320,7 @@ class baza():
if "LIMIT" in special_args:
results = results.limit(special_args["LIMIT"])
results_objs = results.all()
print(f"[{round(time.time())}] SELECT RESULTS: {results_objs}")
return results_objs
return results
@exit_gracefully
def simple_insert_one(self, entity_type, **kwargs):
@@ -283,7 +343,7 @@ class baza():
f"Aby naprawić dodawanie z autoinkrementującym kluczem zobacz {c.WARNING}https://stackoverflow.com/a/8745101{c.ENDC}\n"
f"Zostałeś ostrzeżony!")
print(f"[{round(time.time())}] INSERT")
print(f"[{round(time.time())}] INSERT {entity_type}")
obj = self.entities[entity_type](**kwargs)
#with Session(self.db.engine) as session:
@@ -307,6 +367,183 @@ class baza():
return 0
#return 1
@exit_gracefully
def increment_fields(self, entity_type, record_id, **increments):
"""
Dodaje wartości do istniejących pól (np. goli_sum += 2).
Użycie:
increment_fields(ldb.statystyki_sportowcow, 123, goli_sum=2, asyst_sum=1)
"""
if not isinstance(entity_type, str):
entity_type = entity_type.__name__
if entity_type not in self.entities:
print(f"Nieznany typ encji: {entity_type}")
return -1
model_class = self.entities[entity_type]
print(f"[{round(time.time())}] INCREMENT {entity_type}")
record = self.session.get(model_class, record_id)
if not record:
print(f"Rekord z ID {record_id} nie istnieje w tabeli {entity_type}")
return -1
for key, increment_value in increments.items():
if hasattr(record, key):
current_value = getattr(record, key, 0)
setattr(record, key, current_value + increment_value)
else:
print(f"⚠️ Pole '{key}' nie istnieje w modelu '{entity_type}' - pomijam.")
self.session.commit()
return 0
@exit_gracefully
def get_id_meczu_by_zewnetrzne_id(self, external_id: str) -> int | None:
"""
Zwraca id_meczu na podstawie zewnetrzne_id_meczu.
:param external_id: zewnętrzne ID meczu
:return: id_meczu lub None jeśli nie znaleziono
"""
stmt = select(mecze.id_meczu).where(mecze.zewnetrzne_id_meczu == external_id)
result = self.session.execute(stmt).scalar_one_or_none()
return result
@exit_gracefully
def get_id_zawodnika_by_zewnetrzne_id(self, external_id: str) -> int | None:
"""
Zwraca id_zawodnika na podstawie zewnetrzne_id_zawodnika.
:param external_id: zewnętrzne ID meczu
:return: id_zawodnika lub None jeśli nie znaleziono
"""
stmt = select(sportowcy.id_zawodnika).where(sportowcy.zewnetrzne_id_zawodnika == external_id)
result = self.session.execute(stmt).scalar_one_or_none()
return result
@exit_gracefully
def simple_update_one(self, entity_type, record_id, **kwargs):
"""
Użycie:
simple_update_one(ldb.kluby, "polska", pelna_nazwa="Nowa Nazwa", skrocona_nazwa="NN")
Aktualizuje pojedynczy rekord w bazie danych na podstawie ID.
"""
if not isinstance(entity_type, str):
entity_type = entity_type.__name__
if entity_type not in self.entities:
print(f"Nieznany typ encji: {entity_type}")
return -1
model_class = self.entities[entity_type]
print(f"[{round(time.time())}] UPDATE {entity_type}")
record = self.session.get(model_class, record_id)
if not record:
print(f"Rekord z ID {record_id} nie istnieje w tabeli {entity_type}")
return -1
for key, value in kwargs.items():
if hasattr(record, key):
setattr(record, key, value)
else:
print(f"⚠️ Pole '{key}' nie istnieje w modelu '{entity_type}' - pomijam.")
self.session.commit()
return 0
@exit_gracefully
def dodaj_sportowca_w_meczu(self, entity_type, record_id, **kwargs):
"""
Użycie:
dodaj_sportowca_w_meczu(ldb.sportowcy_w_meczu, ...)
Dodaje pojedynczy rekord w bazie danych.
"""
if not isinstance(entity_type, str):
entity_type = entity_type.__name__
if entity_type not in self.entities:
print(f"Nieznany typ encji: {entity_type}")
return -1
model_class = self.entities[entity_type]
print(f"[{round(time.time())}] INSERT {entity_type}")
record = self.session.get(model_class, record_id)
if not record:
print(f"Rekord z ID {record_id} nie istnieje w tabeli {entity_type}")
return -1
for key, value in kwargs.items():
if hasattr(record, key):
setattr(record, key, value)
else:
print(f"⚠️ Pole '{key}' nie istnieje w modelu '{entity_type}' - pomijam.")
self.session.commit()
return 0
@exit_gracefully
def get_sportowcy_w_meczach_by_sportsman_id(self, id_zawodnika = None, zewnetrzne_id_zawodnika = None, order = "DESC"):
if zewnetrzne_id_zawodnika is not None:
id_zawodnika = self.get_id_zawodnika_by_zewnetrzne_id(zewnetrzne_id_zawodnika)
# Aliasy (dla czytelności)
SportowcyWMeczach = self.entities["sportowcy_w_meczach"]
Sportowcy = self.entities["sportowcy"]
query = self.session.query(
SportowcyWMeczach
).join(
SportowcyWMeczach.zawodnik
).filter(
Sportowcy.id_zawodnika == id_zawodnika
)
# print(f"get_sportowcy_w_meczach_by_sportsman_id: {query}")
if order.lower() == "desc":
query = query.order_by(SportowcyWMeczach.id_rekordu.desc())
return query.all()
@exit_gracefully
def get_sportsman_matches(self, id_zawodnika = None, zewnetrzne_id_zawodnika = None, order = "DESC"):
if zewnetrzne_id_zawodnika is not None:
id_zawodnika = self.get_id_zawodnika_by_zewnetrzne_id(zewnetrzne_id_zawodnika)
# Aliasy
Mecze = self.entities["mecze"]
SportowcyWMeczach = self.entities["sportowcy_w_meczach"]
Sportowcy = self.entities["sportowcy"]
query = self.session.query(
Mecze
).join(
SportowcyWMeczach, Mecze.zewnetrzne_id_meczu == SportowcyWMeczach.zewnetrzne_id_meczu
).join(
Sportowcy, SportowcyWMeczach.id_zawodnika == Sportowcy.id_zawodnika
).filter(
Sportowcy.id_zawodnika == id_zawodnika
)
# print(f"get_sportsman_matches: {query}")
if order.lower() == "desc":
query = query.order_by(Mecze.data.desc())
return query.all()
@exit_gracefully
def sample_data_init(self, override_safety_check=False):
"""
@@ -340,41 +577,53 @@ class baza():
goscie_id="undefined",
gosp_wynik=0,
gosc_wynik=0,
sezon="1970/1970",
sezon="1970-1970",
nazwa_turnieju="Nieznany turniej",
skrocona_nazwa_turnieju="N/A",
flaga=0)
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=0,
minut_gry=0,
gier_sum=0,
goli_sum=0,
asyst_sum=0,
interwencji_sum=0,
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=0,
czerwonych_kartek_sum=0,
wygranych_sum=0,
wynik_sum=0,
meczow_do_wynikow_sum=0)
# Robercik
self.simple_insert_one(kluby,
id_klubu="barcelona",
pelna_nazwa="Barcelona",
skrocona_nazwa="BAR"
)
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=487,
minut_gry=0, # Brak danych
gier_sum=487,
goli_sum=381,
asyst_sum=44,
interwencji_sum=0,
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=2,
czerwonych_kartek_sum=0,
wygranych_sum=0, # Brak danych
przegranych_sum=0, # Brak danych
remisow_sum=0, # Brak danych
wynik_sum=0, # Brak danych
meczow_do_wynikow_sum=0 # Brak danych
)
sportowiec = sportowcy(
zewnetrzne_id_zawodnika="MVC8zHZD",
imie="Robert",
nazwisko="Lewandowski",
data_urodzenia="21.08.1988",
czy_aktywny=True,
klub_id="undefined",
narodowosc="PL",
ilosc_trofeow=0,
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=1,
wycena=64_940_000)
zewnetrzne_id_zawodnika="MVC8zHZD",
imie="Robert",
nazwisko="Lewandowski",
data_urodzenia="21.08.1988",
czy_aktywny=True,
klub_id="barcelona",
narodowosc="PL",
ilosc_trofeow=0, # Brak danych na Flashscore
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=1,
wycena=67_000_000 # Wartość rynkowa: €15.3 mln
)
trofeum = trofea(
nazwa="Nieznane trofeum",
sezon="0000/0000",
sezon="0000-0000",
rok=1970)
session.add(sportowiec)
@@ -386,6 +635,548 @@ class baza():
trofeum.zawodnik = sportowiec
sportowiec.ostatnie_trofeum = trofeum
# MIEJSCE NA DANE KOLEJNYCH SPORTOWCÓW
# TRZEBA ZROBIĆ TO RĘCZNIE, ZGODNIE ZE SCHEMATEM:
# self.simple_insert_one(statystyki_sportowcow,
# ostatni_mecz=1,
# ilosc_wystapien=0,
# minut_gry=0,
# gier_sum=0,
# goli_sum=0,
# asyst_sum=0,
# interwencji_sum=0,
# nieobronionych_interwencji_sum=0,
# zoltych_kartek_sum=0,
# czerwonych_kartek_sum=0,
# wygranych_sum=0,
# przegranych_sum=0,
# remisow_sum=0,
# wynik_sum=0,
# meczow_do_wynikow_sum=0)
# self.simple_insert_one(sportowcy,
# zewnetrzne_id_zawodnika="...",
# imie="...",
# nazwisko="...",
# data_urodzenia="...",
# czy_aktywny=True, # NIE WSZYSCY SĄ AKTYWNI!
# klub_id="undefined",
# narodowosc="...",
# ilosc_trofeow=0,
# pierwszy_mecz_id=1,
# ostatni_gol_dla_id="undefined",
# statystyki_id=2, # itd...
# wycena=...) # w złotówkach
# Ronaldo
self.simple_insert_one(kluby,
id_klubu="al-nassr",
pelna_nazwa="Al Nassr Riyadh",
skrocona_nazwa="ANR" #?
)
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=702,
minut_gry=0, # Brak danych
gier_sum=702,
goli_sum=569,
asyst_sum=81,
interwencji_sum=0,
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=92,
czerwonych_kartek_sum=8,
wygranych_sum=0, # Brak danych
przegranych_sum=0, # Brak danych
remisow_sum=0, # Brak danych
wynik_sum=0, # Brak danych
meczow_do_wynikow_sum=0 # Brak danych
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="WGOY4FSt",
imie="Cristiano",
nazwisko="Ronaldo",
data_urodzenia="05.02.1985",
czy_aktywny=True,
klub_id="al-nassr",
narodowosc="PT",
ilosc_trofeow=0, # Brak danych na Flashscore
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=2,
wycena=55_000_000 # Wartość rynkowa: €12.1 mln
)
# Messi
self.simple_insert_one(kluby,
id_klubu="inter-miami",
pelna_nazwa="Inter Miami",
skrocona_nazwa="INM"
)
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=619,
minut_gry=0, # Brak danych o łącznym czasie gry
gier_sum=619,
goli_sum=528,
asyst_sum=53,
interwencji_sum=0,
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=0,
czerwonych_kartek_sum=0,
wygranych_sum=0, # Brak danych
przegranych_sum=0, # Brak danych
remisow_sum=0, # Brak danych
wynik_sum=0, # Brak danych
meczow_do_wynikow_sum=0 # Brak danych
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="vgOOdZbd",
imie="Lionel",
nazwisko="Messi",
data_urodzenia="24.06.1987",
czy_aktywny=True,
klub_id="inter-miami",
narodowosc="AR",
ilosc_trofeow=0, # Brak danych na Flashscore
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=3,
wycena=67_000_000 # Wartość rynkowa: €15.3 mln
)
# Mbappe
self.simple_insert_one(kluby,
id_klubu="real-madryt",
pelna_nazwa="Real Madryt",
skrocona_nazwa="RMA"
)
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=280,
minut_gry=0, # Brak danych o łącznym czasie gry
gier_sum=280,
goli_sum=222,
asyst_sum=67,
interwencji_sum=0,
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=38,
czerwonych_kartek_sum=2,
wygranych_sum=0, # Brak danych
przegranych_sum=0, # Brak danych
remisow_sum=0, # Brak danych
wynik_sum=0, # Brak danych
meczow_do_wynikow_sum=0 # Brak danych
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="Wn6E2SED",
imie="Kylian",
nazwisko="Mbappé",
data_urodzenia="20.12.1998",
czy_aktywny=True,
klub_id="real-madryt",
narodowosc="FR",
ilosc_trofeow=0, # Brak danych na Flashscore
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=4,
wycena=185_700_000 # Wartość rynkowa: €185.7 mln
)
# Zlatan
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=519,
minut_gry=0, # Brak danych o łącznym czasie gry
gier_sum=519,
goli_sum=355,
asyst_sum=63,
interwencji_sum=0,
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=87,
czerwonych_kartek_sum=9,
wygranych_sum=0, # Brak danych
przegranych_sum=0, # Brak danych
remisow_sum=0, # Brak danych
wynik_sum=0, # Brak danych
meczow_do_wynikow_sum=0 # Brak danych
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="AiH2zDve",
imie="Zlatan",
nazwisko="Ibrahimović",
data_urodzenia="03.10.1981",
czy_aktywny=False,
klub_id="undefined",
narodowosc="SE",
ilosc_trofeow=0, # Brak danych na Flashscore
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=5,
wycena=0 # Zakończył karierę
)
# Suarez
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=608,
minut_gry=0, # Brak danych o łącznym czasie gry
gier_sum=608,
goli_sum=399,
asyst_sum=122,
interwencji_sum=0,
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=0,
czerwonych_kartek_sum=0,
wygranych_sum=0, # Brak danych
przegranych_sum=0, # Brak danych
remisow_sum=0, # Brak danych
wynik_sum=0, # Brak danych
meczow_do_wynikow_sum=0 # Brak danych
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="dUShzrBp",
imie="Luis",
nazwisko="Suárez",
data_urodzenia="24.01.1987",
czy_aktywny=True,
klub_id="inter-miami",
narodowosc="UY",
ilosc_trofeow=0, # Brak danych na Flashscore
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=6,
wycena=8_800_000 # Wartość rynkowa: €2.0 mln
)
# Haaland
self.simple_insert_one(kluby,
id_klubu="manchester-city",
pelna_nazwa="Manchester City",
skrocona_nazwa="MCI"
)
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=235,
minut_gry=0, # Brak danych o łącznym czasie gry
gier_sum=235,
goli_sum=178,
asyst_sum=41,
interwencji_sum=0,
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=18,
czerwonych_kartek_sum=0,
wygranych_sum=0, # Brak danych
przegranych_sum=0, # Brak danych
remisow_sum=0, # Brak danych
wynik_sum=0, # Brak danych
meczow_do_wynikow_sum=0 # Brak danych
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="UmV9iQmE",
imie="Erling",
nazwisko="Haaland",
data_urodzenia="21.07.2000",
czy_aktywny=True,
klub_id="manchester-city",
narodowosc="NO",
ilosc_trofeow=0, # Brak danych
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=7,
wycena=777_500_000 # ~€178.7 mln
)
# Karim Benzema
self.simple_insert_one(kluby,
id_klubu="al-ittihad",
pelna_nazwa="Al-Ittihad FC", #?
skrocona_nazwa="RMA" #?
)
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=589,
minut_gry=0, # Brak danych
gier_sum=589,
goli_sum=305,
asyst_sum=0,
interwencji_sum=0,
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=14,
czerwonych_kartek_sum=0,
wygranych_sum=0, # Brak danych
przegranych_sum=0,
remisow_sum=0,
wynik_sum=0,
meczow_do_wynikow_sum=0
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="tpV0VX0S",
imie="Karim",
nazwisko="Benzema",
data_urodzenia="19.12.1987",
czy_aktywny=True,
klub_id="al-ittihad",
narodowosc="FR",
ilosc_trofeow=0, # Flashscore nie podaje
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=8,
wycena=31_950_000 # €7.1 mln
)
# Sergio Aguero
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=443,
minut_gry=0, # Brak danych o czasie gry
gier_sum=443,
goli_sum=254,
asyst_sum=38,
interwencji_sum=0,
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=47,
czerwonych_kartek_sum=2,
wygranych_sum=0, # Brak danych
przegranych_sum=0, # Brak danych
remisow_sum=0, # Brak danych
wynik_sum=0, # Brak danych
meczow_do_wynikow_sum=0 # Brak danych
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="vw8ZV7HC",
imie="Sergio",
nazwisko="Agüero",
data_urodzenia="02.06.1988",
czy_aktywny=False,
klub_id="undefined",
narodowosc="AR",
ilosc_trofeow=0, # Brak danych
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=9,
wycena=0 # Zakończył karierę
)
# Edinson Cavani
self.simple_insert_one(kluby,
id_klubu="boca-juniors",
pelna_nazwa="Boca Juniors", #?
skrocona_nazwa="CAB" #?
)
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=507,
minut_gry=0, # Brak danych o czasie gry
gier_sum=507,
goli_sum=273,
asyst_sum=62,
interwencji_sum=0,
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=3,
czerwonych_kartek_sum=0,
wygranych_sum=0, # Brak danych
przegranych_sum=0, # Brak danych
remisow_sum=0, # Brak danych
wynik_sum=0, # Brak danych
meczow_do_wynikow_sum=0 # Brak danych
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="Qgx2trzH",
imie="Edinson",
nazwisko="Cavani",
data_urodzenia="14.02.1987",
czy_aktywny=True,
klub_id="boca-juniors",
narodowosc="UY",
ilosc_trofeow=0,
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=10,
wycena=4_212_000 # PLN
)
# Radamel Falcao
self.simple_insert_one(kluby,
id_klubu="millonarios",
pelna_nazwa="Millonarios",
skrocona_nazwa="MLA" #?
)
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=360,
minut_gry=0, # brak danych o czasie gry
gier_sum=360,
goli_sum=170,
asyst_sum=17,
interwencji_sum=54,
nieobronionych_interwencji_sum=1,
zoltych_kartek_sum=0, # brak danych
czerwonych_kartek_sum=0,
wygranych_sum=0,
przegranych_sum=0,
remisow_sum=0,
wynik_sum=0,
meczow_do_wynikow_sum=0
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="2oMimkAU",
imie="Radamel",
nazwisko="Falcao",
data_urodzenia="10.02.1986",
czy_aktywny=True,
klub_id="millonarios",
narodowosc="CO",
ilosc_trofeow=0, # brak danych
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=11,
wycena=1_084_500 #PLN
)
# Wayne Rooney
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=568,
minut_gry=0, # brak danych o czasie gry
gier_sum=568,
goli_sum=234,
asyst_sum=47,
interwencji_sum=0, # brak danych
nieobronionych_interwencji_sum=0, # brak danych
zoltych_kartek_sum=105,
czerwonych_kartek_sum=4,
wygranych_sum=0, # brak danych
przegranych_sum=0,
remisow_sum=0,
wynik_sum=0,
meczow_do_wynikow_sum=0
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="WfXv1DCa",
imie="Wayne",
nazwisko="Rooney",
data_urodzenia="24.10.1985",
czy_aktywny=False, # zakończył karierę
klub_id="undefined",
narodowosc="ENG",
ilosc_trofeow=0, # brak danych
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=12,
wycena=0 # brak wyceny
)
# Robin van Persie
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=374,
minut_gry=0, # brak danych o czasie gry
gier_sum=374,
goli_sum=190,
asyst_sum=16,
interwencji_sum=0, # brak danych
nieobronionych_interwencji_sum=0, # brak danych
zoltych_kartek_sum=54,
czerwonych_kartek_sum=3,
wygranych_sum=0, # brak danych
przegranych_sum=0,
remisow_sum=0,
wynik_sum=0,
meczow_do_wynikow_sum=0
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="0vgcq6un",
imie="Robin",
nazwisko="van Persie",
data_urodzenia="06.08.1983",
czy_aktywny=False,
klub_id="undefined", # zakończył karierę
narodowosc="NED",
ilosc_trofeow=0, # brak danych
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=13,
wycena=0 # zakończył karierę
)
# Harry Kane
self.simple_insert_one(kluby,
id_klubu="bayern",
pelna_nazwa="Bayern Monachium",
skrocona_nazwa="BAY"
)
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=439,
minut_gry=0, # brak danych o czasie gry
gier_sum=439,
goli_sum=295,
asyst_sum=64,
interwencji_sum=0, # brak danych
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=48,
czerwonych_kartek_sum=1,
wygranych_sum=0, # brak danych
przegranych_sum=0,
remisow_sum=0,
wynik_sum=0,
meczow_do_wynikow_sum=0
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="v5HSlEAa",
imie="Harry",
nazwisko="Kane",
data_urodzenia="28.07.1993",
czy_aktywny=True,
klub_id="bayern",
narodowosc="ENG",
ilosc_trofeow=0, # brak danych
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=14,
wycena=344_700_000 # PLN
)
# Ciro Immobile
self.simple_insert_one(kluby,
id_klubu="besiktas",
pelna_nazwa="Besiktas",
skrocona_nazwa="BES"
)
self.simple_insert_one(statystyki_sportowcow,
ostatni_mecz=1,
ilosc_wystapien=455,
minut_gry=0, # brak danych o czasie gry
gier_sum=455,
goli_sum=250,
asyst_sum=51,
interwencji_sum=0, # brak danych
nieobronionych_interwencji_sum=0,
zoltych_kartek_sum=71,
czerwonych_kartek_sum=3,
wygranych_sum=0,
przegranych_sum=0,
remisow_sum=0,
wynik_sum=0,
meczow_do_wynikow_sum=0
)
self.simple_insert_one(sportowcy,
zewnetrzne_id_zawodnika="4S9fNUYh",
imie="Ciro",
nazwisko="Immobile",
data_urodzenia="20.02.1990",
czy_aktywny=True,
klub_id="besiktas",
narodowosc="IT",
ilosc_trofeow=0, # brak danych
pierwszy_mecz_id=1,
ostatni_gol_dla_id="undefined",
statystyki_id=15,
wycena=12_600_000 # PLN
)
session.commit()
return 0
return 1
return 1

View File

@@ -1,4 +1,4 @@
from git import Repo # hash ostatniego commitu
#from git import Repo # hash ostatniego commitu
import os
import time
import toml
@@ -23,9 +23,9 @@ def safeTraverse(obj: dict, path: list, default=None):
try:
for x in path:
result = result[x]
except KeyError:
except (KeyError, TypeError):
result = default
# print(f"error reading: {' -> '.join(path)} - returning: {default}")
print(f"error reading: {' -> '.join(path)} - returning: {default}")
finally:
return result
@@ -36,11 +36,11 @@ def getCommit() -> str | None:
return None
def getCommitInFormattedHTML():
repo = "<p>Brak informacji o wersji skryptu</p>"
repo = "<center><p style=\"color: white;\">Brak informacji o wersji skryptu</p></center>"
commit = getCommit()
if commit is not None:
repo = f"<p>Commit: <a href='https://gitea.7o7.cx/roberteam/lewangoalski/commit/{commit}'>{commit[:11]}</a></p>"
repo = f"<center><p style=\"color: white;\">Commit: <a href='https://gitea.7o7.cx/roberteam/lewangoalski/commit/{commit}'>{commit[:11]}</a></p></center>"
return repo
@@ -155,4 +155,4 @@ config = {}
configfile = "config.toml"
version = getCommitWithFailsafe()
apiVersion = "1"
randomly_generated_passcode = 0
randomly_generated_passcode = 0

View File

@@ -2,37 +2,38 @@ from flask import render_template, request, make_response
import lewy_api_v1
import lewy_db
import lewy_globals
import json
from lewy_api_v1 import get_matches
def get_lewy_stats():
return {
'all_time_stats': {
'goals': 380,
'assists': 120,
'matches': 450,
'goals': 589+85,
'assists':154+35,
'matches': 791+158,
},
'club_stats': {
'goals': 132,
'assists': 112,
'matches': 245,
'goals': 589,
'assists': 154,
'matches': 791,
},
'nation_stats': {
'goals': 86,
'assists': 52,
'matches': 158,
'goals': 85,
'assists': 35,
'matches': 158,
},
'worldcup': {
'goals': 7,
'assists': 2,
'matches': 11,
'international_cups': {
'goals': 110,
'assists': 19,
'matches': 152,
},
'euro': {
'goals': 6,
'assists': 2,
'matches': 18,
'national_cups': {
'goals': 58,
'assists': 4,
'matches': 74,
},
'cards': {
'yellow': 24,
'red': 4,
'yellow': 86,
'red': 2,
}
}
@@ -58,11 +59,12 @@ def index():
def mecze():
# Możesz dostarczyć szczegóły dotyczące meczów
matches = [
{'date': '2024-10-12', 'opponent': 'Real Madrid', 'goals': 2, 'assists': 1, 'minutes': 90},
{'date': '2024-10-19', 'opponent': 'Valencia', 'goals': 1, 'assists': 0, 'minutes': 85},
]
return render_template('matches.html', matches=matches)
selected_date = request.args.get("date", '2025')
#with open("static/lewandowski_matches.json", "r") as file:
# data = json.load(file)
status, msg, matches = get_matches(None, id_zawodnika=1)
return render_template('matches.html', matches=matches, selected_date=selected_date)
def statystyki():
dane=get_lewy_stats()
@@ -71,31 +73,107 @@ def statystyki():
def clubs():
selected_club = request.args.get("club","FC Barcelona")
clubs = [
{'club': 'FC Barcelona', 'goals': 22,'assist':12},
{'club': 'Bayern Monachium', 'goals': 132,'assist':12},
{'club': 'Borussia Dortmund', 'goals': 132,'assist':12},
{'club': 'Lech Poznan', 'goals': 132,'assist':12},
{'club': 'FC Barcelona', 'goals': 101,'assist':20, 'matches':147,'minutes_played': 11684,'yellow_card':12,'red_card': 1, 'wins':101, 'draws': 14,'lost': 32},
{'club': 'Bayern Monachium', 'goals': 344,'assist':73,'matches':375,'minutes_played': 31759,'yellow_card':36,'red_card': 0, 'wins':307, 'draws': 35,'lost': 33},
{'club': 'Borussia Dortmund', 'goals': 103,'assist':42,'matches':187,'minutes_played': 14374,'yellow_card':19,'red_card': 1, 'wins':120, 'draws': 40,'lost': 27},
{'club': 'Lech Poznan', 'goals': 41,'assist':19,'matches':82,'minutes_played': 6858,'yellow_card':9,'red_card': 0, 'wins':'-', 'draws': '-','lost': '-'},
]
return render_template('club.html', clubs=clubs, selected_club=selected_club)
def representation():
nation_stats = {
'goals': 86,
'assists': 52,
'goals': 85,
'assists': 35,
'matches': 158,
'minutes_played': 12108,
'yellow_card':10,
'red_card': 0,
'wins':75,
'draws': 35,
'lost': 48
}
return render_template('representation.html', nation_stats=nation_stats)
def compare():
selected_player = request.args.get("player","Leo Messi")
lewy=get_lewy_stats()
player2 = [
{'name':'Leo Messi','goals': 34,'assists': 12},
{'name':'Leo Messi','goals': 865,'assists': 384,'matches':1103},
{'name':'Cristiano Ronaldo','goals': 937,'assists': 257,'matches':1280},
{'name':'Kylian Mbappé','goals': 330,'assists': 187,'matches':436},
{'name':'Zlatan Ibrahimović','goals': 573,'assists': 200,'matches':988},
{'name':'Luis Suárez','goals': 511,'assists': 277,'matches':858},
{'name':'Erling Haaland','goals': 276,'assists': 49,'matches':342},
{'name':'Karim Benzema','goals': 438,'assists': 160,'matches':820},
{'name':'Sergio Agüero','goals': 385,'assists': 118,'matches':685},
]
return render_template('compare.html',player2=player2, selected_player=selected_player,**lewy, )
def trophies():
trophy = [
{'name': 'asdasd', 'year': 2023},
{'name': 'ssss', 'sezon': '2022/2023'},
{'name': 'Piłkarz Roku FIFA', 'year':'2021'},
{'name': 'Piłkarz Roku FIFA', 'year':'2020'},
{'name': 'UEFA Best Player in Europe', 'year':'2020'},
{'name': 'Zdobywca Złotego Buta (Europe)', 'sezon':'2021/2022'},
{'name': 'Zdobywca Złotego Buta (Europe)', 'sezon':'2020/2021'},
{'name': 'Piłkarz roku', 'sezon':'2021'},
{'name': 'Piłkarz roku', 'sezon':'2021'},
{'name': 'Piłkarz roku', 'sezon':'2020'},
{'name': 'Piłkarz roku', 'sezon':'2020'},
{'name': 'Piłkarz roku', 'sezon':'2019'},
{'name': 'Piłkarz roku', 'sezon':'2017'},
{'name': 'Piłkarz roku', 'sezon':'2016'},
{'name': 'Piłkarz roku', 'sezon':'2015'},
{'name': 'Piłkarz roku', 'sezon':'2014'},
{'name': 'Piłkarz roku', 'sezon':'2013'},
{'name': 'Piłkarz roku', 'sezon':'2012'},
{'name': 'Piłkarz roku', 'sezon':'2011'},
{'name': 'Król strzelców', 'sezon':'2022/2023'},
{'name': 'Król strzelców', 'sezon':'2021/2022'},
{'name': 'Król strzelców', 'sezon':'2020/2021'},
{'name': 'Król strzelców', 'sezon':'2019/2020'},
{'name': 'Król strzelców', 'sezon':'2019/2020'},
{'name': 'Król strzelców', 'sezon':'2018/2019'},
{'name': 'Król strzelców', 'sezon':'2018/2019'},
{'name': 'Król strzelców', 'sezon':'2017/2018'},
{'name': 'Król strzelców', 'sezon':'2017/2018'},
{'name': 'Król strzelców', 'sezon':'2016/2017'},
{'name': 'Król strzelców', 'sezon':'2016/2017'},
{'name': 'Król strzelców', 'sezon':'2015/2016'},
{'name': 'Król strzelców', 'sezon':'2015/2016'},
{'name': 'Król strzelców', 'sezon':'2013/2014'},
{'name': 'Król strzelców', 'sezon':'2011/2012'},
{'name': 'Król strzelców', 'sezon':'2009/2010'},
{'name': 'Zdobywca Ligi Mistrzów', 'sezon':'2019/2020'},
{'name': 'Zdobywca Klubowych Mistrzostw Świata', 'sezon':'2021'},
{'name': 'Mistrz Hiszpanii', 'sezon':'2024/2025'},
{'name': 'Mistrz Hiszpanii', 'sezon':'2022/2023'},
{'name': 'Mistrz Niemiec', 'sezon':'2021/2022'},
{'name': 'Mistrz Niemiec', 'sezon':'2020/2021'},
{'name': 'Mistrz Niemiec', 'sezon':'2019/2020'},
{'name': 'Mistrz Niemiec', 'sezon':'2018/2019'},
{'name': 'Mistrz Niemiec', 'sezon':'2017/2018'},
{'name': 'Mistrz Niemiec', 'sezon':'2016/2017'},
{'name': 'Mistrz Niemiec', 'sezon':'2015/2016'},
{'name': 'Mistrz Niemiec', 'sezon':'2014/2015'},
{'name': 'Mistrz Niemiec', 'sezon':'2011/2012'},
{'name': 'Mistrz Niemiec', 'sezon':'2010/2011'},
{'name': 'Mistrz Niemiec', 'sezon':'2020/2021'},
{'name': 'Zdobywca Superpucharu UEFA', 'sezon':'2020/2021'},
{'name': 'Zdobywca Pucharu Niemiec', 'sezon':'2019/2020'},
{'name': 'Zdobywca Pucharu Niemiec', 'sezon':'2018/2019'},
{'name': 'Zdobywca Pucharu Niemiec', 'sezon':'2015/2016'},
{'name': 'Zdobywca Pucharu Niemiec', 'sezon':'2011/2012'},
{'name': 'Zdobywca Pucharu Hiszpanii', 'sezon':'2024/2025'},
{'name': 'Zdobywca Superpucharu Niemiec', 'sezon':'2021/2022'},
{'name': 'Zdobywca Superpucharu Niemiec', 'sezon':'2020/2021'},
{'name': 'Zdobywca Superpucharu Niemiec', 'sezon':'2018/2019'},
{'name': 'Zdobywca Superpucharu Niemiec', 'sezon':'2017/2018'},
{'name': 'Zdobywca Superpucharu Niemiec', 'sezon':'2016/2017'},
{'name': 'Zdobywca Superpucharu Niemiec', 'sezon':'2013/2014'},
{'name': 'Zdobywca Superpucharu Hiszpanii', 'sezon':'2024/2025'},
{'name': 'Zdobywca Superpucharu Hiszpanii', 'sezon':'2022/2023'},
{'name': 'Mistrz Polski', 'sezon':'2009/2010'},
{'name': 'Zdobywca Superpucharu Polski', 'sezon': '2009/2010'},
{'name': 'Zdobywca Pucharu Polski', 'sezon': '2008/2009'},
]
return render_template('trophies.html',trophy=trophy)

File diff suppressed because it is too large Load Diff

View File

@@ -386,34 +386,6 @@ header button {
display: none;
}
}
@media (max-width: 600px){
.section__matches
{
font-size: 10px;
}
.section__matches th{
padding: 3px;
}
.club-stats-grid
{
grid-template-columns: 1fr 1fr !important;
}
.club-stats-grid .stat-box
{
width: 100%;
}
.section-stats-center .section-stats .stats
{
display: flex;
flex-direction: column;
align-items: center;
}
.section-stats-center .section-stats .stats .stat-box
{
width: 100%;
padding: 0;
}
}
@@ -744,6 +716,26 @@ body.poland-mode .general-stats-section .grid article:nth-child(2){background-co
body.poland-mode .general-stats-section .grid article:nth-child(3){background-color: var(--barca-blue);}
body.poland-mode .about-section-image::after{
background:var(--polska-red-dark);
}
body.poland-mode .section__matches
{
background-color: white;
}
body.poland-mode .section__matches h2
{
background-color: var(--polska-red);
color: white;
}
body.poland-mode .section__matches th
{
color: var(--barca-red)
}
body.poland-mode .section__matches tr:hover:has(:not(th))
{
background-color: #ff5959;
}
body.poland-mode select{
background:var(--section-color);
}
/* Przyciski i elementy */
@@ -761,6 +753,13 @@ body.poland-mode .about-section-image::after{
margin-bottom: 10px;
width: 100%;
}
.section-stats h1{
font-size: 34px;
color: var(--barca-red);
}
body.poland-mode .section-stats h1{
color: white;
}
.section-stats-center
{
display: grid;
@@ -879,4 +878,32 @@ select
background-color: var(--barca-blue);
border-radius: 10px;
border: 2px solid white;
}
@media (max-width: 600px){
.sectionmatches
{
font-size: 10px;
}
.sectionmatches th{
padding: 3px;
}
.club-stats-grid
{
grid-template-columns: 1fr 1fr !important;
}
.club-stats-grid .stat-box
{
width: 100%;
}
.section-stats-center .section-stats .stats
{
display: flex;
flex-direction: column;
align-items: center;
}
.section-stats-center .section-stats .stats .stat-box
{
width: 100%;
padding: 0;
}
}

View File

@@ -33,25 +33,25 @@
<p>Asysty:</p> <span class="stat-box-special"> {{ stats.assist }} </span>
</div>
<div class="stat-box">
<p>Występy:</p> <span class="stat-box-special"> {{ stats.goals }} </span>
<p>Występy:</p> <span class="stat-box-special"> {{ stats.matches }} </span>
</div>
<div class="stat-box">
<p>Łączny czas gry:</p> <span class="stat-box-special"> {{ stats.goals }}</span>
<p>Łączny czas gry:</p> <span class="stat-box-special"> {{ stats.minutes_played }}</span>
</div>
<div class="stat-box">
<p>Hat-tricki:</p> <span class="stat-box-special"> {{ stats.goals }}</span>
<p>Żółte kartki:</p> <span class="stat-box-special"> {{ stats.yellow_card }}</span>
</div>
<div class="stat-box">
<p>Zwycięstwa:</p> <span class="stat-box-special"> {{ stats.goals }}</span>
<p>Czerwone kartki:</p> <span class="stat-box-special"> {{ stats.red_card }}</span>
</div>
<div class="stat-box">
<p>Porażki:</p> <span class="stat-box-special"> {{ stats.goals }}</span>
<p>Zwycięstwa:</p> <span class="stat-box-special"> {{ stats.wins }}</span>
</div>
<div class="stat-box">
<p>Żółte kartki:</p> <span class="stat-box-special"> {{ stats.goals }}</span>
<p>Remisy:</p> <span class="stat-box-special"> {{ stats.draws }}</span>
</div>
<div class="stat-box">
<p>Czerwone kartki:</p> <span class="stat-box-special"> {{ stats.goals }}</span>
<p>Porażki:</p> <span class="stat-box-special"> {{ stats.lost }}</span>
</div>
</div>
</section>

View File

@@ -6,13 +6,19 @@
<select onchange="location = this.value;">
<option disabled selected>Wybierz zawodnika</option>
<option value="{{ url_for('compare', player='Leo Messi') }}">Leo Messi</option>
<option value="{{ url_for('compare', player='Ronaldo') }}">Cristiano Ronaldo</option>
<option value="{{ url_for('compare', player='Neymar') }}">Neymar</option>
<option value="{{ url_for('compare', player='Cristiano Ronaldo') }}">Cristiano Ronaldo</option>
<option value="{{ url_for('compare', player='Kylian Mbappé') }}">Kylian Mbappé</option>
<option value="{{ url_for('compare', player='Zlatan Ibrahimović') }}">Zlatan Ibrahimović</option>
<option value="{{ url_for('compare', player='Luis Suárez') }}">Luis Suárez</option>
<option value="{{ url_for('compare', player='Erling Haaland') }}">Erling Haaland</option>
<option value="{{ url_for('compare', player='Karim Benzema') }}">Karim Benzema</option>
<option value="{{ url_for('compare', player='Sergio Agüero') }}">Sergio Agüero</option>
</select>
{%for player in player2 %}
{% if player.name == selected_player %}
<section class="section-stats">
<h1>Robert Lewandowski VS {{selected_player}}</h1>
<h2>Gole</h2>
<div class="stats">
<div class="stat-box">
@@ -41,26 +47,15 @@
<h2>Wystąpienia</h2>
<div class="stats">
<div class="stat-box">
<h3>{{ all_time_stats.assists }}</h3>
<h3>{{ all_time_stats.matches }}</h3>
</div>
<div class="stat-box">
<h3>{{ player.assists}}</h3>
<h3>{{ player.matches}}</h3>
</div>
</div>
<h2>Minuty zagrane</h2>
<div class="stats">
<div class="stat-box">
<h3>{{ all_time_stats.assists }}</h3>
</div>
<div class="stat-box">
<h3>{{ player.assists}}</h3>
</div>
</div>
</section>
{% endif %}
{% endfor%}

View File

@@ -10,27 +10,27 @@
<article class="article__how-it-works">
<h3>Jak to działa?</h3>
<h4>Pobieranie statystyk</h4>
<p>Lorem ipsum dolor sit amet consectetur adipisicing elit. Ratione harum minus hic, voluptate perspiciatis laborum? Alias maxime, voluptate reprehenderit iusto dolorem officiis porro voluptatibus repellat dicta doloribus, blanditiis similique accusantium.</p>
<p>Tu znajdziesz najświeższe statystyki, aktualizowane zaraz po zakończeniu meczu Lewego</p>
<h4>Porównywanie zawodników</h4>
<p>Lorem, ipsum dolor sit amet consectetur adipisicing elit. Fuga, in perspiciatis. Sequi laborum et animi quas sit voluptatibus alias sed ad molestias nulla vel cum, consectetur commodi odio aliquam officia.</p>
<p>Robert Lewandowski jest uznawany za jednego z najlepszych zawodników na świecie, zobacz jak wypada w porównaniu do innych czołowych zawodników i sprawdź czy zasługuje na miano jednego z najlepszych napastników świata.</p>
</article>
<div class="about-section-image">
<img src="{{ url_for('static', filename='gigabuła.png') }}">
</div>
<article>
<h3>Mecze</h3>
<p>Lorem ipsum dolor sit amet consectetur adipisicing elit. Soluta ullam iusto ex? Quo amet officia aliquam odio sint harum nam eaque nihil ipsa quos aliquid, illum voluptatum, numquam, magnam omnis?</p>
<p>Nie oglądałeś ostatniego meczu Barcelony? Wypadło Ci coś i nie widziałeś najnowszego spotkania reprezentacji Polski? Wejdź i sprawdź jak poradził sobie Robert.</p>
<a href="/mecze">Zobacz mecze</a>
</article>
<article>
<h3>Statystyki</h3>
<p>Lorem ipsum dolor sit amet consectetur adipisicing elit. Temporibus dolore tenetur nulla sint recusandae illo dolores aspernatur ducimus, omnis vitae ipsam neque animi voluptates eos porro, nihil iusto veniam commodi!</p>
<p>Jesteś zainteresowany jak Robert radzi sobie na przestrzeni tylu lat gry? Jesteś ciekaw czy napastnik oprócz strzelania bramek lubi kolekcjonować kartki? Sprawdź u nas jak wygląda kariera Roberta Lewandowskiego na podstawie jego gry w kadrze Polski, klubach oraz Pucharach ligowych i międzynarodowych.</p>
<a href="/statystyki">Zobacz statystyki</a>
</article>
<article>
<h3>Osiągnięcia</h3>
<p>Lorem ipsum dolor sit amet consectetur adipisicing elit. Quod dicta veritatis quibusdam eligendi corrupti. Expedita delectus assumenda ipsum illum molestias a voluptates, voluptas quia reprehenderit, quod non, eum veritatis tenetur!</p>
<a href="/club">Zobacz osiągnięcia</a>
<h3>Trofea</h3>
<p>Zastanawia Cie jakie było ostatnie trofeum Roberta? A może z kolegami sprawdzacie, który lepiej pamięta złote chwile w karierze naszego idola? Rozwiejcie wasze wątpliwości i zobaczcie wielką kolekcji zdobyczy pucharowych najlepszego polskiego zawodnika.</p>
<a href="/club">Zobacz trofea</a>
</article>
</section>
<!--
@@ -57,5 +57,7 @@
{% endblock %}
{% block footer %}
<!--
{{ commit_in_html | safe }}
-->
{% endblock %}

View File

@@ -3,36 +3,53 @@
{% block title %}Lista meczów{% endblock %}
{% block content %}
<select>
<select onchange="location = this.value;">
<option disabled selected>Wybierz rok</option>
<option value="{{ url_for('compare', player='Leo Messi') }}">2024/2025</option>
<option value="{{ url_for('compare', player='Ronaldo') }}">2023/2024</option>
<option value="{{ url_for('compare', player='Neymar') }}">2022/2023</option>
<option value="{{ url_for('compare', player='Neymar') }}">2021/2022</option>
<option value="{{ url_for('compare', player='Neymar') }}">2020/2021</option>
<option value="{{ url_for('compare', player='Neymar') }}">2019/2020</option>
<option value="{{ url_for('compare', player='Neymar') }}">2018/2019</option>
<option value="{{ url_for('compare', player='Neymar') }}">2017/2018</option>
<option value="{{ url_for('compare', player='Neymar') }}">2016/2017</option>
<option value="{{ url_for('mecze', date='2025') }}">2025</option>
<option value="{{ url_for('mecze', date='2024') }}">2024</option>
<option value="{{ url_for('mecze', date='2023') }}">2023</option>
<option value="{{ url_for('mecze', date='2022') }}">2022</option>
<option value="{{ url_for('mecze', date='2021') }}">2021</option>
<option value="{{ url_for('mecze', date='2020') }}">2020</option>
<option value="{{ url_for('mecze', date='2019') }}">2019</option>
<option value="{{ url_for('mecze', date='2018') }}">2018</option>
<option value="{{ url_for('mecze', date='2017') }}">2017</option>
<option value="{{ url_for('mecze', date='2016') }}">2016</option>
<option value="{{ url_for('mecze', date='2015') }}">2015</option>
<option value="{{ url_for('mecze', date='2014') }}">2014</option>
</select>
<section class="section__matches">
<h2>📅 Mecze Roberta</h2>
<table>
<tr>
<th>Data</th>
<!--
<th>Przeciwnik</th>
<th>Gole</th>
<th>Asysty</th>
<th>Minuty</th>
-->
<th>Gospodarze</th>
<th>Wynik</th>
<th>Goście</th>
<th></th>
</tr>
{% for match in matches %}
{% if match.data[:4] == selected_date %}
<tr>
<td>{{ match.date }}</td>
<td>{{ match.opponent }}</td>
<td>{{ match.data }}</td>
<td>{{ match.gospodarze_pelna_nazwa }}</td>
<td>{{ match.gosp_wynik }} : {{ match.gosc_wynik }}</td>
<td>{{ match.goscie_pelna_nazwa }}</td>
<!--
<td>{{ match.goals }}</td>
<td>{{ match.assists }}</td>
<td>{{ match.minutes }}</td>
-->
</tr>
{% endif %}
{% endfor %}
</table>
</section>

View File

@@ -11,28 +11,28 @@
<p>Gole:</p> <span class="stat-box-special"> {{ nation_stats.goals }} </span>
</div>
<div class="stat-box">
<p>Asysty:</p> <span class="stat-box-special"> {{ nation_stats.assist }} </span>
<p>Asysty:</p> <span class="stat-box-special"> {{ nation_stats.assists }} </span>
</div>
<div class="stat-box">
<p>Występy:</p> <span class="stat-box-special"> {{ nation_stats.goals }} </span>
<p>Występy:</p> <span class="stat-box-special"> {{ nation_stats.matches }} </span>
</div>
<div class="stat-box">
<p>Łączny czas gry:</p> <span class="stat-box-special"> {{ nation_stats.goals }}</span>
<p>Łączny czas gry:</p> <span class="stat-box-special"> {{ nation_stats.minutes_played }}</span>
</div>
<div class="stat-box">
<p>Hat-tricki:</p> <span class="stat-box-special"> {{ nation_stats.goals }}</span>
<p>Żółte kartki:</p> <span class="stat-box-special"> {{ nation_stats.yellow_card }}</span>
</div>
<div class="stat-box">
<p>Zwycięstwa:</p> <span class="stat-box-special"> {{ nation_stats.goals }}</span>
<p>Czerwone kartki:</p> <span class="stat-box-special"> {{ nation_stats.red_card }}</span>
</div>
<div class="stat-box">
<p>Porażki:</p> <span class="stat-box-special"> {{ nation_stats.goals }}</span>
<p>Zwycięstwa:</p> <span class="stat-box-special"> {{ nation_stats.wins }}</span>
</div>
<div class="stat-box">
<p>Żółte kartki:</p> <span class="stat-box-special"> {{ nation_stats.goals }}</span>
<p>Remisy:</p> <span class="stat-box-special"> {{ nation_stats.draws }}</span>
</div>
<div class="stat-box">
<p>Czerwone kartki:</p> <span class="stat-box-special"> {{ nation_stats.goals }}</span>
<p>Porażki:</p> <span class="stat-box-special"> {{ nation_stats.lost }}</span>
</div>
</div>
</section>

View File

@@ -56,35 +56,35 @@
</div>
</section>
<section class="section-stats">
<h2>Mistrzostwa świata</h2>
<h2>Puchary międzynarodowe</h2>
<div class="stats">
<div class="stat-box">
<h3>{{ worldcup.goals }}</h3>
<h3>{{ international_cups.goals }}</h3>
<p>Gole</p>
</div>
<div class="stat-box">
<h3>{{ worldcup.assists }}</h3>
<h3>{{ international_cups.assists }}</h3>
<p>Asysty</p>
</div>
<div class="stat-box">
<h3>{{ worldcup.matches }}</h3>
<h3>{{ international_cups.matches }}</h3>
<p>Występy</p>
</div>
</div>
</section>
<section class="section-stats">
<h2>EURO</h2>
<h2>Puchary krajowe</h2>
<div class="stats">
<div class="stat-box">
<h3>{{ euro.goals }}</h3>
<h3>{{ national_cups.goals }}</h3>
<p>Gole</p>
</div>
<div class="stat-box">
<h3>{{ euro.assists }}</h3>
<h3>{{ national_cups.assists }}</h3>
<p>Asysty</p>
</div>
<div class="stat-box">
<h3>{{ euro.matches }}</h3>
<h3>{{ national_cups.matches }}</h3>
<p>Występy</p>
</div>
</div>

View File

@@ -4,7 +4,7 @@
{% block content %}
<section class="section__matches">
<h2>📅 Trofea</h2>
<h2>🏆 Trofea</h2>
<table>
<tr>
<th>Nazwa</th>

Binary file not shown.