4

我正在使用 Ruby 和 Rails 开发 IMAP 客户端。我可以成功导入邮件、邮箱等... 但是,在初始导入之后,如何检测自上次同步以来发生的任何更改?

目前,我将 UID 和 UID 有效性值存储在数据库中,比较它们并进行适当的搜索。这可行,但它不会检测到已删除的消息或对消息标志的更改等。

我是否必须每次都提取所有消息才能检测到这些更改?其他 IMAP 客户端是如何做到这么快的(即 Apple Mail 和 Postbox)。我的脚本每个帐户已经花费了 10 多秒,而电子邮件地址很少:

# select ourself as the current mailbox
@imap_connection.examine(self.location)

# grab all new messages and update them in the database
# if the uid's are still valid, we will just fetch the newest UIDs
# otherwise, we need to search when we last synced, which is slower :(
if self.uid_validity.nil? || uid_validity == self.uid_validity
  # for some IMAP servers, if a mailbox is empty, a uid_fetch will fail, so then
  begin
    messages = @imap_connection.uid_fetch(uid_range, ['UID', 'RFC822', 'FLAGS'])
  rescue
    # gmail cries if the folder is empty
    uids = @imap_connection.uid_search(['ALL'])
    messages = @imap_connection.uid_fetch(uids, ['UID', 'RFC822', 'FLAGS']) unless uids.empty?
  end

  messages.each do |imap_message|
    Message.create_from_imap!(imap_message, self.id)
  end unless messages.nil?
else
  query = self.last_synced.nil? ? ['All'] : ['SINCE', Net::IMAP.format_datetime(self.last_synced)]
  @imap_connection.search(query).each do |message_id|
    imap_message = @imap_connection.fetch(message_id, ['RFC822', 'FLAGS', 'UID'])[0]

    # don't mark the messages as read
    #@imap_connection.store(message_id, '-FLAGS', [:Seen])

    Message.create_from_imap!(imap_message, self.id)
  end
end

# now assume all UIDs are valid
self.uid_validity = uid_validity

# now remember that we just fetched all those messages
self.last_synced = Time.now
self.save!
4

2 回答 2

12

快速标志更改重新同步( RFC-4551 )有一个 IMAP 扩展。使用此扩展,可以搜索自上次同步以来已更改的所有消息(基于某种时间戳)。但是,据我所知,此扩展并未得到广泛支持。

有一个信息 RFC 描述了 IMAP 客户端应该如何进行同步(RFC-4549,第 4.3 节)。文中建议发出以下两条命令:

tag1 UID FETCH <lastseenuid+1>:* <descriptors>
tag2 UID FETCH 1:<lastseenuid> FLAGS

第一个命令用于获取所有未知邮件的所需信息(不知道有多少邮件)。第二个命令用于同步已看到邮件的标志。

AFAIK 这种方法被广泛使用。因此,许多 IMAP 服务器包含优化以快速提供此信息。通常,网络带宽是限制因素。

于 2012-04-11T17:36:35.763 回答
1

不幸的是,IMAP 协议以这种方式脑死亡。例如,IDLE 确实应该能够在连接时返回这种东西。上面的 FETCH FLAGS 建议是唯一的方法。

然而,需要注意的一件事是,根据规范,UID 仅对给定会话有效。您不应该存储它们,即使某些服务器会保留它们。

于 2012-08-24T03:56:50.683 回答