Fix getuidvalidity crash (UIDVALIDITY returning None)
Rename getuidvalidity -> get_uidvalidity and cache the IMAP result. 1) Start modernizing our function names using more underscores 2) IMAPs implementation of get_uidvalidity was removing the UIDVALIDITY result from the imaplib2 result stack, so subsequent calls would return "None". As various functions can invoke this, this led to some errors that we avoid by caching the current UIDVALIDITY value in the Folder instance. There are more simplifications and improvements to be made. Signed-off-by: Sebastian Spaeth <Sebastian@SSpaeth.de>
This commit is contained in:
parent
7c4fea906f
commit
9453e1d955
@ -26,3 +26,4 @@ Bug Fixes
|
|||||||
|
|
||||||
* Fix python2.6 compatibility with the TTYUI backend (crash)
|
* Fix python2.6 compatibility with the TTYUI backend (crash)
|
||||||
* Fix TTYUI regression from 6.5.2 in refresh loop (crash)
|
* Fix TTYUI regression from 6.5.2 in refresh loop (crash)
|
||||||
|
* Fix crashes related to UIDVALIDITY returning "None"
|
||||||
|
@ -378,7 +378,7 @@ def syncfolder(account, remotefolder, quick):
|
|||||||
statusfolder = statusrepos.getfolder(remotefolder.getvisiblename().\
|
statusfolder = statusrepos.getfolder(remotefolder.getvisiblename().\
|
||||||
replace(remoterepos.getsep(),
|
replace(remoterepos.getsep(),
|
||||||
statusrepos.getsep()))
|
statusrepos.getsep()))
|
||||||
if localfolder.getuidvalidity() == None:
|
if localfolder.get_uidvalidity() == None:
|
||||||
# This is a new folder, so delete the status cache to be sure
|
# This is a new folder, so delete the status cache to be sure
|
||||||
# we don't have a conflict.
|
# we don't have a conflict.
|
||||||
statusfolder.deletemessagelist()
|
statusfolder.deletemessagelist()
|
||||||
|
@ -110,13 +110,14 @@ class BaseFolder(object):
|
|||||||
return basename
|
return basename
|
||||||
|
|
||||||
def isuidvalidityok(self):
|
def isuidvalidityok(self):
|
||||||
"""Does the cached UID match the real UID
|
"""Tests if the cached UIDVALIDITY match the real current one
|
||||||
|
|
||||||
If required it caches the UID. In this case the function is not
|
If required it saves the UIDVALIDITY value. In this case the
|
||||||
threadsafe. So don't attempt to call it from concurrent threads."""
|
function is not threadsafe. So don't attempt to call it from
|
||||||
|
concurrent threads."""
|
||||||
|
|
||||||
if self.getsaveduidvalidity() != None:
|
if self.getsaveduidvalidity() != None:
|
||||||
return self.getsaveduidvalidity() == self.getuidvalidity()
|
return self.getsaveduidvalidity() == self.get_uidvalidity()
|
||||||
else:
|
else:
|
||||||
self.saveuidvalidity()
|
self.saveuidvalidity()
|
||||||
return 1
|
return 1
|
||||||
@ -142,7 +143,7 @@ class BaseFolder(object):
|
|||||||
|
|
||||||
This function is not threadsafe, so don't attempt to call it
|
This function is not threadsafe, so don't attempt to call it
|
||||||
from concurrent threads."""
|
from concurrent threads."""
|
||||||
newval = self.getuidvalidity()
|
newval = self.get_uidvalidity()
|
||||||
uidfilename = self._getuidfilename()
|
uidfilename = self._getuidfilename()
|
||||||
|
|
||||||
file = open(uidfilename + ".tmp", "wt")
|
file = open(uidfilename + ".tmp", "wt")
|
||||||
@ -151,7 +152,11 @@ class BaseFolder(object):
|
|||||||
os.rename(uidfilename + ".tmp", uidfilename)
|
os.rename(uidfilename + ".tmp", uidfilename)
|
||||||
self._base_saved_uidvalidity = newval
|
self._base_saved_uidvalidity = newval
|
||||||
|
|
||||||
def getuidvalidity(self):
|
def get_uidvalidity(self):
|
||||||
|
"""Retrieve the current connections UIDVALIDITY value
|
||||||
|
|
||||||
|
This function needs to be implemented by each Backend
|
||||||
|
:returns: UIDVALIDITY as a (long) number"""
|
||||||
raise NotImplementedException
|
raise NotImplementedException
|
||||||
|
|
||||||
def cachemessagelist(self):
|
def cachemessagelist(self):
|
||||||
|
@ -65,17 +65,23 @@ class IMAPFolder(BaseFolder):
|
|||||||
def getcopyinstancelimit(self):
|
def getcopyinstancelimit(self):
|
||||||
return 'MSGCOPY_' + self.repository.getname()
|
return 'MSGCOPY_' + self.repository.getname()
|
||||||
|
|
||||||
def getuidvalidity(self):
|
def get_uidvalidity(self):
|
||||||
|
"""Retrieve the current connections UIDVALIDITY value
|
||||||
|
|
||||||
|
UIDVALIDITY value will be cached on the first call.
|
||||||
|
:returns: The UIDVALIDITY as (long) number."""
|
||||||
|
if hasattr(self, '_uidvalidity'):
|
||||||
|
# use cached value if existing
|
||||||
|
return self._uidvalidity
|
||||||
imapobj = self.imapserver.acquireconnection()
|
imapobj = self.imapserver.acquireconnection()
|
||||||
try:
|
try:
|
||||||
# SELECT receives UIDVALIDITY response
|
# SELECT (if not already done) and get current UIDVALIDITY
|
||||||
self.selectro(imapobj)
|
self.selectro(imapobj)
|
||||||
# note: we would want to use .response() here but that
|
typ, uidval = imapobj.response('UIDVALIDITY')
|
||||||
# often seems to return [None], even though we have
|
assert uidval != [None] and uidval != None, \
|
||||||
# data. TODO
|
"response('UIDVALIDITY') returned [None]!"
|
||||||
uidval = imapobj._get_untagged_response('UIDVALIDITY')
|
self._uidvalidity = long(uidval[-1])
|
||||||
assert uidval != [None], "response('UIDVALIDITY') returned [None]!"
|
return self._uidvalidity
|
||||||
return long(uidval[-1])
|
|
||||||
finally:
|
finally:
|
||||||
self.imapserver.releaseconnection(imapobj)
|
self.imapserver.releaseconnection(imapobj)
|
||||||
|
|
||||||
|
@ -83,8 +83,10 @@ class MaildirFolder(BaseFolder):
|
|||||||
"""Return the absolute file path to the Maildir folder (sans cur|new)"""
|
"""Return the absolute file path to the Maildir folder (sans cur|new)"""
|
||||||
return self._fullname
|
return self._fullname
|
||||||
|
|
||||||
def getuidvalidity(self):
|
def get_uidvalidity(self):
|
||||||
"""Maildirs have no notion of uidvalidity, so we just return a magic
|
"""Retrieve the current connections UIDVALIDITY value
|
||||||
|
|
||||||
|
Maildirs have no notion of uidvalidity, so we just return a magic
|
||||||
token."""
|
token."""
|
||||||
return 42
|
return 42
|
||||||
|
|
||||||
|
@ -70,7 +70,7 @@ class MachineUI(UIBase):
|
|||||||
def validityproblem(s, folder):
|
def validityproblem(s, folder):
|
||||||
s._printData('validityproblem', "%s\n%s\n%s\n%s" % \
|
s._printData('validityproblem', "%s\n%s\n%s\n%s" % \
|
||||||
(folder.getname(), folder.getrepository().getname(),
|
(folder.getname(), folder.getrepository().getname(),
|
||||||
folder.getsaveduidvalidity(), folder.getuidvalidity()))
|
folder.getsaveduidvalidity(), folder.get_uidvalidity()))
|
||||||
|
|
||||||
def connecting(s, hostname, port):
|
def connecting(s, hostname, port):
|
||||||
s._printData('connecting', "%s\n%s" % (hostname, str(port)))
|
s._printData('connecting', "%s\n%s" % (hostname, str(port)))
|
||||||
|
@ -309,9 +309,9 @@ class UIBase(object):
|
|||||||
def validityproblem(self, folder):
|
def validityproblem(self, folder):
|
||||||
self.logger.warning("UID validity problem for folder %s (repo %s) "
|
self.logger.warning("UID validity problem for folder %s (repo %s) "
|
||||||
"(saved %d; got %d); skipping it. Please see FAQ "
|
"(saved %d; got %d); skipping it. Please see FAQ "
|
||||||
"and manual how to handle this." % \
|
"and manual on how to handle this." % \
|
||||||
(folder, folder.getrepository(),
|
(folder, folder.getrepository(),
|
||||||
folder.getsaveduidvalidity(), folder.getuidvalidity()))
|
folder.getsaveduidvalidity(), folder.get_uidvalidity()))
|
||||||
|
|
||||||
def loadmessagelist(self, repos, folder):
|
def loadmessagelist(self, repos, folder):
|
||||||
self.logger.debug("Loading message list for %s[%s]" % (
|
self.logger.debug("Loading message list for %s[%s]" % (
|
||||||
|
Loading…
Reference in New Issue
Block a user