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:
		| @@ -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 | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Nicolas Sebrecht
					Nicolas Sebrecht