Enable sychronization of events
This commit is contained in:
parent
db10fb13e2
commit
b68394ec68
@ -1,5 +1,6 @@
|
||||
import datetime as dt
|
||||
import re, logging
|
||||
import zoneinfo
|
||||
|
||||
from .sync import Event
|
||||
|
||||
@ -15,21 +16,26 @@ class IcalHelper:
|
||||
'Sun': 6
|
||||
}
|
||||
|
||||
tzDE = zoneinfo.ZoneInfo('Europe/Berlin')
|
||||
deltaWeek = dt.timedelta(weeks=1)
|
||||
deltaDay = dt.timedelta(days=1)
|
||||
firstDay = dt.date(year=dt.datetime.now().year, month=1, day=1)
|
||||
|
||||
def __init__(self):
|
||||
self._l = logging.getLogger(__name__)
|
||||
self.dDay = dt.timedelta(days=1)
|
||||
self.firstDay = dt.datetime(year=dt.datetime.now().year, month=1, day=1)
|
||||
self.reTime = re.compile(r'(\d{1,2}):(\d{2})')
|
||||
|
||||
def getStart(self, event: Event, holidays):
|
||||
self._getFirstWeekdayInYear(event.day)
|
||||
self._getFirstOccurence(event, holidays)
|
||||
pass
|
||||
firstDay = self._getFirstOccurence(event, holidays)
|
||||
|
||||
if firstDay is None:
|
||||
return None, None
|
||||
return firstDay, dt.datetime.combine(firstDay, self._getTime(event.start), tzinfo=self.tzDE)
|
||||
|
||||
def _getFirstWeekdayInYear(self, weekday):
|
||||
candidate = self.firstDay
|
||||
while candidate.weekday() != self._WEEKDAY_MAP[weekday]:
|
||||
candidate += self.dDay
|
||||
candidate += self.deltaDay
|
||||
|
||||
self._l.log(5, 'First %s in year is %s', weekday, candidate)
|
||||
|
||||
@ -41,6 +47,72 @@ class IcalHelper:
|
||||
def _getSortedFeasts(self, holidays):
|
||||
return sorted(holidays['feasts'])
|
||||
|
||||
def _isFeast(self, candidate, feasts):
|
||||
return candidate in feasts
|
||||
|
||||
def _isHoliday(self, candidate, holidays):
|
||||
for h in holidays:
|
||||
if h['from'] <= candidate <= h['to']:
|
||||
return True
|
||||
|
||||
return False
|
||||
|
||||
def _getLastHolidayException(self, holidays):
|
||||
holidayList = list(self._getSortedHolidays(holidays))
|
||||
holidayList.reverse()
|
||||
lastHoliday = holidayList[0]
|
||||
lastHolidayDay = lastHoliday['to']
|
||||
|
||||
feastList = list(self._getSortedFeasts(holidays))
|
||||
feastList.reverse()
|
||||
lastFeast = feastList[0]
|
||||
lastFeastDay = lastFeast
|
||||
|
||||
return max(lastHolidayDay, lastFeastDay)
|
||||
|
||||
def _getFirstOccurence(self, event: Event, holidays):
|
||||
firstWeekday = self._getFirstWeekdayInYear(event.day)
|
||||
firstWeekday.tzinfo = dt.timezone.tzname('Europe/Berlin')
|
||||
|
||||
|
||||
candidate = firstWeekday
|
||||
while candidate.year == firstWeekday.year:
|
||||
if self._isFeast(candidate, holidays['feasts']):
|
||||
self._l.log(5, 'Found feast %s', candidate)
|
||||
candidate += self.deltaWeek
|
||||
continue
|
||||
|
||||
if self._isHoliday(candidate, holidays['holidays']):
|
||||
self._l.log(5, 'Found holiday %s', candidate)
|
||||
candidate += self.deltaWeek
|
||||
continue
|
||||
|
||||
self._l.log(5, 'Found first occurence %s', candidate)
|
||||
return candidate
|
||||
|
||||
return None
|
||||
|
||||
# firstWeekday.tzinfo = dt.timezone.tzname('Europe/Berlin')
|
||||
|
||||
def _getTime(self, startStr):
|
||||
match = self.reTime.match(startStr)
|
||||
return dt.time(int(match.group(1)), int(match.group(2)))
|
||||
|
||||
def getHolidayExceptions(self, event: Event, startDay, holidays):
|
||||
lastHolidayDay = self._getLastHolidayException(holidays)
|
||||
|
||||
pointer = startDay
|
||||
exeptions = []
|
||||
while pointer <= lastHolidayDay:
|
||||
ex = dt.datetime.combine(pointer, self._getTime(event.start), tzinfo=self.tzDE)
|
||||
|
||||
if self._isFeast(pointer, holidays['feasts']):
|
||||
exeptions.append(ex)
|
||||
|
||||
if self._isHoliday(pointer, holidays['holidays']) and not event.external:
|
||||
exeptions.append(ex)
|
||||
|
||||
pointer += self.deltaWeek
|
||||
|
||||
if len(exeptions) > 0:
|
||||
return exeptions
|
||||
return None
|
||||
|
@ -1,9 +1,9 @@
|
||||
import logging, yaml, pprint, json, re
|
||||
import hashlib
|
||||
import hashlib, uuid
|
||||
import caldav
|
||||
import datetime as dt
|
||||
|
||||
from . import login
|
||||
from . import login, debug
|
||||
|
||||
_l = logging.getLogger(__name__)
|
||||
|
||||
@ -50,7 +50,8 @@ class Event:
|
||||
|
||||
def __repr__(self):
|
||||
wAge = f' ({self.age})' if self.age is not None else ''
|
||||
return f'Ev({self.title}{wAge}) [{self.day}, {self.start}, {self.duration}]'
|
||||
ext = 'Ext' if self.external else ''
|
||||
return f'{ext}Ev({self.title}{wAge}) [{self.day}, {self.start}, {self.duration}]'
|
||||
|
||||
def getHash(self, holidays):
|
||||
def fixHolidays(holidays):
|
||||
@ -86,10 +87,22 @@ class Event:
|
||||
# pprint.pprint(data, indent=4, width=100)
|
||||
hasher = hashlib.sha1()
|
||||
hasher.update(json.dumps(data, sort_keys=True).encode('utf-8'))
|
||||
return hasher.hexdigest()
|
||||
first = hasher.hexdigest()
|
||||
second = uuid.uuid4().hex
|
||||
return f'{first}___{second}', first
|
||||
|
||||
dDay = dt.timedelta(days=1)
|
||||
|
||||
_MAP_WEEKDAY_ICAL = {
|
||||
'Mon': 'MO',
|
||||
'Tue': 'TU',
|
||||
'Wed': 'WE',
|
||||
'Thu': 'TH',
|
||||
'Fri': 'FR',
|
||||
'Sat': 'SA',
|
||||
'Sun': 'SU',
|
||||
}
|
||||
|
||||
def addToCalendar(self, calendar: caldav.Calendar, holidays):
|
||||
from . import cal_helper
|
||||
|
||||
@ -99,15 +112,44 @@ class Event:
|
||||
pass
|
||||
|
||||
helper = cal_helper.IcalHelper()
|
||||
helper.getStart(self, holidays)
|
||||
startDay, start = helper.getStart(self, holidays)
|
||||
|
||||
if start is None:
|
||||
_l.warning('Could not get start time for event %s in the current year', self)
|
||||
return
|
||||
|
||||
uid, uidFirst = self.getHash(holidays)
|
||||
icalData = {
|
||||
'uid': self.getHash(holidays),
|
||||
'DTSTART': dt.datetime.now(),
|
||||
'DTEND': dt.datetime.now() + dt.timedelta(minutes=self.duration),
|
||||
'SUMMARY': self.title
|
||||
'uid': uid,
|
||||
'DTSTART': start,
|
||||
'DTEND': start + dt.timedelta(minutes=self.duration),
|
||||
'SUMMARY': self.title,
|
||||
'RRULE': {
|
||||
'FREQ': 'DAILY',
|
||||
'BYDAY': self._MAP_WEEKDAY_ICAL[self.day]
|
||||
}
|
||||
}
|
||||
# calendar.add_event(**icalData)
|
||||
# debug.debugger()
|
||||
|
||||
holidayExceptions = helper.getHolidayExceptions(self, startDay, holidays)
|
||||
if holidayExceptions is not None:
|
||||
icalData['EXDATE'] = holidayExceptions
|
||||
|
||||
desc = ''
|
||||
if self.age is not None:
|
||||
desc += f'Jahrgänge: {self.age}'
|
||||
if len(self.description) > 0:
|
||||
desc += '\n\n'
|
||||
desc += self.description
|
||||
if len(desc) > 0:
|
||||
icalData['DESCRIPTION'] = desc
|
||||
|
||||
_l.log(5, 'Adding event\n%s', pprint.pformat(icalData, indent=4, width=100))
|
||||
|
||||
ical = caldav.lib.vcal.create_ical(**icalData)
|
||||
_l.log(5, 'Created event\n%s', pprint.pformat(ical, indent=4, width=100))
|
||||
_l.log(5, 'Created event\n%s', ical)
|
||||
calendar.add_event(**icalData)
|
||||
|
||||
def _unpackSchedules(schedule):
|
||||
def packSingleCalendar(cal):
|
||||
@ -121,7 +163,7 @@ def _unpackSchedules(schedule):
|
||||
|
||||
if 'age' in ev:
|
||||
e.age = ev['age']
|
||||
if ev.get('extern', False):
|
||||
if ev.get('extern', False) or ev.get('external', False):
|
||||
e.external = True
|
||||
if 'desc' in ev:
|
||||
e.description = ev['desc']
|
||||
@ -216,7 +258,8 @@ class CalendarSynchonizer:
|
||||
ret = {}
|
||||
|
||||
for e in events:
|
||||
uid = str(e.icalendar_component['uid'])
|
||||
uidRaw = str(e.icalendar_component['uid'])
|
||||
uid = uidRaw.split('___', 1)[0]
|
||||
_l.log(5, 'Event with uid %s was found.', uid)
|
||||
ret[uid] = e
|
||||
|
||||
@ -228,9 +271,9 @@ class CalendarSynchonizer:
|
||||
ret = {}
|
||||
|
||||
for e in schedule:
|
||||
uid = e.getHash(holidays)
|
||||
_l.log(5, 'Event with uid %s was found.', uid)
|
||||
ret[uid] = e
|
||||
uid, uidFirst = e.getHash(holidays)
|
||||
_l.log(5, 'Event with uid part %s was found.', uidFirst)
|
||||
ret[uidFirst] = e
|
||||
|
||||
return ret
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user