2002-06-19 07:22:21 +02:00
|
|
|
# Base folder support
|
|
|
|
# Copyright (C) 2002 John Goerzen
|
|
|
|
# <jgoerzen@complete.org>
|
|
|
|
#
|
|
|
|
# This program is free software; you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program; if not, write to the Free Software
|
|
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
|
2002-06-21 08:51:21 +02:00
|
|
|
import __main__
|
2002-07-04 04:14:07 +02:00
|
|
|
from threading import *
|
2002-07-04 05:59:19 +02:00
|
|
|
from offlineimap import threadutil
|
|
|
|
from offlineimap.threadutil import InstanceLimitedThread
|
2002-06-21 08:51:21 +02:00
|
|
|
|
2002-06-19 07:22:21 +02:00
|
|
|
class BaseFolder:
|
|
|
|
def getname(self):
|
|
|
|
"""Returns name"""
|
|
|
|
return self.name
|
|
|
|
|
2002-07-04 03:35:05 +02:00
|
|
|
def suggeststhreads(self):
|
|
|
|
"""Returns true if this folder suggests using threads for actions;
|
|
|
|
false otherwise. Probably only IMAP will return true."""
|
|
|
|
return 0
|
|
|
|
|
|
|
|
def waitforthread(self):
|
|
|
|
"""For threading folders, waits until there is a resource available
|
|
|
|
before firing off a thread. For all others, returns immediately."""
|
|
|
|
pass
|
|
|
|
|
2002-07-04 05:59:19 +02:00
|
|
|
def getcopyinstancelimit(self):
|
|
|
|
"""For threading folders, returns the instancelimitname for
|
|
|
|
InstanceLimitedThreads."""
|
|
|
|
raise NotImplementedException
|
|
|
|
|
2002-06-21 08:51:21 +02:00
|
|
|
def getvisiblename(self):
|
|
|
|
return self.name
|
|
|
|
|
2002-06-19 07:22:21 +02:00
|
|
|
def getroot(self):
|
|
|
|
"""Returns the root of the folder, in a folder-specific fashion."""
|
|
|
|
return self.root
|
|
|
|
|
|
|
|
def getsep(self):
|
|
|
|
"""Returns the separator for this folder type."""
|
|
|
|
return self.sep
|
|
|
|
|
|
|
|
def getfullname(self):
|
2002-06-20 08:26:28 +02:00
|
|
|
if self.getroot():
|
|
|
|
return self.getroot() + self.getsep() + self.getname()
|
|
|
|
else:
|
|
|
|
return self.getname()
|
2002-06-19 07:22:21 +02:00
|
|
|
|
2002-06-20 05:33:23 +02:00
|
|
|
def isuidvalidityok(self, remotefolder):
|
|
|
|
raise NotImplementedException
|
|
|
|
|
|
|
|
def getuidvalidity(self):
|
|
|
|
raise NotImplementedException
|
|
|
|
|
|
|
|
def saveuidvalidity(self, newval):
|
|
|
|
raise NotImplementedException
|
|
|
|
|
|
|
|
def cachemessagelist(self):
|
|
|
|
"""Reads the message list from disk or network and stores it in
|
|
|
|
memory for later use. This list will not be re-read from disk or
|
|
|
|
memory unless this function is called again."""
|
|
|
|
raise NotImplementedException
|
|
|
|
|
|
|
|
def getmessagelist(self):
|
2002-06-20 05:50:58 +02:00
|
|
|
"""Gets the current message list.
|
|
|
|
You must call cachemessagelist() before calling this function!"""
|
2002-06-20 05:33:23 +02:00
|
|
|
raise NotImplementedException
|
|
|
|
|
2002-06-20 05:50:58 +02:00
|
|
|
def getmessage(self, uid):
|
|
|
|
"""Returns the content of the specified message."""
|
|
|
|
raise NotImplementedException
|
|
|
|
|
2002-06-21 03:12:52 +02:00
|
|
|
def savemessage(self, uid, content, flags):
|
2002-06-20 06:37:36 +02:00
|
|
|
"""Writes a new message, with the specified uid.
|
|
|
|
If the uid is < 0, the backend should assign a new uid and return it.
|
2002-06-21 03:12:52 +02:00
|
|
|
|
|
|
|
If the backend cannot assign a new uid, it returns the uid passed in
|
|
|
|
WITHOUT saving the message.
|
|
|
|
|
2002-06-20 06:37:36 +02:00
|
|
|
IMAP backend should be the only one that can assign a new uid.
|
2002-06-20 13:39:27 +02:00
|
|
|
|
|
|
|
If the uid is > 0, the backend should set the uid to this, if it can.
|
2002-06-21 03:12:52 +02:00
|
|
|
If it cannot set the uid to that, it will save it anyway.
|
2002-06-20 13:39:27 +02:00
|
|
|
It will return the uid assigned in any case.
|
|
|
|
"""
|
2002-06-20 05:50:58 +02:00
|
|
|
raise NotImplementedException
|
|
|
|
|
|
|
|
def getmessageflags(self, uid):
|
|
|
|
"""Returns the flags for the specified message."""
|
|
|
|
raise NotImplementedException
|
|
|
|
|
|
|
|
def savemessageflags(self, uid, flags):
|
|
|
|
"""Sets the specified message's flags to the given set."""
|
|
|
|
raise NotImplementedException
|
|
|
|
|
2002-06-20 06:37:36 +02:00
|
|
|
def addmessageflags(self, uid, flags):
|
|
|
|
"""Adds the specified flags to the message's flag set. If a given
|
|
|
|
flag is already present, it will not be duplicated."""
|
|
|
|
newflags = self.getmessageflags(uid)
|
|
|
|
for flag in flags:
|
|
|
|
if not flag in newflags:
|
|
|
|
newflags.append(flag)
|
|
|
|
newflags.sort()
|
|
|
|
self.savemessageflags(uid, newflags)
|
|
|
|
|
2002-07-03 07:05:49 +02:00
|
|
|
def addmessagesflags(self, uidlist, flags):
|
|
|
|
for uid in uidlist:
|
|
|
|
self.addmessageflags(uid)
|
|
|
|
|
2002-06-20 06:37:36 +02:00
|
|
|
def deletemessageflags(self, uid, flags):
|
|
|
|
"""Removes each flag given from the message's flag set. If a given
|
|
|
|
flag is already removed, no action will be taken for that flag."""
|
|
|
|
newflags = self.getmessageflags(uid)
|
|
|
|
for flag in flags:
|
|
|
|
if flag in newflags:
|
|
|
|
newflags.remove(flag)
|
|
|
|
newflags.sort()
|
|
|
|
self.savemessageflags(uid, newflags)
|
|
|
|
|
2002-06-20 05:50:58 +02:00
|
|
|
def deletemessage(self, uid):
|
|
|
|
raise NotImplementedException
|
|
|
|
|
2002-06-21 03:55:06 +02:00
|
|
|
def deletemessages(self, uidlist):
|
|
|
|
for uid in uidlist:
|
|
|
|
self.deletemessage(uid)
|
|
|
|
|
2002-06-25 06:47:09 +02:00
|
|
|
def syncmessagesto_neguid(self, dest, applyto):
|
|
|
|
"""Pass 1 of folder synchronization.
|
2002-06-20 05:50:58 +02:00
|
|
|
|
2002-06-25 06:47:09 +02:00
|
|
|
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 dests,
|
|
|
|
and once that succeeds, get the UID, add it to the others for real,
|
|
|
|
add it to local for real, and delete the fake one."""
|
2002-06-20 06:37:36 +02:00
|
|
|
|
|
|
|
for uid in self.getmessagelist().keys():
|
|
|
|
if uid >= 0:
|
|
|
|
continue
|
2002-06-21 07:29:12 +02:00
|
|
|
__main__.ui.copyingmessage(uid, self, applyto)
|
2002-06-20 06:37:36 +02:00
|
|
|
successobject = None
|
|
|
|
successuid = None
|
|
|
|
message = self.getmessage(uid)
|
|
|
|
flags = self.getmessageflags(uid)
|
|
|
|
for tryappend in applyto:
|
2002-06-21 03:12:52 +02:00
|
|
|
successuid = tryappend.savemessage(uid, message, flags)
|
2002-06-20 06:37:36 +02:00
|
|
|
if successuid > 0:
|
|
|
|
successobject = tryappend
|
|
|
|
break
|
|
|
|
# Did we succeed?
|
|
|
|
if successobject != None:
|
|
|
|
# Copy the message to the other remote servers.
|
|
|
|
for appendserver in [x for x in applyto if x != successobject]:
|
2002-06-21 03:12:52 +02:00
|
|
|
appendserver.savemessage(successuid, message, flags)
|
2002-06-20 06:37:36 +02:00
|
|
|
# Copy it to its new name on the local server and delete
|
|
|
|
# the one without a UID.
|
2002-06-21 03:12:52 +02:00
|
|
|
self.savemessage(successuid, message, flags)
|
2002-06-20 06:37:36 +02:00
|
|
|
self.deletemessage(uid)
|
|
|
|
else:
|
2002-06-21 04:02:36 +02:00
|
|
|
# Did not find any server to take this message. Ignore.
|
2002-06-20 06:37:36 +02:00
|
|
|
pass
|
|
|
|
|
2002-07-04 03:35:05 +02:00
|
|
|
def copymessageto(self, uid, applyto):
|
|
|
|
__main__.ui.copyingmessage(uid, self, applyto)
|
|
|
|
message = self.getmessage(uid)
|
|
|
|
flags = self.getmessageflags(uid)
|
|
|
|
for object in applyto:
|
|
|
|
newuid = object.savemessage(uid, message, flags)
|
|
|
|
if newuid > 0 and newuid != uid:
|
|
|
|
# Change the local uid.
|
|
|
|
self.savemessage(newuid, message, flags)
|
|
|
|
self.deletemessage(uid)
|
|
|
|
uid = newuid
|
|
|
|
|
|
|
|
|
2002-06-25 06:47:09 +02:00
|
|
|
def syncmessagesto_copy(self, dest, applyto):
|
|
|
|
"""Pass 2 of folder synchronization.
|
|
|
|
|
|
|
|
Look for messages present in self but not in dest. If any, add
|
|
|
|
them to dest."""
|
2002-07-04 03:35:05 +02:00
|
|
|
threads = []
|
2002-06-20 05:50:58 +02:00
|
|
|
|
|
|
|
for uid in self.getmessagelist().keys():
|
2002-06-20 06:37:36 +02:00
|
|
|
if uid < 0: # Ignore messages that pass 1 missed.
|
|
|
|
continue
|
2002-06-20 05:50:58 +02:00
|
|
|
if not uid in dest.getmessagelist():
|
2002-07-04 03:35:05 +02:00
|
|
|
if self.suggeststhreads():
|
|
|
|
self.waitforthread()
|
2002-07-04 05:59:19 +02:00
|
|
|
thread = InstanceLimitedThread(\
|
|
|
|
self.getcopyinstancelimit(),
|
|
|
|
target = self.copymessageto,
|
|
|
|
args = (uid, applyto))
|
2002-07-04 03:35:05 +02:00
|
|
|
thread.start()
|
|
|
|
threads.append(thread)
|
|
|
|
else:
|
|
|
|
self.copymessageto(uid, applyto)
|
|
|
|
for thread in threads:
|
|
|
|
thread.join()
|
2002-06-20 05:50:58 +02:00
|
|
|
|
2002-06-25 06:47:09 +02:00
|
|
|
def syncmessagesto_delete(self, dest, applyto):
|
|
|
|
"""Pass 3 of folder synchronization.
|
2002-06-20 05:50:58 +02:00
|
|
|
|
2002-06-25 06:47:09 +02:00
|
|
|
Look for message present in dest but not in self.
|
|
|
|
If any, delete them."""
|
2002-07-03 07:05:49 +02:00
|
|
|
deletelist = []
|
2002-06-20 05:50:58 +02:00
|
|
|
for uid in dest.getmessagelist().keys():
|
2002-06-21 04:02:36 +02:00
|
|
|
if uid < 0:
|
|
|
|
continue
|
2002-06-20 05:50:58 +02:00
|
|
|
if not uid in self.getmessagelist():
|
2002-07-03 07:05:49 +02:00
|
|
|
deletelist.append(uid)
|
|
|
|
if len(deletelist):
|
2002-07-03 07:16:56 +02:00
|
|
|
__main__.ui.deletingmessages(deletelist, applyto)
|
2002-07-03 07:05:49 +02:00
|
|
|
for object in applyto:
|
2002-07-03 07:16:56 +02:00
|
|
|
object.deletemessages(deletelist)
|
2002-06-20 05:50:58 +02:00
|
|
|
|
2002-06-25 06:47:09 +02:00
|
|
|
def syncmessagesto_flags(self, dest, applyto):
|
|
|
|
"""Pass 4 of folder synchronization.
|
2002-06-20 06:37:36 +02:00
|
|
|
|
2002-06-25 06:47:09 +02:00
|
|
|
Look for any flag matching issues -- set dest message to have the
|
|
|
|
same flags that we have."""
|
2002-06-20 06:37:36 +02:00
|
|
|
for uid in self.getmessagelist().keys():
|
|
|
|
if uid < 0: # Ignore messages missed by pass 1
|
|
|
|
continue
|
|
|
|
selfflags = self.getmessageflags(uid)
|
|
|
|
destflags = dest.getmessageflags(uid)
|
2002-06-20 05:50:58 +02:00
|
|
|
|
2002-06-20 06:37:36 +02:00
|
|
|
addflags = [x for x in selfflags if x not in destflags]
|
|
|
|
if len(addflags):
|
2002-06-21 07:29:12 +02:00
|
|
|
__main__.ui.addingflags(uid, addflags, applyto)
|
2002-06-20 06:37:36 +02:00
|
|
|
for object in applyto:
|
2002-06-21 06:04:47 +02:00
|
|
|
object.addmessageflags(uid, addflags)
|
2002-06-20 06:14:54 +02:00
|
|
|
|
2002-06-20 06:37:36 +02:00
|
|
|
delflags = [x for x in destflags if x not in selfflags]
|
|
|
|
if len(delflags):
|
2002-06-21 07:29:12 +02:00
|
|
|
__main__.ui.deletingflags(uid, delflags, applyto)
|
2002-06-20 06:37:36 +02:00
|
|
|
for object in applyto:
|
2002-06-21 06:04:47 +02:00
|
|
|
object.deletemessageflags(uid, delflags)
|
2002-06-20 06:37:36 +02:00
|
|
|
|
2002-06-25 06:47:09 +02:00
|
|
|
def syncmessagesto(self, dest, applyto = None):
|
|
|
|
"""Syncs messages in this folder to the destination.
|
|
|
|
If applyto is specified, it should be a list of folders (don't forget
|
|
|
|
to include dest!) to which all write actions should be applied.
|
|
|
|
It defaults to [dest] if not specified. It is important that
|
|
|
|
the UID generator be listed first in applyto; that is, the other
|
|
|
|
applyto ones should be the ones that "copy" the main action."""
|
|
|
|
if applyto == None:
|
|
|
|
applyto = [dest]
|
|
|
|
|
|
|
|
self.syncmessagesto_neguid(dest, applyto)
|
|
|
|
self.syncmessagesto_copy(dest, applyto)
|
|
|
|
self.syncmessagesto_delete(dest, applyto)
|
|
|
|
|
|
|
|
# Now, the message lists should be identical wrt the uids present.
|
|
|
|
# (except for potential negative uids that couldn't be placed
|
|
|
|
# anywhere)
|
|
|
|
|
|
|
|
self.syncmessagesto_flags(dest, applyto)
|
|
|
|
|
2002-06-20 06:37:36 +02:00
|
|
|
|