Merge branch 'ss/better-error-throwing-and-id-sequence' into next
Conflicts: offlineimap/folder/IMAP.py offlineimap/imaputil.py Signed-off-by: Nicolas Sebrecht <nicolas.s-dev@laposte.net>
This commit is contained in:
commit
67863017e2
@ -110,76 +110,70 @@ class IMAPFolder(BaseFolder):
|
|||||||
self.imapserver.releaseconnection(imapobj)
|
self.imapserver.releaseconnection(imapobj)
|
||||||
return False
|
return False
|
||||||
|
|
||||||
# TODO: Make this so that it can define a date that would be the oldest messages etc.
|
|
||||||
def cachemessagelist(self):
|
def cachemessagelist(self):
|
||||||
imapobj = self.imapserver.acquireconnection()
|
maxage = self.config.getdefaultint("Account %s" % self.accountname,
|
||||||
|
"maxage", -1)
|
||||||
|
maxsize = self.config.getdefaultint("Account %s" % self.accountname,
|
||||||
|
"maxsize", -1)
|
||||||
self.messagelist = {}
|
self.messagelist = {}
|
||||||
|
|
||||||
|
imapobj = self.imapserver.acquireconnection()
|
||||||
try:
|
try:
|
||||||
# Primes untagged_responses
|
res_type, imapdata = imapobj.select(self.getfullname(), True)
|
||||||
imaptype, imapdata = imapobj.select(self.getfullname(), readonly = 1, force = 1)
|
|
||||||
|
|
||||||
maxage = self.config.getdefaultint("Account " + self.accountname, "maxage", -1)
|
# By default examine all UIDs in this folder
|
||||||
maxsize = self.config.getdefaultint("Account " + self.accountname, "maxsize", -1)
|
msgsToFetch = '1:*'
|
||||||
|
|
||||||
if (maxage != -1) | (maxsize != -1):
|
if (maxage != -1) | (maxsize != -1):
|
||||||
try:
|
search_cond = "(";
|
||||||
search_condition = "(";
|
|
||||||
|
|
||||||
if(maxage != -1):
|
if(maxage != -1):
|
||||||
#find out what the oldest message is that we should look at
|
#find out what the oldest message is that we should look at
|
||||||
oldest_time_struct = time.gmtime(time.time() - (60*60*24*maxage))
|
oldest_struct = time.gmtime(time.time() - (60*60*24*maxage))
|
||||||
|
|
||||||
#format this manually - otherwise locales could cause problems
|
#format months manually - otherwise locales cause problems
|
||||||
monthnames_standard = ["Jan", "Feb", "Mar", "Apr", "May", \
|
monthnames = ["Jan", "Feb", "Mar", "Apr", "May", \
|
||||||
"Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
|
"Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"]
|
||||||
|
|
||||||
our_monthname = monthnames_standard[oldest_time_struct[1]-1]
|
month = monthnames[oldest_struct[1]-1]
|
||||||
daystr = "%(day)02d" % {'day' : oldest_time_struct[2]}
|
daystr = "%(day)02d" % {'day' : oldest_struct[2]}
|
||||||
date_search_str = "SINCE " + daystr + "-" + our_monthname \
|
|
||||||
+ "-" + str(oldest_time_struct[0])
|
|
||||||
|
|
||||||
search_condition += date_search_str
|
search_cond += "SINCE %s-%s-%s" % (daystr, month,
|
||||||
|
oldest_struct[0])
|
||||||
|
|
||||||
if(maxsize != -1):
|
if(maxsize != -1):
|
||||||
if(maxage != -1): #There are two conditions - add a space
|
if(maxage != -1): # There are two conditions, add space
|
||||||
search_condition += " "
|
search_cond += " "
|
||||||
|
search_cond += "SMALLER %d" % maxsize
|
||||||
|
|
||||||
search_condition += "SMALLER " + self.config.getdefault("Account " + self.accountname, "maxsize", -1)
|
search_cond += ")"
|
||||||
|
|
||||||
search_condition += ")"
|
res_type, res_data = imapobj.search(None, search_cond)
|
||||||
searchresult = imapobj.search(None, search_condition)
|
if res_type != 'OK':
|
||||||
|
raise OfflineImapError("SEARCH in folder [%s]%s failed. "
|
||||||
|
"Search string was '%s'. Server responded '[%s] %s'" % (
|
||||||
|
self.getrepository(), self,
|
||||||
|
search_cond, res_type, res_data),
|
||||||
|
OfflineImapError.ERROR.FOLDER)
|
||||||
|
|
||||||
#result would come back seperated by space - to change into a fetch
|
# Result UIDs are seperated by space, coalesce into ranges
|
||||||
#statement we need to change space to comma
|
msgsToFetch = imaputil.uid_sequence(res_data.split())
|
||||||
messagesToFetch = searchresult[1][0].replace(" ", ",")
|
if not msgsToFetch:
|
||||||
except KeyError:
|
return # No messages to sync
|
||||||
return
|
|
||||||
if len(messagesToFetch) < 1:
|
|
||||||
# No messages; return
|
|
||||||
return
|
|
||||||
else:
|
|
||||||
# 1. Some mail servers do not return an EXISTS response
|
|
||||||
# if the folder is empty. 2. ZIMBRA servers can return
|
|
||||||
# multiple EXISTS replies in the form 500, 1000, 1500,
|
|
||||||
# 1623 so check for potentially multiple replies.
|
|
||||||
if imapdata == [None]:
|
|
||||||
return
|
|
||||||
|
|
||||||
maxmsgid = 0
|
# Get the flags and UIDs for these. single-quotes prevent
|
||||||
for msgid in imapdata:
|
# imaplib2 from quoting the sequence.
|
||||||
maxmsgid = max(long(msgid), maxmsgid)
|
res_type, response = imapobj.fetch("'%s'" % msgsToFetch,
|
||||||
if maxmsgid < 1:
|
'(FLAGS UID)')
|
||||||
#no messages; return
|
if res_type != 'OK':
|
||||||
return
|
raise OfflineImapError("FETCHING UIDs in folder [%s]%s failed. "
|
||||||
messagesToFetch = '1:%d' % maxmsgid;
|
"Server responded '[%s] %s'" % (
|
||||||
|
self.getrepository(), self,
|
||||||
# Now, get the flags and UIDs for these.
|
res_type, response),
|
||||||
# We could conceivably get rid of maxmsgid and just say
|
OfflineImapError.ERROR.FOLDER)
|
||||||
# '1:*' here.
|
|
||||||
response = imapobj.fetch(messagesToFetch, '(FLAGS UID)')[1]
|
|
||||||
finally:
|
finally:
|
||||||
self.imapserver.releaseconnection(imapobj)
|
self.imapserver.releaseconnection(imapobj)
|
||||||
|
|
||||||
for messagestr in response:
|
for messagestr in response:
|
||||||
# looks like: '1 (FLAGS (\\Seen Old) UID 4807)' or None if no msg
|
# looks like: '1 (FLAGS (\\Seen Old) UID 4807)' or None if no msg
|
||||||
# Discard initial message number.
|
# Discard initial message number.
|
||||||
|
@ -191,10 +191,8 @@ def flagsmaildir2imap(maildirflaglist):
|
|||||||
def uid_sequence(uidlist):
|
def uid_sequence(uidlist):
|
||||||
"""Collapse UID lists into shorter sequence sets
|
"""Collapse UID lists into shorter sequence sets
|
||||||
|
|
||||||
[1,2,3,4,5,10,12,13] will return "1:5,10,12:13". This function
|
[1,2,3,4,5,10,12,13] will return "1:5,10,12:13". This function sorts
|
||||||
converts items to numeric type and sorts the list to always produce
|
the list, and only collapses if subsequent entries form a range.
|
||||||
the minimal collapsed set.
|
|
||||||
|
|
||||||
:returns: The collapsed UID list as string"""
|
:returns: The collapsed UID list as string"""
|
||||||
def getrange(start, end):
|
def getrange(start, end):
|
||||||
if start == end:
|
if start == end:
|
||||||
@ -208,6 +206,7 @@ def uid_sequence(uidlist):
|
|||||||
sorted_uids = sorted(map(int, uidlist))
|
sorted_uids = sorted(map(int, uidlist))
|
||||||
|
|
||||||
for item in iter(sorted_uids):
|
for item in iter(sorted_uids):
|
||||||
|
item = int(item)
|
||||||
if start == None: # First item
|
if start == None: # First item
|
||||||
start, end = item, item
|
start, end = item, item
|
||||||
elif item == end + 1: # Next item in a range
|
elif item == end + 1: # Next item in a range
|
||||||
|
Loading…
x
Reference in New Issue
Block a user