1

我正在使用 vobject 在 Django 中创建一个 ical 事件。我在使用较低级别的代码时遇到问题。看起来 ical 正在尝试使用obj.add(TimezoneComponent(tzinfo=getTzid(tzid))). 但后来我raise NonExistentTimeError(dt)从 pytz 那里得到。关于做什么的任何建议?当我使用变量 start1 中的打印语句查看它们时,年、月、日显示正确。

 File "/home/git/chrono/chrono/requests_app/views.py", line 110, in form_valid
    ics_form = create_ics(data)
  File "/home/git/chrono/chrono/requests_app/views.py", line 126, in create_ics
    response = HttpResponse(cal.serialize(), content_type='text/calendar')
  File "/home/one/.virtualenvs/chronos/local/lib/python2.7/site-packages/vobject/base.py", line 186, in serialize
    return behavior.serialize(self, buf, lineLength, validate)
  File "/home/one/.virtualenvs/chronos/local/lib/python2.7/site-packages/vobject/behavior.py", line 147, in serialize
    cls.generateImplicitParameters(obj)
  File "/home/one/.virtualenvs/chronos/local/lib/python2.7/site-packages/vobject/icalendar.py", line 853, in generateImplicitParameters
    obj.add(TimezoneComponent(tzinfo=getTzid(tzid)))
  File "/home/one/.virtualenvs/chronos/local/lib/python2.7/site-packages/vobject/icalendar.py", line 75, in __init__
    self.tzinfo = tzinfo
  File "/home/one/.virtualenvs/chronos/local/lib/python2.7/site-packages/vobject/base.py", line 468, in __setattr__
    prop.fset(self, value)
  File "/home/one/.virtualenvs/chronos/local/lib/python2.7/site-packages/vobject/icalendar.py", line 145, in settzinfo
    transition = getTransition(transitionTo, year, tzinfo)
  File "/home/one/.virtualenvs/chronos/local/lib/python2.7/site-packages/vobject/icalendar.py", line 1856, in getTransition
    uncorrected = firstTransition(generateDates(year, month, day), test)
  File "/home/one/.virtualenvs/chronos/local/lib/python2.7/site-packages/vobject/icalendar.py", line 1816, in firstTransition
    if not test(dt):
  File "/home/one/.virtualenvs/chronos/local/lib/python2.7/site-packages/vobject/icalendar.py", line 1843, in test
    def test(dt): return tzinfo.dst(dt) != zeroDelta
  File "/home/one/.virtualenvs/chronos/local/lib/python2.7/site-packages/pytz/tzinfo.py", line 445, in dst
    dt = self.localize(dt, is_dst)
  File "/home/one/.virtualenvs/chronos/local/lib/python2.7/site-packages/pytz/tzinfo.py", line 327, in localize
    raise NonExistentTimeError(dt)
NonExistentTimeError: 2000-04-02 02:00:00



def create_ics(data):
    start1 = data['date_due']
    print start1.day
    start2 = datetime.datetime(start1.year, start1.month, start1.day)
    start3 = data['action']
    cal = vobject.iCalendar()
    cal.add('method').value = 'PUBLISH'
    vevent = cal.add('vevent')
    vevent.add('dtstart').value = start1
    vevent.add('dtend').value = start2
    vevent.add('dtstamp').value = datetime.datetime.now()
    vevent.add('summary').value = data['action'].name
    response = HttpResponse(cal.serialize(), content_type='text/calendar')
    response['Filename'] = 'filename.ics'
    response['Content-Disposition'] = 'attachment; filename=filename.ics'
    return response

从模型中,日期时间字段:

date_due = models.DateTimeField()

更新:

发现我必须放置:

>>> utc = vobject.icalendar.utc
>>> start = cal.vevent.add('dtstart')
>>> start.value = datetime.datetime(2006, 2, 16, tzinfo = utc)

进入它,它起作用了。

4

1 回答 1

2

简短回答: vobject(从 0.9.2 开始)与 pytz 不兼容。因此,在尝试序列化之前,请确保vobject iCalendar 中的每个.astimezone(pytz.utc)日期时间都已转换为 UTC,使用类似.

(这是每个dtstart、dtend、dtstamp、created、last-modified 以及我忘记的其他一些 vevent 字段。)

长答案: vobject 尝试为非 UTC 日期时间做正确的事情,但遇到了 pytz 的麻烦。“正确的事情”来自指定 iCalendar的RFC 5545 :

  1. 使用DATE-TIME 表格 #3 “带有本地时间和时区参考的日期”表示日期时间。这可能类似于DTSTART;TZID=America/New_York:20160714T133000——注意事件时区的 TZID。

  2. 为您的活动中使用的每个唯一 TZID添加一个VTIMEZONE块到您的 iCalendar。这是该时区的完整定义:如何计算可能出现的任何日期时间的该时区与 UTC 的偏移量,包括夏令时规则。(RFC 5545没有指定任何特定的时区名称,因此您必须在 iCalendar 本身中包含时区定义。vobject 会自动为您执行此操作。)

为了弄清楚时区转换规则,vobject搜索“所有时间”(默认年份 2000-2030),寻找时区与 UTC 的偏移量的变化。这就是问题所在,因为 vobject 代码不处理 pytz 的无效时间错误。

2000 年 4 月 2 日凌晨 2:00 是 2000 年到 2030 年之间的第一个 DST 转换,这就是为什么您会收到关于那个时间的错误(即使您没有在自己的代码中的任何地方使用它)。

选项:

  • 如果您不想要一天中的特定时间(如原始问题中所示),请使用date而不是。datetime日期没有时区,因此这些都不适用。(并且 vobject 可以很好地处理日期。)
  • 将所有转换datetime为 UTC 中的感知日期时间。UTC 不需要 VTIMEZONE 定义。
  • 使用dateutil 时区而不是 pytz。例如,from dateutil import tz; ... tzinfo=tz.gettz('America/Los_Angeles')。由于 dateutil 是 vobject 依赖项,我认为这是 vobject 的 VTIMEZONE 生成器设计的格式。(但尚未进行广泛测试。此外,gettz 需要在您的机器上安装 tzdb 文件,因此不完全可移植。)
  • 为您使用的每个 TZID 添加您自己的 VTIMEZONE 定义到 iCalendar,这应该避免 vobject 中出现问题的自动时区生成代码。(未经测试。在一般情况下很难正确处理。)
  • 提交PR 以修复 vobject 以使用 pytz
于 2016-05-26T23:07:09.043 回答