Class: Icalendar::Parser

Inherits:
Object
  • Object
show all
Defined in:
lib/icalendar/parser.rb

Instance Attribute Summary collapse

Instance Method Summary collapse

Constructor Details

#initialize(source, strict = false) ⇒ Parser

Returns a new instance of Parser.



9
10
11
12
13
14
15
16
17
18
19
20
21
22
# File 'lib/icalendar/parser.rb', line 9

def initialize(source, strict = false)
  if source.respond_to? :gets
    @source = source
  elsif source.respond_to? :to_s
    @source = StringIO.new source.to_s, 'r'
  else
    msg = 'Icalendar::Parser.new must be called with a String or IO object'
    Icalendar.fatal msg
    fail ArgumentError, msg
  end
  read_in_data
  @strict = strict
  @timezone_store = TimezoneStore.new
end

Instance Attribute Details

#component_class=(value) ⇒ Object

Sets the attribute component_class

Parameters:

  • value

    the value to set the attribute component_class to.



6
7
8
# File 'lib/icalendar/parser.rb', line 6

def component_class=(value)
  @component_class = value
end

#sourceObject (readonly)

Returns the value of attribute source.



7
8
9
# File 'lib/icalendar/parser.rb', line 7

def source
  @source
end

#strictObject (readonly)

Returns the value of attribute strict.



7
8
9
# File 'lib/icalendar/parser.rb', line 7

def strict
  @strict
end

#timezone_storeObject (readonly)

Returns the value of attribute timezone_store.



7
8
9
# File 'lib/icalendar/parser.rb', line 7

def timezone_store
  @timezone_store
end

Instance Method Details

#get_wrapper_class(component, fields) ⇒ Object



80
81
82
83
84
85
86
87
88
89
90
# File 'lib/icalendar/parser.rb', line 80

def get_wrapper_class(component, fields)
  klass = component.class.default_property_types[fields[:name]]
  if !fields[:params]['value'].nil?
    klass_name = fields[:params].delete('value').first
    unless klass_name.upcase == klass.value_type
      klass_name = "Icalendar::Values::#{klass_name.downcase.gsub(/(?:\A|-)(.)/) { |m| m[-1].upcase }}"
      klass = Object.const_get klass_name if Object.const_defined?(klass_name)
    end
  end
  klass
end

#parseObject



24
25
26
27
28
29
30
31
32
33
# File 'lib/icalendar/parser.rb', line 24

def parse
  components = []
  while (fields = next_fields)
    component = component_class.new
    if fields[:name] == 'begin' && fields[:value].downcase == component.ical_name.downcase
      components << parse_component(component)
    end
  end
  components
end

#parse_property(component, fields = nil) ⇒ Object



35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
# File 'lib/icalendar/parser.rb', line 35

def parse_property(component, fields = nil)
  fields = next_fields if fields.nil?
  prop_name = %w(class method).include?(fields[:name]) ? "ip_#{fields[:name]}" : fields[:name]
  multi_property = component.class.multiple_properties.include? prop_name
  prop_value = wrap_property_value component, fields, multi_property
  begin
    method_name = if multi_property
      "append_#{prop_name}"
    else
      "#{prop_name}="
    end
    component.send method_name, prop_value
  rescue NoMethodError => nme
    if strict?
      Icalendar.logger.error "No method \"#{method_name}\" for component #{component}"
      raise nme
    else
      Icalendar.logger.warn "No method \"#{method_name}\" for component #{component}. Appending to custom."
      component.append_custom_property prop_name, prop_value
    end
  end
end

#strict?Boolean

Returns:

  • (Boolean)


92
93
94
# File 'lib/icalendar/parser.rb', line 92

def strict?
  !!@strict
end

#wrap_in_array?(klass, value, multi_property) ⇒ Boolean

Returns:

  • (Boolean)


75
76
77
78
# File 'lib/icalendar/parser.rb', line 75

def wrap_in_array?(klass, value, multi_property)
  klass.value_type != 'RECUR' &&
    ((multi_property && value =~ /(?<!\\)[,;]/) || value =~ /(?<!\\);/)
end

#wrap_property_value(component, fields, multi_property) ⇒ Object



58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
# File 'lib/icalendar/parser.rb', line 58

def wrap_property_value(component, fields, multi_property)
  klass = get_wrapper_class component, fields
  if wrap_in_array? klass, fields[:value], multi_property
    delimiter = fields[:value].match(/(?<!\\)([,;])/)[1]
    Icalendar::Values::Array.new fields[:value].split(/(?<!\\)[;,]/),
                                 klass,
                                 fields[:params],
                                 delimiter: delimiter
  else
    klass.new fields[:value], fields[:params]
  end
rescue Icalendar::Values::DateTime::FormatError => fe
  raise fe if strict?
  fields[:params]['value'] = ['DATE']
  retry
end