mirror of
https://git.bolliret.ch/pcs/pcs-website
synced 2026-01-18 13:21:37 +01:00
Stupid copy and paste of scripts to support french. Btw. docker image needs to be recreated
This commit is contained in:
parent
f058484743
commit
134f3dedb2
10 changed files with 463 additions and 6 deletions
|
|
@ -5,6 +5,7 @@ RUN apt install -y python3-pip python3-venv pipx curl locales
|
|||
RUN pipx install lektor
|
||||
RUN mkdir -p /opt/lektor/project && mkdir -p /opt/lektor/output
|
||||
RUN sed -i '/de_DE.UTF-8/s/^# //g' /etc/locale.gen && locale-gen
|
||||
RUN sed -i '/fr_FR.UTF-8/s/^# //g' /etc/locale.gen && locale-gen
|
||||
|
||||
RUN python3 -m venv /opt/venv
|
||||
|
||||
|
|
|
|||
|
|
@ -8,4 +8,10 @@ tempfile=$(mktemp)
|
|||
sh -c ". /opt/venv/bin/activate && exec python /opt/lektor/scripts/calendar-fetcher-main.py ${CALENDAR_URL} > $tempfile"
|
||||
mv $tempfile /opt/lektor/project/content/contents.lr
|
||||
|
||||
sh -c ". /opt/venv/bin/activate && exec python /opt/lektor/scripts/calendar-fetcher+fr.py ${CALENDAR_URL} > /opt/lektor/project/content/termine/contents+fr.lr"
|
||||
# TODO As the file reads from the same file as it's output is afterwards piped into this leads to synchronization/buffering issues, we therefore write to a temporaray file and move it to the right place in a subsequent step
|
||||
tempfile=$(mktemp)
|
||||
sh -c ". /opt/venv/bin/activate && exec python /opt/lektor/scripts/calendar-fetcher-main+fr.py ${CALENDAR_URL} > $tempfile"
|
||||
mv $tempfile /opt/lektor/project/content/contents+fr.lr
|
||||
|
||||
exec "$@"
|
||||
|
|
@ -5,7 +5,7 @@ title: Bienvenue au PC Stammertal
|
|||
html:
|
||||
|
||||
<h2>Notre prochain événement: </h2><br>
|
||||
<div class="nextevent">Samstag <strong>22. März 9:00, Frühlingsschiessen</strong>, Bülach</div>
|
||||
<div class="nextevent">samedi <strong>22. mars 9:00, Frühlingsschiessen</strong>, Bülach</div>
|
||||
<div>
|
||||
<br>
|
||||
<br>
|
||||
|
|
|
|||
251
lektor/lektordata/project/content/termine/contents+fr.lr
Normal file
251
lektor/lektordata/project/content/termine/contents+fr.lr
Normal file
|
|
@ -0,0 +1,251 @@
|
|||
_model: page
|
||||
---
|
||||
title: Événements
|
||||
---
|
||||
body:
|
||||
|
||||
* <div>Frühlingsschiessen</div>
|
||||
* <div>sam. 22. mars 2025</div>
|
||||
* <div>9:00</div>
|
||||
* <div>Bülach</div>
|
||||
* <div>Standreinigung</div>
|
||||
* <div>sam. 22. mars 2025</div>
|
||||
* <div>9:00</div>
|
||||
* <div> </div>
|
||||
* <div>freies Training</div>
|
||||
* <div>lun. 31. mars 2025</div>
|
||||
* <div>18:00</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Eulachschiessen</div>
|
||||
* <div>ven. 4. avril 2025</div>
|
||||
* <div>16:00</div>
|
||||
* <div>Winterthur</div>
|
||||
* <div>Eulachschiessen</div>
|
||||
* <div>sam. 5. avril 2025</div>
|
||||
* <div>8:30</div>
|
||||
* <div>Winterthur</div>
|
||||
* <div>Eulachschiessen</div>
|
||||
* <div>ven. 11. avril 2025</div>
|
||||
* <div>16:00</div>
|
||||
* <div>Winterthur</div>
|
||||
* <div>Eulachschiessen</div>
|
||||
* <div>sam. 12. avril 2025</div>
|
||||
* <div>8:30</div>
|
||||
* <div>Winterthur</div>
|
||||
* <div>Kreis Winterschiessen</div>
|
||||
* <div>lun. 14. avril 2025</div>
|
||||
* <div>18:00</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Obligatorisches Programm und Kreis Winterschiessen</div>
|
||||
* <div>lun. 28. avril 2025</div>
|
||||
* <div>18:00</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>freies Training</div>
|
||||
* <div>lun. 5. mai 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>freies Training</div>
|
||||
* <div>lun. 12. mai 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Schlossschiessen</div>
|
||||
* <div>jeu. 15. mai 2025</div>
|
||||
* <div>17:30</div>
|
||||
* <div>Wülflingen</div>
|
||||
* <div>Schlossschiessen</div>
|
||||
* <div>ven. 16. mai 2025</div>
|
||||
* <div>17:30</div>
|
||||
* <div>Wülflingen</div>
|
||||
* <div>Schlossschiessen</div>
|
||||
* <div>sam. 17. mai 2025</div>
|
||||
* <div>9:30</div>
|
||||
* <div>Wülflingen</div>
|
||||
* <div>Obligatorisches Programm und freies Training</div>
|
||||
* <div>lun. 19. mai 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Schlossschiessen</div>
|
||||
* <div>jeu. 22. mai 2025</div>
|
||||
* <div>17:30</div>
|
||||
* <div>Wülflingen</div>
|
||||
* <div>Schlossschiessen</div>
|
||||
* <div>ven. 23. mai 2025</div>
|
||||
* <div>17:30</div>
|
||||
* <div>Wülflingen</div>
|
||||
* <div>freies Training</div>
|
||||
* <div>lun. 26. mai 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Kreis Sommerschiessen</div>
|
||||
* <div>lun. 2. juin 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Sommer-Schüsse</div>
|
||||
* <div>ven. 6. juin 2025</div>
|
||||
* <div>17:00</div>
|
||||
* <div>Rafz</div>
|
||||
* <div>Sommer-Schüsse</div>
|
||||
* <div>sam. 14. juin 2025</div>
|
||||
* <div>9:00</div>
|
||||
* <div>Rafz</div>
|
||||
* <div>Obligatorisches Programm und freies Training</div>
|
||||
* <div>lun. 16. juin 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>freies Training</div>
|
||||
* <div>lun. 23. juin 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>freies Training</div>
|
||||
* <div>lun. 30. juin 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Bezirks Sommerschiessen</div>
|
||||
* <div>sam. 5. juillet 2025</div>
|
||||
* <div>17:00</div>
|
||||
* <div>Flurlingen</div>
|
||||
* <div>freies Training</div>
|
||||
* <div>lun. 7. juillet 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Terassenfest</div>
|
||||
* <div>ven. 11. juillet 2025</div>
|
||||
* <div>19:00</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Bezirks Sommerschiessen</div>
|
||||
* <div>sam. 12. juillet 2025</div>
|
||||
* <div>14:00</div>
|
||||
* <div>Flurlingen</div>
|
||||
* <div>freies Training</div>
|
||||
* <div>lun. 14. juillet 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Obligatorisches Programm und freies Training</div>
|
||||
* <div>lun. 11. août 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>ZHKSF Ausbildung Pistole</div>
|
||||
* <div>jeu. 14. août 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div> </div>
|
||||
* <div>Zürcher Kantonalschützenfest</div>
|
||||
* <div>ven. 15. août 2025</div>
|
||||
* <div> </div>
|
||||
* <div> </div>
|
||||
* <div>Zürcher Kantonalschützenfest</div>
|
||||
* <div>sam. 16. août 2025</div>
|
||||
* <div> </div>
|
||||
* <div> </div>
|
||||
* <div>Zürcher Kantonalschützenfest</div>
|
||||
* <div>dim. 17. août 2025</div>
|
||||
* <div> </div>
|
||||
* <div> </div>
|
||||
* <div>KEIN Training</div>
|
||||
* <div>lun. 18. août 2025</div>
|
||||
* <div> </div>
|
||||
* <div> </div>
|
||||
* <div>ZHKSF intern</div>
|
||||
* <div>jeu. 21. août 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div> </div>
|
||||
* <div>Zürcher Kantonalschützenfest</div>
|
||||
* <div>ven. 22. août 2025</div>
|
||||
* <div> </div>
|
||||
* <div> </div>
|
||||
* <div>Zürcher Kantonalschützenfest</div>
|
||||
* <div>sam. 23. août 2025</div>
|
||||
* <div> </div>
|
||||
* <div> </div>
|
||||
* <div>Zürcher Kantonalschützenfest</div>
|
||||
* <div>dim. 24. août 2025</div>
|
||||
* <div> </div>
|
||||
* <div> </div>
|
||||
* <div>Zürcher Kantonalschützenfest</div>
|
||||
* <div>lun. 25. août 2025</div>
|
||||
* <div> </div>
|
||||
* <div> </div>
|
||||
* <div>Zürcher Kantonalschützenfest</div>
|
||||
* <div>ven. 29. août 2025</div>
|
||||
* <div> </div>
|
||||
* <div> </div>
|
||||
* <div>Zürcher Kantonalschützenfest</div>
|
||||
* <div>sam. 30. août 2025</div>
|
||||
* <div> </div>
|
||||
* <div> </div>
|
||||
* <div>Zürcher Kantonalschützenfest</div>
|
||||
* <div>dim. 31. août 2025</div>
|
||||
* <div> </div>
|
||||
* <div> </div>
|
||||
* <div>freies Training</div>
|
||||
* <div>lun. 1. septembre 2025</div>
|
||||
* <div>18:30</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Kreismatch</div>
|
||||
* <div>lun. 8. septembre 2025</div>
|
||||
* <div>18:00</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Orientierungslauf Stammerberg (kein Schiessbetrieb!)</div>
|
||||
* <div>sam. 13. septembre 2025</div>
|
||||
* <div> </div>
|
||||
* <div>Unterstammheim</div>
|
||||
* <div>Kreismatch</div>
|
||||
* <div>lun. 15. septembre 2025</div>
|
||||
* <div>18:00</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Schwaderlohschiessen</div>
|
||||
* <div>sam. 20. septembre 2025</div>
|
||||
* <div>13:30</div>
|
||||
* <div>Alterswilen</div>
|
||||
* <div>freies Training</div>
|
||||
* <div>lun. 22. septembre 2025</div>
|
||||
* <div>18:00</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Schwaderlohschiessen (unser Schiesstag)</div>
|
||||
* <div>mar. 23. septembre 2025</div>
|
||||
* <div>17:30</div>
|
||||
* <div>Alterswilen</div>
|
||||
* <div>Schwaderlohschiessen</div>
|
||||
* <div>sam. 27. septembre 2025</div>
|
||||
* <div>8:00</div>
|
||||
* <div>Alterswilen</div>
|
||||
* <div>Schwaderlohschiessen</div>
|
||||
* <div>dim. 28. septembre 2025</div>
|
||||
* <div>10:00</div>
|
||||
* <div>Alterswilen</div>
|
||||
* <div>Endschiessen</div>
|
||||
* <div>lun. 29. septembre 2025</div>
|
||||
* <div>17:00</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Endschiessen</div>
|
||||
* <div>lun. 6. octobre 2025</div>
|
||||
* <div>17:00</div>
|
||||
* <div>Stammheim</div>
|
||||
* <div>Niklausschiessen</div>
|
||||
* <div>sam. 25. octobre 2025</div>
|
||||
* <div>9:00</div>
|
||||
* <div>Diessenhofen</div>
|
||||
* <div>Niklausschiessen</div>
|
||||
* <div>dim. 26. octobre 2025</div>
|
||||
* <div>9:00</div>
|
||||
* <div>Diessenhofen</div>
|
||||
* <div>Absenden</div>
|
||||
* <div>ven. 31. octobre 2025</div>
|
||||
* <div>19:00</div>
|
||||
* <div> </div>
|
||||
* <div>Niklausschiessen</div>
|
||||
* <div>sam. 1. novembre 2025</div>
|
||||
* <div>9:00</div>
|
||||
* <div>Diessenhofen</div>
|
||||
* <div>Appenzeller Lupi Meisterschaft</div>
|
||||
* <div>dim. 11. janvier 2026</div>
|
||||
* <div>9:00</div>
|
||||
* <div> </div>
|
||||
* <div>Generalversammlung</div>
|
||||
* <div>jeu. 5. février 2026</div>
|
||||
* <div>19:00</div>
|
||||
* <div> </div>
|
||||
|
||||
---
|
||||
_template: page.html
|
||||
|
||||
|
||||
|
|
@ -39,7 +39,7 @@
|
|||
['/termine', 'Événements'],
|
||||
['/vorstand', 'Direction'],
|
||||
['/about', 'À propos'],
|
||||
['/kontakt', 'Contactez nous']
|
||||
['/kontakt', 'Contactez-nous']
|
||||
]
|
||||
} %}
|
||||
|
||||
|
|
|
|||
57
lektor/lektordata/scripts/calendar-fetcher+fr.py
Normal file
57
lektor/lektordata/scripts/calendar-fetcher+fr.py
Normal file
|
|
@ -0,0 +1,57 @@
|
|||
from datetime import datetime, date, timezone
|
||||
import locale
|
||||
import sys
|
||||
from calendarstuff import get_events
|
||||
|
||||
|
||||
def fetch_upcoming_events(ics_url):
|
||||
|
||||
events = get_events(ics_url, 'fr_FR.UTF-8')
|
||||
|
||||
for event in events:
|
||||
|
||||
start = event.get('dtstart').dt
|
||||
out_summary = event.get('summary')
|
||||
location = event.get('location', 'No location specified')
|
||||
|
||||
out_startdate = start.strftime("%a %-d. %B %Y")
|
||||
|
||||
# Format output based on whether it's an all-day event
|
||||
if isinstance(start, date) and not isinstance(start, datetime):
|
||||
out_starttime = " "
|
||||
else:
|
||||
out_starttime = start.strftime('%-H:%M')
|
||||
|
||||
if location != 'No location specified':
|
||||
out_location = location
|
||||
else:
|
||||
out_location = " "
|
||||
|
||||
print(f"* <div>{out_summary}</div> ")
|
||||
print(f" * <div>{out_startdate}</div> ")
|
||||
print(f" * <div>{out_starttime}</div> ")
|
||||
print(f" * <div>{out_location}</div> ")
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
|
||||
ics_url = sys.argv[1]
|
||||
|
||||
# ics_url = "https://backoffice.pc-stammertal.ch/remote.php/dav/public-calendars/RqLX5wj25aY6cpnP?export"
|
||||
|
||||
print("_model: page")
|
||||
print("---")
|
||||
print("title: Événements")
|
||||
print("---")
|
||||
print("body:")
|
||||
print("")
|
||||
# print("<center><a href=\"https://backoffice.pc-stammertal.ch/remote.php/dav/public-calendars/RqLX5wj25aY6cpnP?export\">Kalender Abonnieren</a></center>")
|
||||
# print("")
|
||||
# print("<center><a href=\"https://backoffice.pc-stammertal.ch/remote.php/dav/public-calendars/RqLX5wj25aY6cpnP?export\"><img src=\"/images/calendar.png\" alt=\"Link als QR code\"></a></center>")
|
||||
# print("")
|
||||
fetch_upcoming_events(ics_url)
|
||||
print("")
|
||||
print("---")
|
||||
print("_template: page.html")
|
||||
print("")
|
||||
print("")
|
||||
142
lektor/lektordata/scripts/calendar-fetcher-main+fr.py
Normal file
142
lektor/lektordata/scripts/calendar-fetcher-main+fr.py
Normal file
|
|
@ -0,0 +1,142 @@
|
|||
#!/usr/bin/env python3
|
||||
|
||||
import sys
|
||||
from pathlib import Path
|
||||
from datetime import datetime, date, timezone
|
||||
from typing import Optional, NamedTuple
|
||||
from dataclasses import dataclass
|
||||
from bs4 import BeautifulSoup
|
||||
import locale
|
||||
from calendarstuff import get_events
|
||||
|
||||
|
||||
@dataclass
|
||||
class EventDetails:
|
||||
weekday: str
|
||||
date: str
|
||||
time: str
|
||||
summary: str
|
||||
location: str
|
||||
|
||||
def to_html(self) -> str:
|
||||
return (
|
||||
f'<div class="nextevent">{self.weekday} '
|
||||
f'<strong>{self.date}{self.time}, {self.summary}</strong>'
|
||||
f'{self.location}</div>'
|
||||
)
|
||||
|
||||
class EventProcessor:
|
||||
def __init__(self, ics_url: str, content_file: str):
|
||||
self.ics_url = ics_url
|
||||
self.content_file = Path(content_file)
|
||||
self.fallback_html = (
|
||||
'<div class="nextevent">Malheureusement inconnu, mais '
|
||||
'<strong>demande à la direction</strong> qui devrait le savoir</div>'
|
||||
)
|
||||
|
||||
@property
|
||||
def fallback_content(self) -> str:
|
||||
return f"""_model: htmlpage
|
||||
---
|
||||
title: Bienvenue au PC Stammertal
|
||||
---
|
||||
html:
|
||||
|
||||
<h2>Notre prochain événement: </h2><br>
|
||||
{self.fallback_html}
|
||||
<div>
|
||||
<br>
|
||||
<br>
|
||||
<br>
|
||||
</div>
|
||||
<h3>...et ne manquez pas non plus :</h3>
|
||||
<a href="https://www.wyland25.ch/" target="_blank"><img src="/images/zhksf.png" alt="27. Zürcher Kantonalschützenfest 2025" class="stamp"></a>
|
||||
<div class="threecolumn">
|
||||
<div>
|
||||
<a href="termine/"><img src=" /images/termine_square.jpg" alt="Calendrier"> Tous nos événements</a>
|
||||
</div>
|
||||
<div>
|
||||
<a href="about/"><img src="/images/about_square.jpg" alt="Livre"> À propos de nous</a>
|
||||
</div>
|
||||
<div>
|
||||
<a href="kontakt/"><img src="/images/kontakt_square.jpg" alt="Lettres"> Contactez-nous</a>
|
||||
</div>
|
||||
</div>
|
||||
---
|
||||
_template:
|
||||
|
||||
page.html
|
||||
"""
|
||||
|
||||
def setup_locale(self) -> None:
|
||||
try:
|
||||
locale.setlocale(locale.LC_TIME, 'de_DE.UTF-8')
|
||||
except locale.Error as e:
|
||||
print(f"Warning: Failed to set locale: {e}", file=sys.stderr)
|
||||
|
||||
def format_event_time(self, start: datetime | date) -> str:
|
||||
return "" if isinstance(start, date) and not isinstance(start, datetime) else start.strftime(" %-H:%M")
|
||||
|
||||
def get_next_event(self) -> Optional[EventDetails]:
|
||||
try:
|
||||
|
||||
events = get_events(self.ics_url, 'fr_FR.UTF-8')
|
||||
|
||||
event = events[0]
|
||||
start = event.get('dtstart').dt
|
||||
|
||||
return EventDetails(
|
||||
weekday=start.strftime("%A"),
|
||||
date=start.strftime("%-d. %B"),
|
||||
time=self.format_event_time(start),
|
||||
summary=event.get('summary', ''),
|
||||
location=f", {event.get('location')}" if event.get('location') else ""
|
||||
)
|
||||
|
||||
except (StopIteration, AttributeError) as e:
|
||||
print(f"No upcoming events found: {e}", file=sys.stderr)
|
||||
return None
|
||||
|
||||
def read_current_content(self) -> tuple[str, str]:
|
||||
try:
|
||||
content = self.content_file.read_text()
|
||||
soup = BeautifulSoup(content, 'html.parser')
|
||||
events = soup.find_all('div', {'class': 'nextevent'})
|
||||
return content, str(events[0]) if len(events) == 1 else ""
|
||||
|
||||
except (IOError, IndexError) as e:
|
||||
print(f"Error reading content file: {e}", file=sys.stderr)
|
||||
return "", ""
|
||||
|
||||
def process(self) -> str:
|
||||
self.setup_locale()
|
||||
|
||||
content, source_str = self.read_current_content()
|
||||
if not content:
|
||||
return self.fallback_content
|
||||
|
||||
if len(source_str) > 0 and source_str in content:
|
||||
event = self.get_next_event()
|
||||
if not event:
|
||||
return content.replace(source_str, self.fallback_html).rstrip()
|
||||
|
||||
return content.replace(source_str, event.to_html()).rstrip()
|
||||
|
||||
return self.fallback_content
|
||||
|
||||
def main() -> None:
|
||||
if len(sys.argv) != 2:
|
||||
print(f"Usage: {sys.argv[0]} <ICS_URL>", file=sys.stderr)
|
||||
sys.exit(1)
|
||||
|
||||
processor = EventProcessor(
|
||||
ics_url=sys.argv[1],
|
||||
content_file="/opt/lektor/project/content/contents+fr.lr"
|
||||
)
|
||||
|
||||
print(processor.process(), end='')
|
||||
print("")
|
||||
print("")
|
||||
|
||||
if __name__ == "__main__":
|
||||
main()
|
||||
|
|
@ -80,7 +80,7 @@ page.html
|
|||
def get_next_event(self) -> Optional[EventDetails]:
|
||||
try:
|
||||
|
||||
events = get_events(self.ics_url)
|
||||
events = get_events(self.ics_url, 'de_DE.UTF-8')
|
||||
|
||||
event = events[0]
|
||||
start = event.get('dtstart').dt
|
||||
|
|
|
|||
|
|
@ -6,7 +6,7 @@ from calendarstuff import get_events
|
|||
|
||||
def fetch_upcoming_events(ics_url):
|
||||
|
||||
events = get_events(ics_url)
|
||||
events = get_events(ics_url, 'de_DE.UTF-8')
|
||||
|
||||
for event in events:
|
||||
|
||||
|
|
|
|||
|
|
@ -56,9 +56,9 @@ def split_multiday_events(events):
|
|||
return split_events
|
||||
|
||||
|
||||
def get_events(ics_url):
|
||||
def get_events(ics_url, tgt_locale):
|
||||
# Set German locale
|
||||
locale.setlocale(locale.LC_TIME, 'de_DE.UTF-8')
|
||||
locale.setlocale(locale.LC_TIME, tgt_locale)
|
||||
|
||||
response = requests.get(ics_url)
|
||||
calendar = Calendar.from_ical(response.content)
|
||||
|
|
|
|||
Loading…
Add table
Reference in a new issue