我收到以下错误:
Vpim::InvalidEncodingError (email@email.net):
2011-06-07T01:37:06+00:00 app[web.1]: .bundle/gems/ruby/1.8/gems/vpim-0.695/lib/vpim/field.rb:110:in `decode0'
它适用于其他vcards。数据看起来正确 - 它应该是一封电子邮件:
这是一个示例vcard,当有电子邮件时会爆炸...我已经做的修复它是手动删除第二封电子邮件,但这很痛苦:
BEGIN:VCARD
VERSION:2.1
N:Roberts;Paul;;;
FN:Paul Roberts
ORG:Sonoma Technology Inc
TITLE:EVP Business Dev/Chief Scientific Officer
TEL;WORK;VOICE:707-665-9900
TEL;WORK;FAX:707-665-9800
ADR;WORK;ENCODING=QUOTED-PRINTABLE:;;1455 N McDowell Blvd Suite D;Petaluma;CA;94954;USA
LABEL;WORK;ENCODING=QUOTED-PRINTABLE:1455 N McDowell Blvd Suite D=0D=0APetaluma, CA 94954=0D=0AUSA
URL:http://www.sonomatech.com
URL:http://www.sonomatech.com
EMAIL;PREF;INTERNET:paul@sonomatech.com
paul@sonomatech.com
NOTE;ENCODING=QUOTED-PRINTABLE:=0D=0A Data provided by Lead411, http://www.lead411.com/=0D=0A =0D=0A
END:VCARD
这是我的控制器使用paperclip和vpim:
68 unless @contact.vcard.path.blank?
69
70 paperclip_vcard = File.new(@contact.vcard.path)
71
72 # try to scrub the vcard
73 scrub_vcf(paperclip_vcard)
74
75 @vcard = Vpim::Vcard.decode(paperclip_vcard).first
76 @contact.title = @vcard.title
77 @contact.email = @vcard.email
78 @contact.first_name = @vcard.name.given
79 @contact.last_name = @vcard.name.family
80 @contact.phone = @vcard.telephones[0]
81 @contact.fax = @vcard.telephones[1]
82
83 @contact.address.street1 = @vcard.address.street
84 @contact.address.city = @vcard.address.locality
85 @contact.address.state = @vcard.address.region
86 @contact.address.zip = @vcard.address.postalcode
87 @contact.company_name = @vcard.org.fetch(0)
88
89 end
答案 0 :(得分:4)
您需要了解如何创建Vcards;第14行的第二封电子邮件不是有效的属性定义,这导致解析器搞砸了(这就是为什么如果手动删除它就会成功解析的原因)。
您可以在the Vcard 2.1 specification的第2节中阅读有关属性定义的内容(RTF版本 - 可读性更高,可用here)。
根据您提供的信息,这在解码方面看起来不像问题Vpim,而是您的Vcards的创建方式。如果你自己创建Vcards,我会看看你的编码逻辑。如果您是从外部来源接收它们,那么您可能想要编写一些自定义清理逻辑来摆脱不正确的属性定义,因此您不必自己手动删除它们。
您应该可以通过对每行进行快速正则表达式检查来轻松完成此操作:
def scrub_vcf(vcard)
line_arr = File.readlines(vcard)
line_arr.delete_if { |line| line.match(/^.+\:.+$/).nil? }
File.open(vcard, "w") do |f|
line_arr.each{|line| f.puts(line)}
end
end
# use the scrubbed vcf with vpim
当然,将它保存在数组中可能会更快,而不是将其写回文件,仅供参考。
希望有所帮助。
更新:如果您不想保留该文件,可以返回一个字符串Vpim can decode而不是文件:
def scrub_vcf(vcard)
line_arr = File.readlines(vcard)
line_arr.delete_if { |line| line.match(/^.+\:.+$/).nil? }
return line_arr.join
end
# use the scrubbed vcf with vpim #=> Vpim::Vcard.decode(scrub_vcf(vcard))
请注意,在运行ruby 1.9.x时,由于Vpim::Vcard.decode
类不再具有String
方法,因此在使用带each
的字符串时遇到了问题。但是,Ruby 1.8.7运行良好。 Vpim看起来自2008/2009以来一直没有升级,所以它可能没有升级用于ruby 1.9.x。
再次更新:这是一个更新的Vpim版本,用于ruby 1.9.x(正好修复了我之前遇到过的问题):https://github.com/sam-github/vpim