tilia-vobject 4.0.0.pre.alpha2
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitignore +19 -0
- data/.rubocop.yml +32 -0
- data/.simplecov +4 -0
- data/.travis.yml +3 -0
- data/CHANGELOG.sabre.md +626 -0
- data/CONTRIBUTING.md +25 -0
- data/Gemfile +17 -0
- data/Gemfile.lock +68 -0
- data/LICENSE +27 -0
- data/LICENSE.sabre +27 -0
- data/README.md +63 -0
- data/Rakefile +17 -0
- data/bin/vobject +7 -0
- data/lib/tilia/v_object/birthday_calendar_generator.rb +142 -0
- data/lib/tilia/v_object/cli.rb +582 -0
- data/lib/tilia/v_object/component/available.rb +107 -0
- data/lib/tilia/v_object/component/v_alarm.rb +114 -0
- data/lib/tilia/v_object/component/v_availability.rb +128 -0
- data/lib/tilia/v_object/component/v_calendar.rb +468 -0
- data/lib/tilia/v_object/component/v_card.rb +457 -0
- data/lib/tilia/v_object/component/v_event.rb +127 -0
- data/lib/tilia/v_object/component/v_free_busy.rb +81 -0
- data/lib/tilia/v_object/component/v_journal.rb +75 -0
- data/lib/tilia/v_object/component/v_time_zone.rb +51 -0
- data/lib/tilia/v_object/component/v_todo.rb +147 -0
- data/lib/tilia/v_object/component.rb +591 -0
- data/lib/tilia/v_object/date_time_parser.rb +486 -0
- data/lib/tilia/v_object/document.rb +218 -0
- data/lib/tilia/v_object/element_list.rb +18 -0
- data/lib/tilia/v_object/eof_exception.rb +8 -0
- data/lib/tilia/v_object/free_busy_data.rb +149 -0
- data/lib/tilia/v_object/free_busy_generator.rb +465 -0
- data/lib/tilia/v_object/i_tip/broker.rb +909 -0
- data/lib/tilia/v_object/i_tip/i_tip_exception.rb +9 -0
- data/lib/tilia/v_object/i_tip/message.rb +109 -0
- data/lib/tilia/v_object/i_tip/same_organizer_for_all_components_exception.rb +13 -0
- data/lib/tilia/v_object/i_tip.rb +10 -0
- data/lib/tilia/v_object/node.rb +192 -0
- data/lib/tilia/v_object/parameter.rb +327 -0
- data/lib/tilia/v_object/parse_exception.rb +7 -0
- data/lib/tilia/v_object/parser/json.rb +149 -0
- data/lib/tilia/v_object/parser/mime_dir.rb +543 -0
- data/lib/tilia/v_object/parser/parser.rb +61 -0
- data/lib/tilia/v_object/parser/xml/element/key_value.rb +60 -0
- data/lib/tilia/v_object/parser/xml/element.rb +11 -0
- data/lib/tilia/v_object/parser/xml.rb +322 -0
- data/lib/tilia/v_object/parser.rb +10 -0
- data/lib/tilia/v_object/property/binary.rb +96 -0
- data/lib/tilia/v_object/property/boolean.rb +57 -0
- data/lib/tilia/v_object/property/flat_text.rb +52 -0
- data/lib/tilia/v_object/property/float_value.rb +107 -0
- data/lib/tilia/v_object/property/i_calendar/cal_address.rb +49 -0
- data/lib/tilia/v_object/property/i_calendar/date.rb +15 -0
- data/lib/tilia/v_object/property/i_calendar/date_time.rb +330 -0
- data/lib/tilia/v_object/property/i_calendar/duration.rb +65 -0
- data/lib/tilia/v_object/property/i_calendar/period.rb +124 -0
- data/lib/tilia/v_object/property/i_calendar/recur.rb +173 -0
- data/lib/tilia/v_object/property/i_calendar.rb +14 -0
- data/lib/tilia/v_object/property/integer_value.rb +60 -0
- data/lib/tilia/v_object/property/text.rb +352 -0
- data/lib/tilia/v_object/property/time.rb +85 -0
- data/lib/tilia/v_object/property/unknown.rb +30 -0
- data/lib/tilia/v_object/property/uri.rb +78 -0
- data/lib/tilia/v_object/property/utc_offset.rb +56 -0
- data/lib/tilia/v_object/property/v_card/date.rb +31 -0
- data/lib/tilia/v_object/property/v_card/date_and_or_time.rb +343 -0
- data/lib/tilia/v_object/property/v_card/date_time.rb +22 -0
- data/lib/tilia/v_object/property/v_card/language_tag.rb +41 -0
- data/lib/tilia/v_object/property/v_card/time_stamp.rb +74 -0
- data/lib/tilia/v_object/property/v_card.rb +13 -0
- data/lib/tilia/v_object/property.rb +532 -0
- data/lib/tilia/v_object/reader.rb +73 -0
- data/lib/tilia/v_object/recur/event_iterator.rb +417 -0
- data/lib/tilia/v_object/recur/no_instances_exception.rb +11 -0
- data/lib/tilia/v_object/recur/r_date_iterator.rb +138 -0
- data/lib/tilia/v_object/recur/r_rule_iterator.rb +717 -0
- data/lib/tilia/v_object/recur.rb +10 -0
- data/lib/tilia/v_object/settings.rb +32 -0
- data/lib/tilia/v_object/splitter/i_calendar.rb +95 -0
- data/lib/tilia/v_object/splitter/splitter_interface.rb +31 -0
- data/lib/tilia/v_object/splitter/v_card.rb +56 -0
- data/lib/tilia/v_object/splitter.rb +9 -0
- data/lib/tilia/v_object/string_util.rb +58 -0
- data/lib/tilia/v_object/time_zone_data/exchange_zones.rb +96 -0
- data/lib/tilia/v_object/time_zone_data/lotus_zones.rb +104 -0
- data/lib/tilia/v_object/time_zone_data/php_zones.rb +49 -0
- data/lib/tilia/v_object/time_zone_data/windows_zones.rb +121 -0
- data/lib/tilia/v_object/time_zone_data.rb +10 -0
- data/lib/tilia/v_object/time_zone_util.rb +213 -0
- data/lib/tilia/v_object/uuid_util.rb +51 -0
- data/lib/tilia/v_object/v_card_converter.rb +354 -0
- data/lib/tilia/v_object/version.rb +9 -0
- data/lib/tilia/v_object/writer.rb +56 -0
- data/lib/tilia/v_object.rb +45 -0
- data/lib/tilia/vobject.rb +1 -0
- data/resources/schema/xcal.rng +1192 -0
- data/resources/schema/xcard.rng +388 -0
- data/test/test_helper.rb +56 -0
- data/test/v_object/attach_issue_test.rb +19 -0
- data/test/v_object/birthday_calendar_generator_test.rb +463 -0
- data/test/v_object/cli_mock.rb +19 -0
- data/test/v_object/cli_test.rb +460 -0
- data/test/v_object/component/available_test.rb +59 -0
- data/test/v_object/component/v_alarm_test.rb +160 -0
- data/test/v_object/component/v_availability_test.rb +388 -0
- data/test/v_object/component/v_calendar_test.rb +646 -0
- data/test/v_object/component/v_card_test.rb +258 -0
- data/test/v_object/component/v_event_test.rb +85 -0
- data/test/v_object/component/v_free_busy_test.rb +59 -0
- data/test/v_object/component/v_journal_test.rb +85 -0
- data/test/v_object/component/v_time_zone_test.rb +47 -0
- data/test/v_object/component/v_todo_test.rb +172 -0
- data/test/v_object/component_test.rb +419 -0
- data/test/v_object/date_time_parser_test.rb +526 -0
- data/test/v_object/document_test.rb +71 -0
- data/test/v_object/element_list_test.rb +27 -0
- data/test/v_object/em_client_test.rb +53 -0
- data/test/v_object/empty_parameter_test.rb +65 -0
- data/test/v_object/empty_value_issue_test.rb +25 -0
- data/test/v_object/fake_component.rb +21 -0
- data/test/v_object/free_busy_data_test.rb +285 -0
- data/test/v_object/free_busy_generator_test.rb +637 -0
- data/test/v_object/google_colon_escaping_test.rb +27 -0
- data/test/v_object/i_calendar/attach_parse_test.rb +24 -0
- data/test/v_object/i_tip/broker_attendee_reply_test.rb +1042 -0
- data/test/v_object/i_tip/broker_delete_event_test.rb +175 -0
- data/test/v_object/i_tip/broker_new_event_test.rb +440 -0
- data/test/v_object/i_tip/broker_process_message_test.rb +153 -0
- data/test/v_object/i_tip/broker_process_reply_test.rb +402 -0
- data/test/v_object/i_tip/broker_tester.rb +71 -0
- data/test/v_object/i_tip/broker_update_event_test.rb +763 -0
- data/test/v_object/i_tip/evolution_test.rb +2644 -0
- data/test/v_object/i_tip/message_test.rb +25 -0
- data/test/v_object/issue153.vcf +352 -0
- data/test/v_object/issue153_test.rb +12 -0
- data/test/v_object/issue26_test.rb +25 -0
- data/test/v_object/issue36_work_around_test.rb +37 -0
- data/test/v_object/issue40_test.rb +26 -0
- data/test/v_object/issue64.vcf +351 -0
- data/test/v_object/issue64_test.rb +17 -0
- data/test/v_object/issue96_test.rb +22 -0
- data/test/v_object/issue_undefined_index_test.rb +24 -0
- data/test/v_object/j_cal_test.rb +150 -0
- data/test/v_object/j_card_test.rb +192 -0
- data/test/v_object/line_folding_issue_test.rb +19 -0
- data/test/v_object/mock_document.rb +6 -0
- data/test/v_object/parameter_test.rb +109 -0
- data/test/v_object/parser/json_test.rb +370 -0
- data/test/v_object/parser/mime_dir_test.rb +14 -0
- data/test/v_object/parser/quoted_printable_test.rb +78 -0
- data/test/v_object/parser/xml_test.rb +2563 -0
- data/test/v_object/property/binary_test.rb +12 -0
- data/test/v_object/property/boolean_test.rb +18 -0
- data/test/v_object/property/compound_test.rb +43 -0
- data/test/v_object/property/float_test.rb +20 -0
- data/test/v_object/property/i_calendar/cal_address_test.rb +26 -0
- data/test/v_object/property/i_calendar/date_time_test.rb +303 -0
- data/test/v_object/property/i_calendar/duration_test.rb +14 -0
- data/test/v_object/property/i_calendar/recur_test.rb +39 -0
- data/test/v_object/property/text_test.rb +81 -0
- data/test/v_object/property/v_card/date_and_or_time_test.rb +205 -0
- data/test/v_object/property/v_card/language_tag_test.rb +35 -0
- data/test/v_object/property_test.rb +338 -0
- data/test/v_object/reader_test.rb +403 -0
- data/test/v_object/recur/event_iterator/by_month_in_daily_test.rb +52 -0
- data/test/v_object/recur/event_iterator/by_set_pos_hang_test.rb +55 -0
- data/test/v_object/recur/event_iterator/expand_floating_times_test.rb +109 -0
- data/test/v_object/recur/event_iterator/fifth_tuesday_problem_test.rb +45 -0
- data/test/v_object/recur/event_iterator/incorrect_expand_test.rb +53 -0
- data/test/v_object/recur/event_iterator/infinite_loop_problem_test.rb +75 -0
- data/test/v_object/recur/event_iterator/issue48_test.rb +43 -0
- data/test/v_object/recur/event_iterator/issue50_test.rb +123 -0
- data/test/v_object/recur/event_iterator/main_test.rb +1222 -0
- data/test/v_object/recur/event_iterator/missing_overridden_test.rb +55 -0
- data/test/v_object/recur/event_iterator/no_instances_test.rb +32 -0
- data/test/v_object/recur/event_iterator/override_first_event_test.rb +106 -0
- data/test/v_object/recur/r_date_iterator_test.rb +44 -0
- data/test/v_object/recur/r_rule_iterator_test.rb +608 -0
- data/test/v_object/recurrence_iterator/UntilRespectsTimezoneTest.ics +39 -0
- data/test/v_object/slash_r_test.rb +15 -0
- data/test/v_object/splitter/i_calendar_test.rb +299 -0
- data/test/v_object/splitter/v_card_test.rb +173 -0
- data/test/v_object/string_util_test.rb +37 -0
- data/test/v_object/test_case.rb +42 -0
- data/test/v_object/time_zone_util_test.rb +271 -0
- data/test/v_object/uuid_util_test.rb +18 -0
- data/test/v_object/v_card21_test.rb +43 -0
- data/test/v_object/v_card_converter_test.rb +419 -0
- data/test/v_object/version_test.rb +15 -0
- data/test/v_object/writer_test.rb +33 -0
- data/tilia-vobject.gemspec +17 -0
- metadata +308 -0
@@ -0,0 +1,75 @@
|
|
1
|
+
module Tilia
|
2
|
+
module VObject
|
3
|
+
class Component
|
4
|
+
# VJournal component.
|
5
|
+
#
|
6
|
+
# This component contains some additional functionality specific for VJOURNALs.
|
7
|
+
class VJournal < Component
|
8
|
+
# Returns true or false depending on if the event falls in the specified
|
9
|
+
# time-range. This is used for filtering purposes.
|
10
|
+
#
|
11
|
+
# The rules used to determine if an event falls within the specified
|
12
|
+
# time-range is based on the CalDAV specification.
|
13
|
+
#
|
14
|
+
# @param DateTimeInterface start
|
15
|
+
# @param DateTimeInterface end
|
16
|
+
#
|
17
|
+
# @return bool
|
18
|
+
def in_time_range?(start, ending)
|
19
|
+
dtstart = key?('DTSTART') ? self['DTSTART'].date_time : nil
|
20
|
+
if dtstart
|
21
|
+
effective_end = dtstart
|
22
|
+
effective_end += 1.day unless self['DTSTART'].time?
|
23
|
+
|
24
|
+
return start <= effective_end && ending > dtstart
|
25
|
+
end
|
26
|
+
|
27
|
+
false
|
28
|
+
end
|
29
|
+
|
30
|
+
# A simple list of validation rules.
|
31
|
+
#
|
32
|
+
# This is simply a list of properties, and how many times they either
|
33
|
+
# must or must not appear.
|
34
|
+
#
|
35
|
+
# Possible values per property:
|
36
|
+
# * 0 - Must not appear.
|
37
|
+
# * 1 - Must appear exactly once.
|
38
|
+
# * + - Must appear at least once.
|
39
|
+
# * * - Can appear any number of times.
|
40
|
+
# * ? - May appear, but not more than once.
|
41
|
+
#
|
42
|
+
# @var array
|
43
|
+
def validation_rules
|
44
|
+
{
|
45
|
+
'UID' => 1,
|
46
|
+
'DTSTAMP' => 1,
|
47
|
+
|
48
|
+
'CLASS' => '?',
|
49
|
+
'CREATED' => '?',
|
50
|
+
'DTSTART' => '?',
|
51
|
+
'LAST-MODIFIED' => '?',
|
52
|
+
'ORGANIZER' => '?',
|
53
|
+
'RECURRENCE-ID' => '?',
|
54
|
+
'SEQUENCE' => '?',
|
55
|
+
'STATUS' => '?',
|
56
|
+
'SUMMARY' => '?',
|
57
|
+
'URL' => '?',
|
58
|
+
|
59
|
+
'RRULE' => '?',
|
60
|
+
|
61
|
+
'ATTACH' => '*',
|
62
|
+
'ATTENDEE' => '*',
|
63
|
+
'CATEGORIES' => '*',
|
64
|
+
'COMMENT' => '*',
|
65
|
+
'CONTACT' => '*',
|
66
|
+
'DESCRIPTION' => '*',
|
67
|
+
'EXDATE' => '*',
|
68
|
+
'RELATED-TO' => '*',
|
69
|
+
'RDATE' => '*'
|
70
|
+
}
|
71
|
+
end
|
72
|
+
end
|
73
|
+
end
|
74
|
+
end
|
75
|
+
end
|
@@ -0,0 +1,51 @@
|
|
1
|
+
module Tilia
|
2
|
+
module VObject
|
3
|
+
class Component
|
4
|
+
# The VTimeZone component.
|
5
|
+
#
|
6
|
+
# This component adds functionality to a component, specific for VTIMEZONE
|
7
|
+
# components.
|
8
|
+
class VTimeZone < Component
|
9
|
+
# Returns the PHP DateTimeZone for this VTIMEZONE component.
|
10
|
+
#
|
11
|
+
# If we can't accurately determine the timezone, this method will return
|
12
|
+
# UTC.
|
13
|
+
#
|
14
|
+
# @return \DateTimeZone
|
15
|
+
def time_zone
|
16
|
+
Tilia::VObject::TimeZoneUtil.time_zone(self['TZID'].to_s, @root)
|
17
|
+
end
|
18
|
+
|
19
|
+
# A simple list of validation rules.
|
20
|
+
#
|
21
|
+
# This is simply a list of properties, and how many times they either
|
22
|
+
# must or must not appear.
|
23
|
+
#
|
24
|
+
# Possible values per property:
|
25
|
+
# * 0 - Must not appear.
|
26
|
+
# * 1 - Must appear exactly once.
|
27
|
+
# * + - Must appear at least once.
|
28
|
+
# * * - Can appear any number of times.
|
29
|
+
# * ? - May appear, but not more than once.
|
30
|
+
#
|
31
|
+
# @var array
|
32
|
+
def validation_rules
|
33
|
+
{
|
34
|
+
'TZID' => 1,
|
35
|
+
|
36
|
+
'LAST-MODIFIED' => '?',
|
37
|
+
'TZURL' => '?',
|
38
|
+
|
39
|
+
# At least 1 STANDARD or DAYLIGHT must appear, or more. But both
|
40
|
+
# cannot appear in the same VTIMEZONE.
|
41
|
+
#
|
42
|
+
# The validator is not specific yet to pick this up, so these
|
43
|
+
# rules are too loose.
|
44
|
+
'STANDARD' => '*',
|
45
|
+
'DAYLIGHT' => '*'
|
46
|
+
}
|
47
|
+
end
|
48
|
+
end
|
49
|
+
end
|
50
|
+
end
|
51
|
+
end
|
@@ -0,0 +1,147 @@
|
|
1
|
+
module Tilia
|
2
|
+
module VObject
|
3
|
+
class Component
|
4
|
+
# VTodo component.
|
5
|
+
#
|
6
|
+
# This component contains some additional functionality specific for VTODOs.
|
7
|
+
class VTodo < Component
|
8
|
+
# Returns true or false depending on if the event falls in the specified
|
9
|
+
# time-range. This is used for filtering purposes.
|
10
|
+
#
|
11
|
+
# The rules used to determine if an event falls within the specified
|
12
|
+
# time-range is based on the CalDAV specification.
|
13
|
+
#
|
14
|
+
# @param DateTimeInterface start
|
15
|
+
# @param DateTimeInterface end
|
16
|
+
#
|
17
|
+
# @return bool
|
18
|
+
def in_time_range?(start, ending)
|
19
|
+
dtstart = key?('DTSTART') ? self['DTSTART'].date_time : nil
|
20
|
+
duration = key?('DURATION') ? Tilia::VObject::DateTimeParser.parse_duration(self['DURATION']) : nil
|
21
|
+
due = key?('DUE') ? self['DUE'].date_time : nil
|
22
|
+
completed = key?('COMPLETED') ? self['COMPLETED'].date_time : nil
|
23
|
+
created = key?('CREATED') ? self['CREATED'].date_time : nil
|
24
|
+
|
25
|
+
if dtstart
|
26
|
+
if duration
|
27
|
+
effective_ending = dtstart + duration
|
28
|
+
return start <= effective_ending && ending > dtstart
|
29
|
+
elsif due
|
30
|
+
return (start < due || start <= dtstart) && (ending > dtstart || ending >= due)
|
31
|
+
else
|
32
|
+
return start <= dtstart && ending > dtstart
|
33
|
+
end
|
34
|
+
end
|
35
|
+
|
36
|
+
return start < due && ending >= due if due
|
37
|
+
|
38
|
+
if completed && created
|
39
|
+
return (start <= created || start <= completed) && (ending >= created || ending >= completed)
|
40
|
+
end
|
41
|
+
|
42
|
+
return (start <= completed && ending >= completed) if completed
|
43
|
+
|
44
|
+
return (ending > created) if created
|
45
|
+
|
46
|
+
true
|
47
|
+
end
|
48
|
+
|
49
|
+
# A simple list of validation rules.
|
50
|
+
#
|
51
|
+
# This is simply a list of properties, and how many times they either
|
52
|
+
# must or must not appear.
|
53
|
+
#
|
54
|
+
# Possible values per property:
|
55
|
+
# * 0 - Must not appear.
|
56
|
+
# * 1 - Must appear exactly once.
|
57
|
+
# * + - Must appear at least once.
|
58
|
+
# * * - Can appear any number of times.
|
59
|
+
# * ? - May appear, but not more than once.
|
60
|
+
#
|
61
|
+
# @var array
|
62
|
+
def validation_rules
|
63
|
+
{
|
64
|
+
'UID' => 1,
|
65
|
+
'DTSTAMP' => 1,
|
66
|
+
|
67
|
+
'CLASS' => '?',
|
68
|
+
'COMPLETED' => '?',
|
69
|
+
'CREATED' => '?',
|
70
|
+
'DESCRIPTION' => '?',
|
71
|
+
'DTSTART' => '?',
|
72
|
+
'GEO' => '?',
|
73
|
+
'LAST-MODIFIED' => '?',
|
74
|
+
'LOCATION' => '?',
|
75
|
+
'ORGANIZER' => '?',
|
76
|
+
'PERCENT' => '?',
|
77
|
+
'PRIORITY' => '?',
|
78
|
+
'RECURRENCE-ID' => '?',
|
79
|
+
'SEQUENCE' => '?',
|
80
|
+
'STATUS' => '?',
|
81
|
+
'SUMMARY' => '?',
|
82
|
+
'URL' => '?',
|
83
|
+
|
84
|
+
'RRULE' => '?',
|
85
|
+
'DUE' => '?',
|
86
|
+
'DURATION' => '?',
|
87
|
+
|
88
|
+
'ATTACH' => '*',
|
89
|
+
'ATTENDEE' => '*',
|
90
|
+
'CATEGORIES' => '*',
|
91
|
+
'COMMENT' => '*',
|
92
|
+
'CONTACT' => '*',
|
93
|
+
'EXDATE' => '*',
|
94
|
+
'REQUEST-STATUS' => '*',
|
95
|
+
'RELATED-TO' => '*',
|
96
|
+
'RESOURCES' => '*',
|
97
|
+
'RDATE' => '*'
|
98
|
+
}
|
99
|
+
end
|
100
|
+
|
101
|
+
# Validates the node for correctness.
|
102
|
+
#
|
103
|
+
# The following options are supported:
|
104
|
+
# Node::REPAIR - May attempt to automatically repair the problem.
|
105
|
+
#
|
106
|
+
# This method returns an array with detected problems.
|
107
|
+
# Every element has the following properties:
|
108
|
+
#
|
109
|
+
# * level - problem level.
|
110
|
+
# * message - A human-readable string describing the issue.
|
111
|
+
# * node - A reference to the problematic node.
|
112
|
+
#
|
113
|
+
# The level means:
|
114
|
+
# 1 - The issue was repaired (only happens if REPAIR was turned on)
|
115
|
+
# 2 - An inconsequential issue
|
116
|
+
# 3 - A severe issue.
|
117
|
+
#
|
118
|
+
# @param int options
|
119
|
+
#
|
120
|
+
# @return array
|
121
|
+
def validate(options = 0)
|
122
|
+
result = super(options)
|
123
|
+
if key?('DUE') && key?('DTSTART')
|
124
|
+
due = self['DUE']
|
125
|
+
dt_start = self['DTSTART']
|
126
|
+
|
127
|
+
if due.value_type != dt_start.value_type
|
128
|
+
result << {
|
129
|
+
'level' => 3,
|
130
|
+
'message' => 'The value type (DATE or DATE-TIME) must be identical for DUE and DTSTART',
|
131
|
+
'node' => due
|
132
|
+
}
|
133
|
+
elsif due.date_time < dt_start.date_time
|
134
|
+
result << {
|
135
|
+
'level' => 3,
|
136
|
+
'message' => 'DUE must occur after DTSTART',
|
137
|
+
'node' => due
|
138
|
+
}
|
139
|
+
end
|
140
|
+
end
|
141
|
+
|
142
|
+
result
|
143
|
+
end
|
144
|
+
end
|
145
|
+
end
|
146
|
+
end
|
147
|
+
end
|