Implement APPENDUID support
Rather than inserting our own home-grown header, everytime we save a message to an IMAP server, we check if we suport the UIDPLUS extension which provides us with an APPENDUID reply. Use that to find the new UID if possible, but keep the old way if we don't have that extension. If a folder is read-only, return the uid that we have passed in per API description in folder.Base.py Signed-off-by: Sebastian Spaeth <Sebastian@SSpaeth.de> Signed-off-by: Nicolas Sebrecht <nicolas.s-dev@laposte.net>
This commit is contained in:
		 Sebastian Spaeth
					Sebastian Spaeth
				
			
				
					committed by
					
						 Nicolas Sebrecht
						Nicolas Sebrecht
					
				
			
			
				
	
			
			
			 Nicolas Sebrecht
						Nicolas Sebrecht
					
				
			
						parent
						
							96db608e4d
						
					
				
				
					commit
					3b8e1f91cd
				
			| @@ -12,6 +12,9 @@ others. | |||||||
|  |  | ||||||
| New Features | New Features | ||||||
| ------------ | ------------ | ||||||
|  | * Implement UIDPLUS extension support. OfflineIMAP will now not insert | ||||||
|  |   an X-OfflineIMAP header if the mail server supports the UIDPLUS | ||||||
|  |   extension. | ||||||
|  |  | ||||||
| * SSL: support subjectAltName. | * SSL: support subjectAltName. | ||||||
|  |  | ||||||
|   | |||||||
| @@ -274,9 +274,6 @@ class IMAPFolder(BaseFolder): | |||||||
|  |  | ||||||
|  |  | ||||||
|     def savemessage_searchforheader(self, imapobj, headername, headervalue): |     def savemessage_searchforheader(self, imapobj, headername, headervalue): | ||||||
|         if imapobj.untagged_responses.has_key('APPENDUID'): |  | ||||||
|             return long(imapobj.untagged_responses['APPENDUID'][-1].split(' ')[1]) |  | ||||||
|  |  | ||||||
|         self.ui.debug('imap', 'savemessage_searchforheader called for %s: %s' % \ |         self.ui.debug('imap', 'savemessage_searchforheader called for %s: %s' % \ | ||||||
|                  (headername, headervalue)) |                  (headername, headervalue)) | ||||||
|         # Now find the UID it got. |         # Now find the UID it got. | ||||||
| @@ -389,38 +386,42 @@ class IMAPFolder(BaseFolder): | |||||||
|         This function will update the self.messagelist dict to contain |         This function will update the self.messagelist dict to contain | ||||||
|         the new message after sucessfully saving it. |         the new message after sucessfully saving it. | ||||||
|  |  | ||||||
|         :param rtime: A timestamp to be |         :param rtime: A timestamp to be used as the mail date | ||||||
|         :returns: the UID of the new message as assigned by the |         :returns: the UID of the new message as assigned by the | ||||||
|                   server. If the folder is read-only it will return 0.""" |                   server. If the folder is read-only it will return 0.""" | ||||||
|         imapobj = self.imapserver.acquireconnection() |  | ||||||
|         self.ui.debug('imap', 'savemessage: called') |         self.ui.debug('imap', 'savemessage: called') | ||||||
|  |  | ||||||
|         try: |         try: | ||||||
|  |             imapobj = self.imapserver.acquireconnection() | ||||||
|  |  | ||||||
|             try: |             try: | ||||||
|                 imapobj.select(self.getfullname()) # Needed for search |                 imapobj.select(self.getfullname()) # Needed for search and making the box READ-WRITE | ||||||
|             except imapobj.readonly: |             except imapobj.readonly: | ||||||
|  |                 # readonly exception. Return original uid to notify that | ||||||
|  |                 # we did not save the message. (see savemessage in Base.py) | ||||||
|                 self.ui.msgtoreadonly(self, uid, content, flags) |                 self.ui.msgtoreadonly(self, uid, content, flags) | ||||||
|                 # Return indicating message taken, but no UID assigned. |                 return uid | ||||||
|                 return 0 |  | ||||||
|  |             # UIDPLUS extension provides us with an APPENDUID response to our append() | ||||||
|  |             use_uidplus = 'UIDPLUS' in imapobj.capabilities | ||||||
|  |  | ||||||
|             # get the date of the message file, so we can pass it to the server. |             # get the date of the message file, so we can pass it to the server. | ||||||
|             date = self.getmessageinternaldate(content, rtime) |             date = self.getmessageinternaldate(content, rtime) | ||||||
|  |  | ||||||
|             self.ui.debug('imap', 'savemessage: using date %s' % date) |             self.ui.debug('imap', 'savemessage: using date %s' % date) | ||||||
|  |      | ||||||
|             content = re.sub("(?<!\r)\n", "\r\n", content) |             content = re.sub("(?<!\r)\n", "\r\n", content) | ||||||
|             self.ui.debug('imap', 'savemessage: initial content is: ' + repr(content)) |  | ||||||
|      |      | ||||||
|             (headername, headervalue) = self.generate_randomheader(content) |             if not use_uidplus: | ||||||
|             ui.debug('imap', 'savemessage: new headers are: %s: %s' % \ |                 # insert a random unique header that we can fetch later | ||||||
|                      (headername, headervalue)) |                 (headername, headervalue) = self.generate_randomheader(content) | ||||||
|             content = self.savemessage_addheader(content, headername, |                 self.ui.debug('imap', 'savemessage: new headers are: %s: %s' % \ | ||||||
|                                                  headervalue) |                              (headername, headervalue)) | ||||||
|             self.ui.debug('imap', 'savemessage: new content is: ' + repr(content)) |                 content = self.savemessage_addheader(content, headername, | ||||||
|             self.ui.debug('imap', 'savemessage: new content length is ' + \ |                                                      headervalue)     | ||||||
|                      str(len(content))) |             self.ui.debug('imap', 'savemessage: content is: ' + repr(content)) | ||||||
|  |  | ||||||
|             # TODO: append could raise a ValueError if the date is not in |             # TODO: - append could raise a ValueError if the date is not in | ||||||
|             #       valid format...? |             #         valid format...? | ||||||
|             (typ,dat) = imapobj.append(self.getfullname(), |             (typ,dat) = imapobj.append(self.getfullname(), | ||||||
|                                        imaputil.flagsmaildir2imap(flags), |                                        imaputil.flagsmaildir2imap(flags), | ||||||
|                                        date, content) |                                        date, content) | ||||||
| @@ -430,25 +431,37 @@ class IMAPFolder(BaseFolder): | |||||||
|             (typ,dat) = imapobj.check() |             (typ,dat) = imapobj.check() | ||||||
|             assert(typ == 'OK') |             assert(typ == 'OK') | ||||||
|  |  | ||||||
|             # Keep trying until we get the UID. |             # get the UID. | ||||||
|             self.ui.debug('imap', 'savemessage: first attempt to get new UID') |             if use_uidplus: | ||||||
|             uid = self.savemessage_searchforheader(imapobj, headername, |                 # get the new UID from the APPENDUID response, it could look like | ||||||
|                                                    headervalue) |                 # OK [APPENDUID 38505 3955] APPEND completed | ||||||
|             # See docs for savemessage in Base.py for explanation of this and other return values |                 # with 38505 bein folder UIDvalidity and 3955 the new UID | ||||||
|             if uid <= 0: |                 if not imapobj.untagged_responses.has_key('APPENDUID'): | ||||||
|                 self.ui.debug('imap', 'savemessage: first attempt to get new UID failed.  Going to run a NOOP and try again.') |                     self.ui.warn("Server supports UIDPLUS but got no APPENDUID appending a message.") | ||||||
|                 assert(imapobj.noop()[0] == 'OK') |                     return 0 | ||||||
|  |                 uid = long(imapobj.untagged_responses['APPENDUID'][-1].split(' ')[1]) | ||||||
|  |  | ||||||
|  |             else: | ||||||
|  |                 # we don't support UIDPLUS | ||||||
|                 uid = self.savemessage_searchforheader(imapobj, headername, |                 uid = self.savemessage_searchforheader(imapobj, headername, | ||||||
|                                                        headervalue) |                                                        headervalue) | ||||||
|  |                 # See docs for savemessage in Base.py for explanation of this and other return values | ||||||
|  |                 if uid == 0: | ||||||
|  |                     self.ui.debug('imap', 'savemessage: first attempt to get new UID failed.  Going to run a NOOP and try again.') | ||||||
|  |                     assert(imapobj.noop()[0] == 'OK') | ||||||
|  |                     uid = self.savemessage_searchforheader(imapobj, headername, | ||||||
|  |                                                        headervalue) | ||||||
|  |  | ||||||
|         finally: |         finally: | ||||||
|             self.imapserver.releaseconnection(imapobj) |             self.imapserver.releaseconnection(imapobj) | ||||||
|  |  | ||||||
|         if uid: # avoid UID FETCH 0 crash happening later on |         if uid: # avoid UID FETCH 0 crash happening later on | ||||||
|             self.messagelist[uid] = {'uid': uid, 'flags': flags} |             self.messagelist[uid] = {'uid': uid, 'flags': flags} | ||||||
|  |  | ||||||
|         self.ui.debug('imap', 'savemessage: returning %d' % uid) |         self.ui.debug('imap', 'savemessage: returning new UID %d' % uid) | ||||||
|         return uid |         return uid | ||||||
|  |  | ||||||
|  |  | ||||||
|     def savemessageflags(self, uid, flags): |     def savemessageflags(self, uid, flags): | ||||||
|         imapobj = self.imapserver.acquireconnection() |         imapobj = self.imapserver.acquireconnection() | ||||||
|         try: |         try: | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user