Remove upload neguid pass from sync logic
In order to optimize performance, we fold the 1st and 2nd pass of our sync strategy into one. They were essentially doing the same thing: uploading a message to the other side. The only difference was that in one case we have a negative UID locally, and in the other case, we have a positive one already. This saves some time, as we don't have to run through that function on IMAP servers anyway (they always have positive UIDs), and 2nd were we stalling further copying until phase 1 was finished. So uploading a single new message would prevent us from starting to copy existing regular messages. Signed-off-by: Sebastian Spaeth <Sebastian@SSpaeth.de> Signed-off-by: Nicolas Sebrecht <nicolas.s-dev@laposte.net>
This commit is contained in:
parent
12e11429b5
commit
e20d8b9679
@ -16,6 +16,10 @@ New Features
|
|||||||
Changes
|
Changes
|
||||||
-------
|
-------
|
||||||
|
|
||||||
|
* Reduced our sync logic from 4 passes to 3 passes (integrating upload of
|
||||||
|
"new" and "existing" messages into one function). This should result in a
|
||||||
|
slight speedup.
|
||||||
|
|
||||||
Bug Fixes
|
Bug Fixes
|
||||||
---------
|
---------
|
||||||
|
|
||||||
|
@ -222,81 +222,6 @@ class BaseFolder:
|
|||||||
for uid in uidlist:
|
for uid in uidlist:
|
||||||
self.deletemessage(uid)
|
self.deletemessage(uid)
|
||||||
|
|
||||||
def syncmessagesto_neguid_msg(self, uid, dstfolder, statusfolder,
|
|
||||||
register = 1):
|
|
||||||
"""Copy a single message from self to dests.
|
|
||||||
|
|
||||||
This is called by meth:`syncmessagesto_neguid`, possibly in a
|
|
||||||
new thread. It does not return anything.
|
|
||||||
|
|
||||||
:param dstfolder: A BaseFolder-derived instance
|
|
||||||
:param statusfolder: A LocalStatusFolder instance
|
|
||||||
:param register: If True, output that a new thread was created
|
|
||||||
(and register it with the ui)."""
|
|
||||||
successobject = None
|
|
||||||
successuid = None
|
|
||||||
|
|
||||||
if register:
|
|
||||||
self.ui.registerthread(self.getaccountname())
|
|
||||||
self.ui.copyingmessage(uid, self, [dstfolder])
|
|
||||||
|
|
||||||
message = self.getmessage(uid)
|
|
||||||
flags = self.getmessageflags(uid)
|
|
||||||
rtime = self.getmessagetime(uid)
|
|
||||||
|
|
||||||
#Save messages to dstfolder and see if a valid UID was returned
|
|
||||||
successuid = dstfolder.savemessage(uid, message, flags, rtime)
|
|
||||||
|
|
||||||
#Succeeded? -> IMAP actually assigned a UID
|
|
||||||
#If successuid remained negative, no server was willing to assign us
|
|
||||||
#an UID. Ignore message.
|
|
||||||
if successuid >= 0:
|
|
||||||
# Copy the message to the statusfolder
|
|
||||||
statusfolder.savemessage(successuid, message, flags, rtime)
|
|
||||||
# Copy to its new name son the local server and delete
|
|
||||||
# the one without a UID.
|
|
||||||
self.savemessage(successuid, message, flags, rtime)
|
|
||||||
#TODO: above means, we read in the message and write it out
|
|
||||||
#to the very same dir with a different UID. Investigate if we
|
|
||||||
#cannot simply rename it.
|
|
||||||
|
|
||||||
# delete the negative uid message. We have it with a good UID now.
|
|
||||||
self.deletemessage(uid)
|
|
||||||
|
|
||||||
|
|
||||||
def syncmessagesto_neguid(self, dstfolder, statusfolder):
|
|
||||||
"""Pass 1 of folder synchronization.
|
|
||||||
|
|
||||||
Look for messages in self with a negative uid. These are
|
|
||||||
messages in Maildirs that were not added by us. Try to add them
|
|
||||||
to the dstfolder. If that succeeds, get the new UID, add
|
|
||||||
it to the statusfolder, add it to local for real, and delete the
|
|
||||||
old fake (negative) one.
|
|
||||||
|
|
||||||
:param dstfolder: A BaseFolder-derived instance
|
|
||||||
:param statusfolder: A LocalStatusFolder instance"""
|
|
||||||
|
|
||||||
uidlist = [uid for uid in self.getmessageuidlist() if uid < 0]
|
|
||||||
threads = []
|
|
||||||
|
|
||||||
for uid in uidlist:
|
|
||||||
if dstfolder.suggeststhreads():
|
|
||||||
dstfolder.waitforthread()
|
|
||||||
thread = threadutil.InstanceLimitedThread(\
|
|
||||||
dstfolder.getcopyinstancelimit(),
|
|
||||||
target = self.syncmessagesto_neguid_msg,
|
|
||||||
name = "New msg sync from %s" % self.getvisiblename(),
|
|
||||||
args = (uid, dstfolder, statusfolder))
|
|
||||||
thread.setDaemon(1)
|
|
||||||
thread.start()
|
|
||||||
threads.append(thread)
|
|
||||||
else:
|
|
||||||
self.syncmessagesto_neguid_msg(uid, dstfolder, statusfolder,
|
|
||||||
register = 0)
|
|
||||||
#wait for all uploads to finish
|
|
||||||
for thread in threads:
|
|
||||||
thread.join()
|
|
||||||
|
|
||||||
def copymessageto(self, uid, dstfolder, statusfolder, register = 1):
|
def copymessageto(self, uid, dstfolder, statusfolder, register = 1):
|
||||||
"""Copies a message from self to dst if needed, updating the status
|
"""Copies a message from self to dst if needed, updating the status
|
||||||
|
|
||||||
@ -311,33 +236,46 @@ class BaseFolder:
|
|||||||
# self.getmessage(). So, don't call self.getmessage unless
|
# self.getmessage(). So, don't call self.getmessage unless
|
||||||
# really needed.
|
# really needed.
|
||||||
try:
|
try:
|
||||||
if register:
|
if register: # output that we start a new thread
|
||||||
self.ui.registerthread(self.getaccountname())
|
self.ui.registerthread(self.getaccountname())
|
||||||
|
|
||||||
message = None
|
message = None
|
||||||
flags = self.getmessageflags(uid)
|
flags = self.getmessageflags(uid)
|
||||||
rtime = self.getmessagetime(uid)
|
rtime = self.getmessagetime(uid)
|
||||||
|
|
||||||
if dstfolder.uidexists(uid):
|
if uid > 0 and dstfolder.uidexists(uid):
|
||||||
# dst has message with that UID already, only update status
|
# dst has message with that UID already, only update status
|
||||||
statusfolder.savemessage(uid, None, flags, rtime)
|
statusfolder.savemessage(uid, None, flags, rtime)
|
||||||
return
|
return
|
||||||
|
|
||||||
# really need to copy to dst...
|
|
||||||
self.ui.copyingmessage(uid, self, [dstfolder])
|
self.ui.copyingmessage(uid, self, [dstfolder])
|
||||||
|
|
||||||
# If any of the destinations actually stores the message body,
|
# If any of the destinations actually stores the message body,
|
||||||
# load it up.
|
# load it up.
|
||||||
if dstfolder.storesmessages():
|
if dstfolder.storesmessages():
|
||||||
message = self.getmessage(uid)
|
|
||||||
|
|
||||||
|
message = self.getmessage(uid)
|
||||||
|
#Succeeded? -> IMAP actually assigned a UID. If newid
|
||||||
|
#remained negative, no server was willing to assign us an
|
||||||
|
#UID. If newid is 0, saving succeeded, but we could not
|
||||||
|
#retrieve the new UID. Ignore message in this case.
|
||||||
newuid = dstfolder.savemessage(uid, message, flags, rtime)
|
newuid = dstfolder.savemessage(uid, message, flags, rtime)
|
||||||
if newuid > 0 and newuid != uid:
|
if newuid > 0:
|
||||||
# Change the local uid.
|
if newuid != uid:
|
||||||
|
# Got new UID, change the local uid.
|
||||||
|
#TODO: Maildir could do this with a rename rather than
|
||||||
|
#load/save/del operation, IMPLEMENT a changeuid()
|
||||||
|
#function or so.
|
||||||
self.savemessage(newuid, message, flags, rtime)
|
self.savemessage(newuid, message, flags, rtime)
|
||||||
self.deletemessage(uid)
|
self.deletemessage(uid)
|
||||||
uid = newuid
|
uid = newuid
|
||||||
|
# Save uploaded status in the statusfolder
|
||||||
statusfolder.savemessage(uid, message, flags, rtime)
|
statusfolder.savemessage(uid, message, flags, rtime)
|
||||||
|
else:
|
||||||
|
raise UserWarning("Trying to save msg (uid %d) on folder "
|
||||||
|
"%s returned invalid uid %d" % \
|
||||||
|
(uid,
|
||||||
|
dstfolder.getvisiblename(),
|
||||||
|
newuid))
|
||||||
except (KeyboardInterrupt):
|
except (KeyboardInterrupt):
|
||||||
raise
|
raise
|
||||||
except:
|
except:
|
||||||
@ -347,10 +285,10 @@ class BaseFolder:
|
|||||||
raise
|
raise
|
||||||
|
|
||||||
def syncmessagesto_copy(self, dstfolder, statusfolder):
|
def syncmessagesto_copy(self, dstfolder, statusfolder):
|
||||||
"""Pass2: Copy locally existing messages
|
"""Pass1: Copy locally existing messages not on the other side
|
||||||
|
|
||||||
This will copy messages with a valid UID but are not on the
|
This will copy messages to dstfolder that exist locally but are
|
||||||
other side yet. The strategy is:
|
not in the statusfolder yet. The strategy is:
|
||||||
|
|
||||||
1) Look for messages present in self but not in statusfolder.
|
1) Look for messages present in self but not in statusfolder.
|
||||||
2) invoke copymessageto() on those which:
|
2) invoke copymessageto() on those which:
|
||||||
@ -359,7 +297,7 @@ class BaseFolder:
|
|||||||
"""
|
"""
|
||||||
threads = []
|
threads = []
|
||||||
|
|
||||||
copylist = filter(lambda uid: uid>=0 and not \
|
copylist = filter(lambda uid: not \
|
||||||
statusfolder.uidexists(uid),
|
statusfolder.uidexists(uid),
|
||||||
self.getmessageuidlist())
|
self.getmessageuidlist())
|
||||||
for uid in copylist:
|
for uid in copylist:
|
||||||
@ -381,7 +319,7 @@ class BaseFolder:
|
|||||||
thread.join()
|
thread.join()
|
||||||
|
|
||||||
def syncmessagesto_delete(self, dstfolder, statusfolder):
|
def syncmessagesto_delete(self, dstfolder, statusfolder):
|
||||||
"""Pass 3: Remove locally deleted messages on dst
|
"""Pass 2: Remove locally deleted messages on dst
|
||||||
|
|
||||||
Get all UIDS in statusfolder but not self. These are messages
|
Get all UIDS in statusfolder but not self. These are messages
|
||||||
that were deleted in 'self'. Delete those from dstfolder and
|
that were deleted in 'self'. Delete those from dstfolder and
|
||||||
@ -397,7 +335,7 @@ class BaseFolder:
|
|||||||
folder.deletemessages(deletelist)
|
folder.deletemessages(deletelist)
|
||||||
|
|
||||||
def syncmessagesto_flags(self, dstfolder, statusfolder):
|
def syncmessagesto_flags(self, dstfolder, statusfolder):
|
||||||
"""Pass 4: Flag synchronization
|
"""Pass 3: Flag synchronization
|
||||||
|
|
||||||
Compare flag mismatches in self with those in statusfolder. If
|
Compare flag mismatches in self with those in statusfolder. If
|
||||||
msg has a valid UID and exists on dstfolder (has not e.g. been
|
msg has a valid UID and exists on dstfolder (has not e.g. been
|
||||||
@ -449,15 +387,12 @@ class BaseFolder:
|
|||||||
This is the high level entry for syncing messages in one direction.
|
This is the high level entry for syncing messages in one direction.
|
||||||
Syncsteps are:
|
Syncsteps are:
|
||||||
|
|
||||||
Pass1: Transfer new local messages
|
Pass1: Copy locally existing messages
|
||||||
Upload msg with negative/no UIDs to dstfolder. dstfolder
|
|
||||||
might assign that message a new UID. Update statusfolder.
|
|
||||||
|
|
||||||
Pass2: Copy locally existing messages
|
|
||||||
Copy messages in self, but not statusfolder to dstfolder if not
|
Copy messages in self, but not statusfolder to dstfolder if not
|
||||||
already in dstfolder. Update statusfolder.
|
already in dstfolder. dstfolder might assign a new UID (e.g. if
|
||||||
|
uploading to IMAP). Update statusfolder.
|
||||||
|
|
||||||
Pass3: Remove locally deleted messages
|
Pass2: Remove locally deleted messages
|
||||||
Get all UIDS in statusfolder but not self. These are messages
|
Get all UIDS in statusfolder but not self. These are messages
|
||||||
that were deleted in 'self'. Delete those from dstfolder and
|
that were deleted in 'self'. Delete those from dstfolder and
|
||||||
statusfolder.
|
statusfolder.
|
||||||
@ -466,18 +401,16 @@ class BaseFolder:
|
|||||||
uids present (except for potential negative uids that couldn't
|
uids present (except for potential negative uids that couldn't
|
||||||
be placed anywhere).
|
be placed anywhere).
|
||||||
|
|
||||||
Pass4: Synchronize flag changes
|
Pass3: Synchronize flag changes
|
||||||
Compare flag mismatches in self with those in statusfolder. If
|
Compare flag mismatches in self with those in statusfolder. If
|
||||||
msg has a valid UID and exists on dstfolder (has not e.g. been
|
msg has a valid UID and exists on dstfolder (has not e.g. been
|
||||||
deleted there), sync the flag change to both dstfolder and
|
deleted there), sync the flag change to both dstfolder and
|
||||||
statusfolder.
|
statusfolder.
|
||||||
|
|
||||||
|
|
||||||
:param dstfolder: Folderinstance to sync the msgs to.
|
:param dstfolder: Folderinstance to sync the msgs to.
|
||||||
:param statusfolder: LocalStatus instance to sync against.
|
:param statusfolder: LocalStatus instance to sync against.
|
||||||
"""
|
"""
|
||||||
passes = [('uploading negative UIDs', self.syncmessagesto_neguid),
|
passes = [('copying messages' , self.syncmessagesto_copy),
|
||||||
('copying messages' , self.syncmessagesto_copy),
|
|
||||||
('deleting messages' , self.syncmessagesto_delete),
|
('deleting messages' , self.syncmessagesto_delete),
|
||||||
('syncing flags' , self.syncmessagesto_flags)]
|
('syncing flags' , self.syncmessagesto_flags)]
|
||||||
|
|
||||||
@ -490,5 +423,4 @@ class BaseFolder:
|
|||||||
self.ui.warn("ERROR attempting to sync flags " \
|
self.ui.warn("ERROR attempting to sync flags " \
|
||||||
+ "for account " + self.getaccountname() \
|
+ "for account " + self.getaccountname() \
|
||||||
+ ":" + traceback.format_exc())
|
+ ":" + traceback.format_exc())
|
||||||
|
|
||||||
raise
|
raise
|
||||||
|
@ -242,11 +242,6 @@ class MappingFolderMixIn:
|
|||||||
self._mb.deletemessages(self, self._uidlist(self.r2l, uidlist))
|
self._mb.deletemessages(self, self._uidlist(self.r2l, uidlist))
|
||||||
self._mapped_delete(uidlist)
|
self._mapped_delete(uidlist)
|
||||||
|
|
||||||
#def syncmessagesto_neguid_msg(self, uid, dest, applyto, register = 1):
|
|
||||||
# does not need changes because it calls functions that make the changes
|
|
||||||
# same goes for all other sync messages types.
|
|
||||||
|
|
||||||
|
|
||||||
# Define a class for local part of IMAP.
|
# Define a class for local part of IMAP.
|
||||||
class MappedIMAPFolder(MappingFolderMixIn, IMAPFolder):
|
class MappedIMAPFolder(MappingFolderMixIn, IMAPFolder):
|
||||||
def __init__(self, *args, **kwargs):
|
def __init__(self, *args, **kwargs):
|
||||||
|
Loading…
x
Reference in New Issue
Block a user