# Copyright (C) 2008 Sam Roberts # This library is free software; you can redistribute it and/or modify it # under the same terms as the ruby language itself, see the file COPYING for # details. module Vcard class DirectoryInfo # A field in a directory info object. class Field # TODO # - Field should know which param values and field values are # case-insensitive, configurably, so it can down case them # - perhaps should have pvalue_set/del/add, perhaps case-insensitive, or # pvalue_iset/idel/iadd, where set sets them all, add adds if not present, # and del deletes any that are present # - I really, really, need a case-insensitive string... # - should allow nil as a field value, its not the same as "", if there is # more than one pvalue, the empty string will show up. This isn't strictly # disallowed, but its odd. Should also strip empty strings on decoding, if # I don't already. private_class_method :new def Field.create_array(fields) case fields when Hash fields.map do |name,value| DirectoryInfo::Field.create( name, value ) end else fields.to_ary end end # Encode a field. def Field.encode0(group, name, params={}, value="") # :nodoc: line = "" # A reminder of the line format: # [.];=,: if group line << group << "." end line << name params.each do |pname, pvalues| unless pvalues.respond_to? :to_ary pvalues = [ pvalues ] end line << ";" << pname << "=" sep = "" # set to "," after one pvalue has been appended pvalues.each do |pvalue| # check if we need to do any encoding if pname.casecmp("ENCODING") == 0 && pvalue == :b64 pvalue = "B" # the RFC definition of the base64 param value value = [ value.to_str ].pack("m").gsub("\n", "") end line << sep << pvalue sep =","; end end line << ":" line << Field.value_str(value) line end def Field.value_str(value) # :nodoc: line = "" case value when Date line << ::Vcard.encode_date(value) when Time #, DateTime line << ::Vcard.encode_date_time(value) when Array line << value.map { |v| Field.value_str(v) }.join(";") when Symbol line << value else # FIXME - somewhere along here, values with special chars need escaping... line << value.to_str end line end # Decode a field. def Field.decode0(atline) # :nodoc: if !(atline =~ Bnf::LINE) raise ::Vcard::InvalidEncodingError, atline end atgroup = $1.upcase atname = $2.upcase paramslist = $3 atvalue = $~[-1] # I've seen space that shouldn't be there, as in "BEGIN:VCARD ", so # strip it. I'm not absolutely sure this is allowed... it certainly # breaks round-trip encoding. atvalue.strip! if atgroup.length > 0 atgroup.chomp!(".") else atgroup = nil end atparams = {} # Collect the params, if any. if paramslist.size > 1 # v3.0 and v2.1 params paramslist.scan( Bnf::PARAM ) do # param names are case-insensitive, and multi-valued name = $1.upcase params = $3 # v2.1 params have no "=" sign, figure out what kind of param it # is (either its a known encoding, or we treat it as a "TYPE" # param). if $2 == "" params = $1 case $1 when /quoted-printable/i name = "ENCODING" when /base64/i name = "ENCODING" else name = "TYPE" end end # TODO - In ruby1.8 I can give an initial value to the atparams # hash values instead of this. unless atparams.key? name atparams[name] = [] end params.scan( Bnf::PVALUE ) do atparams[name] << ($1 || $2) end end end [ atgroup, atname, atparams, atvalue ] end def initialize(line) # :nodoc: @line = line.to_str @group, @name, @params, @value = Field.decode0(@line) @params.each do |pname,pvalues| pvalues.freeze end self end # Create a field by decoding +line+, a String which must already be # unfolded. Decoded fields are frozen, but see #copy(). def Field.decode(line) new(line).freeze end # Create a field with name +name+ (a String), value +value+ (see below), # and optional parameters, +params+. +params+ is a hash of the parameter # name (a String) to either a single string or symbol, or an array of # strings and symbols (parameters can be multi-valued). # # If "ENCODING" => :b64 is specified as a parameter, the value will be # base-64 encoded. If it's already base-64 encoded, then use String # values ("ENCODING" => "B"), and no further encoding will be done by # this routine. # # Currently handled value types are: # - Time, encoded as a date-time value # - Date, encoded as a date value # - String, encoded directly # - Array of String, concatentated with ";" between them. # # TODO - need a way to encode String values as TEXT, at least optionally, # so as to escape special chars, etc. def Field.create(name, value="", params={}) line = Field.encode0(nil, name, params, value) begin new(line) rescue ::Vcard::InvalidEncodingError => e raise ArgumentError, e.to_s end end # Create a copy of Field. If the original Field was frozen, this one # won't be. def copy Marshal.load(Marshal.dump(self)) end # The String encoding of the Field. The String will be wrapped to a # maximum line width of +width+, where +0+ means no wrapping, and nil is # to accept the default wrapping (75, recommended by RFC2425). # # Note: AddressBook.app 3.0.3 neither understands to unwrap lines when it # imports vCards (it treats them as raw new-line characters), nor wraps # long lines on export. This is mostly a cosmetic problem, but wrapping # can be disabled by setting width to +0+, if desired. # # FIXME - breaks round-trip encoding, need to change this to not wrap # fields that are already wrapped. def encode(width=nil) width = 75 unless width l = @line # Wrap to width, unless width is zero. if width > 0 l = l.gsub(/.{#{width},#{width}}/) { |m| m + "\n " } end # Make sure it's terminated with no more than a single NL. l.gsub(/\s*\z/, "") + "\n" end alias to_s encode # The name. def name @name end # The group, if present, or nil if not present. def group @group end # An Array of all the param names. def pnames @params.keys end # FIXME - remove my own uses of #params alias params pnames # :nodoc: # The first value of the param +name+, nil if there is no such param, # the param has no value, or the first param value is zero-length. def pvalue(name) v = pvalues( name ) if v v = v.first end if v v = nil unless v.length > 0 end v end # The Array of all values of the param +name+, nil if there is no such # param, [] if the param has no values. If the Field isn't frozen, the # Array is mutable. def pvalues(name) @params[name.upcase] end # FIXME - remove my own uses of #param alias param pvalues # :nodoc: alias [] pvalues # Yield once for each param, +name+ is the parameter name, +value+ is an # array of the parameter values. def each_param(&block) #:yield: name, value if @params @params.each(&block) end end # The decoded value. # # The encoding specified by the #encoding, if any, is stripped. # # Note: Both the RFC 2425 encoding param ("b", meaning base-64) and the # vCard 2.1 encoding params ("base64", "quoted-printable", "8bit", and # "7bit") are supported. # # FIXME: # - should use the VALUE parameter # - should also take a default value type, so it can be converted # if VALUE parameter is not present. def value case encoding when nil, "8BIT", "7BIT" then @value # Hack - if the base64 lines started with 2 SPC chars, which is invalid, # there will be extra spaces in @value. Since no SPC chars show up in # b64 encodings, they can be safely stripped out before unpacking. when "B", "BASE64" then @value.gsub(" ", "").unpack("m*").first when "QUOTED-PRINTABLE" then @value.unpack("M*").first else raise ::Vcard::InvalidEncodingError, "unrecognized encoding (#{encoding})" end end # Is the #name of this Field +name+? Names are case insensitive. def name?(name) @name.casecmp(name) == 0 end # Is the #group of this field +group+? Group names are case insensitive. # A +group+ of nil matches if the field has no group. def group?(group) @group.casecmp(group) == 0 end # Is the value of this field of type +kind+? RFC2425 allows the type of # a fields value to be encoded in the VALUE parameter. Don't rely on its # presence, they aren't required, and usually aren't bothered with. In # cases where the kind of value might vary (an iCalendar DTSTART can be # either a date or a date-time, for example), you are more likely to see # the kind of value specified explicitly. # # The value types defined by RFC 2425 are: # - uri: # - text: # - date: a list of 1 or more dates # - time: a list of 1 or more times # - date-time: a list of 1 or more date-times # - integer: # - boolean: # - float: def kind?(kind) self.kind.casecmp(kind) == 0 end # Is one of the values of the TYPE parameter of this field +type+? The # type parameter values are case insensitive. False if there is no TYPE # parameter. # # TYPE parameters are used for general categories, such as # distinguishing between an email address used at home or at work. def type?(type) type = type.to_str types = param("TYPE") if types types = types.detect { |t| t.casecmp(type) == 0 } end end # Is this field marked as preferred? A vCard field is preferred if # #type?("PREF"). This method is not necessarily meaningful for # non-vCard profiles. def pref? type? "PREF" end # Set whether a field is marked as preferred. See #pref? def pref=(ispref) if ispref pvalue_iadd("TYPE", "PREF") else pvalue_idel("TYPE", "PREF") end end # Is the value of this field +value+? The check is case insensitive. # FIXME - it shouldn't be insensitive, make a #casevalue? method. def value?(value) @value.casecmp(value) == 0 end # The value of the ENCODING parameter, if present, or nil if not # present. def encoding e = param("ENCODING") if e if e.length > 1 raise ::Vcard::InvalidEncodingError, "multi-valued param 'ENCODING' (#{e})" end e = e.first.upcase end e end # The type of the value, as specified by the VALUE parameter, nil if # unspecified. def kind v = param("VALUE") if v if v.size > 1 raise ::Vcard::InvalidEncodingError, "multi-valued param 'VALUE' (#{values})" end v = v.first.downcase end v end # The value as an array of Time objects (all times and dates in # RFC2425 are lists, even where it might not make sense, such as a # birthday). The time will be UTC if marked as so (with a timezone of # "Z"), and in localtime otherwise. # # TODO - support timezone offsets # # TODO - if year is before 1970, this won't work... but some people # are generating calendars saying Canada Day started in 1753! # That's just wrong! So, what to do? I add a message # saying what the year is that breaks, so they at least know that # its ridiculous! I think I need my own DateTime variant. def to_time ::Vcard.decode_date_time_list(value).collect do |d| # We get [ year, month, day, hour, min, sec, usec, tz ] begin if(d.pop == "Z") Time.gm(*d) else Time.local(*d) end rescue ArgumentError => e raise ::Vcard::InvalidEncodingError, "Time.gm(#{d.join(', ')}) failed with #{e.message}" end end rescue ::Vcard::InvalidEncodingError ::Vcard.decode_date_list(value).collect do |d| # We get [ year, month, day ] begin Time.gm(*d) rescue ArgumentError => e raise ::Vcard::InvalidEncodingError, "Time.gm(#{d.join(', ')}) failed with #{e.message}" end end end # The value as an array of Date objects (all times and dates in # RFC2425 are lists, even where it might not make sense, such as a # birthday). # # The field value may be a list of either DATE or DATE-TIME values, # decoding is tried first as a DATE-TIME, then as a DATE, if neither # works an InvalidEncodingError will be raised. def to_date ::Vcard.decode_date_time_list(value).collect do |d| # We get [ year, month, day, hour, min, sec, usec, tz ] Date.new(d[0], d[1], d[2]) end rescue ::Vcard::InvalidEncodingError ::Vcard.decode_date_list(value).collect do |d| # We get [ year, month, day ] Date.new(*d) end end # The value as text. Text can have escaped newlines, commas, and escape # characters, this method will strip them, if present. # # In theory, #value could also do this, but it would need to know that # the value is of type "TEXT", and often for text values the "VALUE" # parameter is not present, so knowledge of the expected type of the # field is required from the decoder. def to_text ::Vcard.decode_text(value) end # The undecoded value, see +value+. def value_raw @value end # TODO def pretty_print() ... # Set the group of this field to +group+. def group=(group) mutate(group, @name, @params, @value) group end # Set the value of this field to +value+. Valid values are as in # Field.create(). def value=(value) mutate(@group, @name, @params, value) value end # Convert +value+ to text, then assign. # # TODO - unimplemented def text=(text) end # Set a the param +pname+'s value to +pvalue+, replacing any value it # currently has. See Field.create() for a description of +pvalue+. # # Example: # if field["TYPE"] # field["TYPE"] << "HOME" # else # field["TYPE"] = [ "HOME" ] # end # # TODO - this could be an alias to #pvalue_set def []=(pname,pvalue) unless pvalue.respond_to?(:to_ary) pvalue = [ pvalue ] end h = @params.dup h[pname.upcase] = pvalue mutate(@group, @name, h, @value) pvalue end # Add +pvalue+ to the param +pname+'s value. The values are treated as a # set so duplicate values won't occur, and String values are case # insensitive. See Field.create() for a description of +pvalue+. def pvalue_iadd(pname, pvalue) pname = pname.upcase # Get a uniq set, where strings are compared case-insensitively. values = [ pvalue, @params[pname] ].flatten.compact values = values.collect do |v| if v.respond_to? :to_str v = v.to_str.upcase end v end values.uniq! h = @params.dup h[pname] = values mutate(@group, @name, h, @value) values end # Delete +pvalue+ from the param +pname+'s value. The values are treated # as a set so duplicate values won't occur, and String values are case # insensitive. +pvalue+ must be a single String or Symbol. def pvalue_idel(pname, pvalue) pname = pname.upcase if pvalue.respond_to? :to_str pvalue = pvalue.to_str.downcase end # Get a uniq set, where strings are compared case-insensitively. values = [ nil, @params[pname] ].flatten.compact values = values.collect do |v| if v.respond_to? :to_str v = v.to_str.downcase end v end values.uniq! values.delete pvalue h = @params.dup h[pname] = values mutate(@group, @name, h, @value) values end # FIXME - should change this so it doesn't assign to @line here, so @line # is used to preserve original encoding. That way, #encode can only wrap # new fields, not old fields. def mutate(g, n, p, v) #:nodoc: line = Field.encode0(g, n, p, v) @group, @name, @params, @value = Field.decode0(line) @line = line self rescue ::Vcard::InvalidEncodingError => e raise ArgumentError, e.to_s end private :mutate end end end