Skip to content

Commit

Permalink
Fixes bug when date-time is recognized as time
Browse files Browse the repository at this point in the history
Date-time was recognized incorrectly as a date or time. This resulted
in wrong representation of some iCalendar strings.

Also adds "errors" list in Component for saving error strings from parsing.

collective#174
collective#168
  • Loading branch information
stlaz committed Dec 18, 2015
1 parent ff1f2ee commit 118169d
Show file tree
Hide file tree
Showing 5 changed files with 30 additions and 19 deletions.
1 change: 1 addition & 0 deletions docs/credits.rst
Original file line number Diff line number Diff line change
Expand Up @@ -36,6 +36,7 @@ icalendar contributors
- Ronan Dunklau <ronan@dunklau.fr>
- Russ <russ@rw.id.au>
- Sidnei da Silva <sidnei@enfoldsystems.com>
- Stanislav Láznička <slaznick@redhat.com>
- Stanislav Ochotnicky <sochotnicky@redhat.com>
- Stefan Schwarzer <sschwarzer@sschwarzer.net>
- Thomas Bruederli <thomas@roundcube.net>
Expand Down
16 changes: 8 additions & 8 deletions src/icalendar/cal.py
Original file line number Diff line number Diff line change
Expand Up @@ -80,8 +80,8 @@ def __init__(self, *args, **kwargs):
super(Component, self).__init__(*args, **kwargs)
# set parameters here for properties that use non-default values
self.subcomponents = [] # Components can be nested.
self.is_broken = False # True if we ignored an exception while
# parsing a property
self.errors = list() # If we ignored exception(s) while
# parsing a property, contains error strings

# def is_compliant(self, name):
# """Returns True is the given property name is compliant with the
Expand Down Expand Up @@ -309,14 +309,14 @@ def from_ical(cls, st, multiple=False):

try:
name, params, vals = line.parts()
except ValueError:
except ValueError as e:
# if unable to parse a line within a component
# that ignores exceptions, mark the component
# as broken and skip the line. otherwise raise.
component = stack[-1] if stack else None
if not component or not component.ignore_exceptions:
raise
component.is_broken = True
component.errors.append((None, str(e)))
continue

uname = name.upper()
Expand All @@ -338,8 +338,7 @@ def from_ical(cls, st, multiple=False):
if not stack: # we are at the end
comps.append(component)
else:
if not component.is_broken:
stack[-1].add_component(component)
stack[-1].add_component(component)
if vals == 'VTIMEZONE' and \
'TZID' in component and \
component['TZID'] not in pytz.all_timezones and \
Expand All @@ -356,10 +355,11 @@ def from_ical(cls, st, multiple=False):
vals = factory(factory.from_ical(vals, params['TZID']))
else:
vals = factory(factory.from_ical(vals))
except ValueError:
except ValueError as e:
if not component.ignore_exceptions:
raise
component.is_broken = True
component.errors.append((uname, str(e)))
component.add(name, None, encode=0)
else:
vals.params = params
component.add(name, vals, encode=0)
Expand Down
14 changes: 8 additions & 6 deletions src/icalendar/prop.py
Original file line number Diff line number Diff line change
Expand Up @@ -313,13 +313,15 @@ def from_ical(cls, ical, timezone=None):
u = ical.upper()
if u.startswith(('P', '-P', '+P')):
return vDuration.from_ical(ical)
try:

if len(ical) in (15, 16):
return vDatetime.from_ical(ical, timezone=timezone)
except ValueError:
try:
return vDate.from_ical(ical)
except ValueError:
return vTime.from_ical(ical)
elif len(ical) == 8:
return vDate.from_ical(ical)
elif len(ical) in (6,7):
return vTime.from_ical(ical)
else:
raise ValueError("Expected datetime, date, or time, got: '%s'" % ical)


class vDate(object):
Expand Down
4 changes: 3 additions & 1 deletion src/icalendar/tests/test_fixed_issues.py
Original file line number Diff line number Diff line change
Expand Up @@ -200,7 +200,9 @@ def test_issue_104__ignore_exceptions(self):
END:VEVENT"""
event = icalendar.Calendar.from_ical(ical_str)
self.assertTrue(isinstance(event, icalendar.Event))
self.assertTrue(event.is_broken)
self.assertEqual(event.errors,
[(None, 'Content line could not be parsed into parts: u\'X\': Invalid content line')]
)

def test_issue_104__no_ignore_exceptions(self):
"""
Expand Down
14 changes: 10 additions & 4 deletions src/icalendar/tests/test_unit_cal.py
Original file line number Diff line number Diff line change
Expand Up @@ -389,8 +389,9 @@ def test_cal_Calendar(self):
directory = tempfile.mkdtemp()
open(os.path.join(directory, 'test.ics'), 'wb').write(cal.to_ical())

# Parsing a complete calendar from a string will silently ignore bogus
# events. The bogosity in the following is the third EXDATE: it has an
# Parsing a complete calendar from a string will silently ignore wrong
# events but adding the error information to the component's 'errors'
# attribute. The error in the following is the third EXDATE: it has an
# empty DATE.
s = '\r\n'.join(('BEGIN:VCALENDAR',
'PRODID:-//Google Inc//Google Calendar 70.9054//EN',
Expand All @@ -405,7 +406,7 @@ def test_cal_Calendar(self):
'EXDATE;VALUE=DATE:20080311',
'END:VEVENT',
'BEGIN:VEVENT',
'DESCRIPTION:Bogus event',
'DESCRIPTION:Wrong event',
'DTSTART;VALUE=DATE:20080303',
'DTEND;VALUE=DATE:20080304',
'RRULE:FREQ=DAILY;UNTIL=20080323T235959Z',
Expand All @@ -416,4 +417,9 @@ def test_cal_Calendar(self):
self.assertEqual(
[e['DESCRIPTION'].to_ical()
for e in icalendar.cal.Calendar.from_ical(s).walk('VEVENT')],
[b'Perfectly OK event'])
[b'Perfectly OK event', b'Wrong event'])
self.assertEqual(
[e.errors
for e in icalendar.cal.Calendar.from_ical(s).walk('VEVENT')],
[[], [('EXDATE', "Expected datetime, date, or time, got: ''")]]
)

0 comments on commit 118169d

Please sign in to comment.