Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix 1367: :khal crashes with zoneinfo attrib error adding new event #1374

Open
wants to merge 2 commits into
base: master
Choose a base branch
from

Conversation

syasma
Copy link

@syasma syasma commented Dec 1, 2024

  1. Managing compatibility between pytz and zoneinfo.ZoneInfo
    The original code was designed to work only with pytz.BaseTzInfo objects, which caused problems when zoneinfo.ZoneInfo objects (introduced with Python 3.9) were used. The solution was to dynamically check which type of timezone object was being used (pytz or zoneinfo), and treat each case differently.

Key changes :
Check timezone type:
If the timezone is a pytz instance, the code applies the methods adapted to pytz to extract time transition information (e.g.: tz._utc_transition_times).
If the timezone is an instance of zoneinfo.ZoneInfo, the code must handle the retrieval of transition information differently, as zoneinfo does not contain the same internal attributes (such as _tzinfos and _utc_transition_times).

  1. Adapting the create_timezone function
    The create_timezone function has been refactored to accept both pytz and zoneinfo.ZoneInfo objects as arguments and to generate a VTIMEZONE component based on the transition information of both timezone types.

Key changes :
zoneinfo.ZoneInfo support: logic that depended on pytz internal attributes (such as _tzinfos and _utc_transition_times) has been replaced by zoneinfo-compatible management. For example, for zoneinfo, the _transitions() method can be used to obtain time transitions, although this is not directly supported as with pytz.
VTIMEZONE generation for both timezone types: Whether the timezone is of the pytz or zoneinfo type, a VTIMEZONE component is created, including the necessary information such as TZNAME, DTSTART, and offsets (start and end) for each time transition.

  1. Dynamic start_date management
    For each timezone, a VTIMEZONE must include time transitions relating to a specific period of time (often associated with the first and last date of an event). The code has been adjusted to dynamically retrieve the event start_date if available. This date is then used to correctly generate the relevant time transitions.

Key change:
Dynamic management of start_date: If self.start (the event start date) is defined, it is used to calculate the period during which time transitions should be generated. Otherwise, the current date is used as the default value.

  1. Refactoring the raw method
    The raw method generates the complete VCALENDAR file, including the VTIMEZONE components for each event. It has been updated to include timezone checks (pytz and zoneinfo), as well as adding VTIMEZONE components only if the timezone is not UTC.

Key change:
Timezone collection: The method scans events to collect all unique timezones (those associated with DTSTART and DTEND). It then adds the corresponding VTIMEZONE components to the calendar, by calling the create_timezone function.
Adding events (VEVENT): After adding the VTIMEZONE components, the events themselves are added to the calendar.

Copy link
Member

@WhyNotHugo WhyNotHugo left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I created an event without this patch and with this patch. The serialised timezones are not the same.

Without this patch:

BEGIN:VTIMEZONE
TZID:Europe/Amsterdam
BEGIN:STANDARD
DTSTART:20241027T020000
TZNAME:CET
TZOFFSETFROM:+0200
TZOFFSETTO:+0100
END:STANDARD
BEGIN:DAYLIGHT
DTSTART:20250330T030000
TZNAME:CEST
TZOFFSETFROM:+0100
TZOFFSETTO:+0200
END:DAYLIGHT
END:VTIMEZONE

With this patch:

BEGIN:VTIMEZONE
TZID:Europe/Amsterdam
BEGIN:STANDARD
DTSTART;TZID=Europe/Amsterdam:20241203T100000
TZNAME:1:00:00
TZOFFSETFROM:+0100
TZOFFSETTO:+0100
END:STANDARD
END:VTIMEZONE

diff:

@@ -1,9 +1,15 @@
 BEGIN:VTIMEZONE
 TZID:Europe/Amsterdam
 BEGIN:STANDARD
-DTSTART;TZID=Europe/Amsterdam:20241203T100000
-TZNAME:1:00:00
-TZOFFSETFROM:+0100
+DTSTART:20241027T020000
+TZNAME:CET
+TZOFFSETFROM:+0200
 TZOFFSETTO:+0100
 END:STANDARD
+BEGIN:DAYLIGHT
+DTSTART:20250330T030000
+TZNAME:CEST
+TZOFFSETFROM:+0100
+TZOFFSETTO:+0200
+END:DAYLIGHT
 END:VTIMEZONE

From the definition for VTIMEZONE in https://www.rfc-editor.org/rfc/rfc5545#section-3.6.5:

The mandatory "DTSTART" property gives the effective onset date
and local time for the time zone sub-component definition.
"DTSTART" in this usage MUST be specified as a date with a local
time value.

If I understand correctly, the new output seems to be valid. It's accepted and rendered properly with Fastmail, no idea about other clients.

The TZNAMEs are lost, but I'm not sure how important they are of if any client actually renders these anyway.

@property
def raw(self) -> str:
"""Creates a VCALENDAR containing VTIMEZONEs
"""
"""Creates a VCALENDAR containing VTIMEZONEs."""
calendar = self._create_calendar()
tzs = []
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

non-blocking nitpick: I would use a set here, so we can just insert new entries without having any duplicates (inserting an existing element into a set is a no-op):

Suggested change
tzs = []
tzs = {}

tz: pytz.BaseTzInfo,
first_date: Optional[dt.datetime]=None,
last_date: Optional[dt.datetime]=None
tz,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please leave type hints here:

Suggested change
tz,
tz: pytz.tzinfo | zoneinfo.ZoneInfo,

Comment on lines +946 to +947
first_date = dt.datetime.today() if not first_date else first_date
last_date = first_date + dt.timedelta(days=1) if not last_date else last_date
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm not entirely familiar with this code. Why did you remove to_naive_utc here?

for subcomp in timezones.values():
elif isinstance(tz, pytz.BaseTzInfo):
# Use pytz's transitions
transitions = tz._utc_transition_times # Avoid using internal attributes directly.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
transitions = tz._utc_transition_times # Avoid using internal attributes directly.
transitions = tz._utc_transition_times # TODO: Avoid using internal attributes directly.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants