2002-06-19 05:39:00 +01:00
|
|
|
# IMAP repository 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
|
2003-04-16 20:23:45 +01:00
|
|
|
# the Free Software Foundation; either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
2002-06-19 05:39:00 +01:00
|
|
|
#
|
|
|
|
# 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
|
2006-08-12 05:15:55 +01:00
|
|
|
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
2002-06-19 05:39:00 +01:00
|
|
|
|
|
|
|
from Base import BaseRepository
|
2003-04-18 03:18:34 +01:00
|
|
|
from offlineimap import folder, imaputil, imapserver
|
|
|
|
from offlineimap.folder.UIDMaps import MappedIMAPFolder
|
|
|
|
from offlineimap.threadutil import ExitNotifyThread
|
2008-03-03 02:21:33 -06:00
|
|
|
import re, types, os, netrc, errno
|
2002-07-03 14:04:40 +01:00
|
|
|
from threading import *
|
2002-06-19 05:39:00 +01:00
|
|
|
|
|
|
|
class IMAPRepository(BaseRepository):
|
2003-04-18 03:18:34 +01:00
|
|
|
def __init__(self, reposname, account):
|
|
|
|
"""Initialize an IMAPRepository object."""
|
|
|
|
BaseRepository.__init__(self, reposname, account)
|
|
|
|
self.imapserver = imapserver.ConfigedIMAPServer(self)
|
2002-06-19 06:55:12 +01:00
|
|
|
self.folders = None
|
2002-06-21 07:51:21 +01:00
|
|
|
self.nametrans = lambda foldername: foldername
|
2002-07-04 07:10:51 +01:00
|
|
|
self.folderfilter = lambda foldername: 1
|
2002-07-09 03:32:35 +01:00
|
|
|
self.folderincludes = []
|
2002-08-09 22:10:38 +01:00
|
|
|
self.foldersort = cmp
|
2003-04-18 03:18:34 +01:00
|
|
|
localeval = self.localeval
|
|
|
|
if self.config.has_option(self.getsection(), 'nametrans'):
|
|
|
|
self.nametrans = localeval.eval(self.getconf('nametrans'),
|
|
|
|
{'re': re})
|
|
|
|
if self.config.has_option(self.getsection(), 'folderfilter'):
|
|
|
|
self.folderfilter = localeval.eval(self.getconf('folderfilter'),
|
|
|
|
{'re': re})
|
|
|
|
if self.config.has_option(self.getsection(), 'folderincludes'):
|
|
|
|
self.folderincludes = localeval.eval(self.getconf('folderincludes'),
|
|
|
|
{'re': re})
|
|
|
|
if self.config.has_option(self.getsection(), 'foldersort'):
|
|
|
|
self.foldersort = localeval.eval(self.getconf('foldersort'),
|
|
|
|
{'re': re})
|
|
|
|
|
|
|
|
def startkeepalive(self):
|
|
|
|
keepalivetime = self.getkeepalive()
|
|
|
|
if not keepalivetime: return
|
|
|
|
self.kaevent = Event()
|
|
|
|
self.kathread = ExitNotifyThread(target = self.imapserver.keepalive,
|
|
|
|
name = "Keep alive " + self.getname(),
|
|
|
|
args = (keepalivetime, self.kaevent))
|
|
|
|
self.kathread.setDaemon(1)
|
|
|
|
self.kathread.start()
|
|
|
|
|
|
|
|
def stopkeepalive(self, abrupt = 0):
|
|
|
|
if not hasattr(self, 'kaevent'):
|
|
|
|
# Keepalive is not active.
|
|
|
|
return
|
|
|
|
|
|
|
|
self.kaevent.set()
|
|
|
|
if not abrupt:
|
|
|
|
self.kathread.join()
|
|
|
|
del self.kathread
|
|
|
|
del self.kaevent
|
|
|
|
|
|
|
|
def holdordropconnections(self):
|
|
|
|
if not self.getholdconnectionopen():
|
|
|
|
self.dropconnections()
|
|
|
|
|
|
|
|
def dropconnections(self):
|
|
|
|
self.imapserver.close()
|
|
|
|
|
|
|
|
def getholdconnectionopen(self):
|
|
|
|
return self.getconfboolean("holdconnectionopen", 0)
|
|
|
|
|
|
|
|
def getkeepalive(self):
|
|
|
|
return self.getconfint("keepalive", 0)
|
2002-06-19 06:22:21 +01:00
|
|
|
|
2002-06-19 07:16:19 +01:00
|
|
|
def getsep(self):
|
|
|
|
return self.imapserver.delim
|
|
|
|
|
2003-04-18 03:18:34 +01:00
|
|
|
def gethost(self):
|
2007-07-04 22:00:14 +01:00
|
|
|
host = None
|
2006-10-17 20:55:03 +01:00
|
|
|
localeval = self.localeval
|
|
|
|
|
|
|
|
if self.config.has_option(self.getsection(), 'remotehosteval'):
|
2007-07-04 22:00:14 +01:00
|
|
|
host = self.getconf('remotehosteval')
|
|
|
|
if host != None:
|
|
|
|
return localeval.eval(host)
|
2006-10-17 20:55:03 +01:00
|
|
|
|
2007-07-04 22:00:14 +01:00
|
|
|
host = self.getconf('remotehost')
|
|
|
|
if host != None:
|
|
|
|
return host
|
2003-04-18 03:18:34 +01:00
|
|
|
|
|
|
|
def getuser(self):
|
2007-07-04 22:00:14 +01:00
|
|
|
user = None
|
2006-10-17 20:55:03 +01:00
|
|
|
localeval = self.localeval
|
|
|
|
|
|
|
|
if self.config.has_option(self.getsection(), 'remoteusereval'):
|
2007-07-04 22:00:14 +01:00
|
|
|
user = self.getconf('remoteusereval')
|
|
|
|
if user != None:
|
|
|
|
return localeval.eval(user)
|
2006-10-17 20:55:03 +01:00
|
|
|
|
2007-07-04 22:00:14 +01:00
|
|
|
user = self.getconf('remoteuser')
|
|
|
|
if user != None:
|
|
|
|
return user
|
2003-04-18 03:18:34 +01:00
|
|
|
|
2008-03-03 02:21:33 -06:00
|
|
|
try:
|
|
|
|
netrcentry = netrc.netrc().authentificator(self.gethost())
|
|
|
|
except IOError, inst:
|
|
|
|
if inst.errno != errno.ENOENT:
|
|
|
|
raise
|
|
|
|
else:
|
|
|
|
if netrcentry:
|
|
|
|
return netrcentry[0]
|
2003-04-18 03:18:34 +01:00
|
|
|
|
|
|
|
def getport(self):
|
|
|
|
return self.getconfint('remoteport', None)
|
|
|
|
|
|
|
|
def getssl(self):
|
|
|
|
return self.getconfboolean('ssl', 0)
|
|
|
|
|
2008-05-23 14:58:18 -05:00
|
|
|
def getsslclientcert(self):
|
|
|
|
return self.getconf('sslclientcert', None)
|
|
|
|
|
|
|
|
def getsslclientkey(self):
|
|
|
|
return self.getconf('sslclientkey', None)
|
|
|
|
|
2003-04-18 03:18:34 +01:00
|
|
|
def getpreauthtunnel(self):
|
|
|
|
return self.getconf('preauthtunnel', None)
|
|
|
|
|
|
|
|
def getreference(self):
|
|
|
|
return self.getconf('reference', '""')
|
|
|
|
|
|
|
|
def getmaxconnections(self):
|
|
|
|
return self.getconfint('maxconnections', 1)
|
|
|
|
|
|
|
|
def getexpunge(self):
|
|
|
|
return self.getconfboolean('expunge', 1)
|
|
|
|
|
|
|
|
def getpassword(self):
|
2007-07-04 22:00:14 +01:00
|
|
|
passwd = None
|
2006-10-17 20:55:03 +01:00
|
|
|
localeval = self.localeval
|
|
|
|
|
|
|
|
if self.config.has_option(self.getsection(), 'remotepasseval'):
|
2007-07-04 22:00:14 +01:00
|
|
|
passwd = self.getconf('remotepasseval')
|
|
|
|
if passwd != None:
|
|
|
|
return localeval.eval(passwd)
|
2006-10-17 20:55:03 +01:00
|
|
|
|
2003-04-18 03:18:34 +01:00
|
|
|
password = self.getconf('remotepass', None)
|
|
|
|
if password != None:
|
|
|
|
return password
|
|
|
|
passfile = self.getconf('remotepassfile', None)
|
|
|
|
if passfile != None:
|
|
|
|
fd = open(os.path.expanduser(passfile))
|
2003-04-29 02:52:03 +01:00
|
|
|
password = fd.readline().strip()
|
|
|
|
fd.close()
|
2007-07-04 22:00:14 +01:00
|
|
|
return password
|
2008-03-03 02:27:13 -06:00
|
|
|
|
2008-03-03 02:21:33 -06:00
|
|
|
try:
|
|
|
|
netrcentry = netrc.netrc().authenticators(self.gethost())
|
|
|
|
except IOError, inst:
|
|
|
|
if inst.errno != errno.ENOENT:
|
|
|
|
raise
|
|
|
|
else:
|
|
|
|
if netrcentry:
|
|
|
|
user = self.getconf('remoteuser')
|
|
|
|
if user == None or user == netrcentry[0]:
|
|
|
|
return netrcentry[2]
|
2003-04-18 03:18:34 +01:00
|
|
|
return None
|
|
|
|
|
2002-06-20 03:55:24 +01:00
|
|
|
def getfolder(self, foldername):
|
2003-04-18 03:18:34 +01:00
|
|
|
return self.getfoldertype()(self.imapserver, foldername,
|
|
|
|
self.nametrans(foldername),
|
|
|
|
self.accountname, self)
|
|
|
|
|
|
|
|
def getfoldertype(self):
|
|
|
|
return folder.IMAP.IMAPFolder
|
2002-06-20 03:55:24 +01:00
|
|
|
|
2007-07-05 05:04:14 +01:00
|
|
|
def connect(self):
|
|
|
|
imapobj = self.imapserver.acquireconnection()
|
|
|
|
self.imapserver.releaseconnection(imapobj)
|
|
|
|
|
2007-07-06 17:46:29 +01:00
|
|
|
def forgetfolders(self):
|
|
|
|
self.folders = None
|
|
|
|
|
2002-06-19 06:22:21 +01:00
|
|
|
def getfolders(self):
|
2002-06-19 06:55:12 +01:00
|
|
|
if self.folders != None:
|
|
|
|
return self.folders
|
2002-06-19 06:22:21 +01:00
|
|
|
retval = []
|
2002-07-04 01:15:32 +01:00
|
|
|
imapobj = self.imapserver.acquireconnection()
|
|
|
|
try:
|
2002-07-05 12:46:55 +01:00
|
|
|
listresult = imapobj.list(directory = self.imapserver.reference)[1]
|
2002-07-04 01:15:32 +01:00
|
|
|
finally:
|
|
|
|
self.imapserver.releaseconnection(imapobj)
|
|
|
|
for string in listresult:
|
2003-04-18 03:18:34 +01:00
|
|
|
if string == None or \
|
|
|
|
(type(string) == types.StringType and string == ''):
|
2002-10-16 06:43:02 +01:00
|
|
|
# Bug in imaplib: empty strings in results from
|
|
|
|
# literals.
|
|
|
|
continue
|
2002-06-19 06:22:21 +01:00
|
|
|
flags, delim, name = imaputil.imapsplit(string)
|
2002-07-06 01:22:23 +01:00
|
|
|
flaglist = [x.lower() for x in imaputil.flagsplit(flags)]
|
|
|
|
if '\\noselect' in flaglist:
|
2002-06-19 06:22:21 +01:00
|
|
|
continue
|
2002-07-04 07:10:51 +01:00
|
|
|
foldername = imaputil.dequote(name)
|
|
|
|
if not self.folderfilter(foldername):
|
|
|
|
continue
|
2003-04-18 03:18:34 +01:00
|
|
|
retval.append(self.getfoldertype()(self.imapserver, foldername,
|
|
|
|
self.nametrans(foldername),
|
|
|
|
self.accountname, self))
|
2003-04-18 05:31:25 +01:00
|
|
|
if len(self.folderincludes):
|
|
|
|
imapobj = self.imapserver.acquireconnection()
|
|
|
|
try:
|
|
|
|
for foldername in self.folderincludes:
|
2003-04-18 08:06:04 +01:00
|
|
|
try:
|
|
|
|
imapobj.select(foldername, readonly = 1)
|
|
|
|
except ValueError:
|
|
|
|
continue
|
|
|
|
retval.append(self.getfoldertype()(self.imapserver,
|
|
|
|
foldername,
|
|
|
|
self.nametrans(foldername),
|
|
|
|
self.accountname, self))
|
2003-04-18 05:31:25 +01:00
|
|
|
finally:
|
|
|
|
self.imapserver.releaseconnection(imapobj)
|
|
|
|
|
2002-08-09 22:10:38 +01:00
|
|
|
retval.sort(lambda x, y: self.foldersort(x.getvisiblename(), y.getvisiblename()))
|
2002-06-19 06:55:12 +01:00
|
|
|
self.folders = retval
|
2002-06-19 06:22:21 +01:00
|
|
|
return retval
|
2003-04-18 03:18:34 +01:00
|
|
|
|
|
|
|
def makefolder(self, foldername):
|
|
|
|
#if self.getreference() != '""':
|
|
|
|
# newname = self.getreference() + self.getsep() + foldername
|
|
|
|
#else:
|
|
|
|
# newname = foldername
|
|
|
|
newname = foldername
|
|
|
|
imapobj = self.imapserver.acquireconnection()
|
|
|
|
try:
|
|
|
|
result = imapobj.create(newname)
|
|
|
|
if result[0] != 'OK':
|
|
|
|
raise RuntimeError, "Repository %s could not create folder %s: %s" % (self.getname(), foldername, str(result))
|
|
|
|
finally:
|
|
|
|
self.imapserver.releaseconnection(imapobj)
|
|
|
|
|
|
|
|
class MappedIMAPRepository(IMAPRepository):
|
|
|
|
def getfoldertype(self):
|
|
|
|
return MappedIMAPFolder
|