2002-06-21 03:20:42 +01:00
|
|
|
# Local status cache 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-21 03:20:42 +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-21 03:20:42 +01:00
|
|
|
|
2011-05-25 21:09:09 +02:00
|
|
|
from offlineimap.folder.LocalStatus import LocalStatusFolder, magicline
|
2011-05-07 11:00:57 +02:00
|
|
|
from offlineimap.folder.LocalStatusSQLite import LocalStatusSQLiteFolder
|
2012-02-05 11:57:19 +01:00
|
|
|
from offlineimap.repository.Base import BaseRepository
|
2011-03-11 22:13:21 +01:00
|
|
|
import os
|
|
|
|
import re
|
2002-06-21 03:20:42 +01:00
|
|
|
|
|
|
|
class LocalStatusRepository(BaseRepository):
|
2003-04-18 03:18:34 +01:00
|
|
|
def __init__(self, reposname, account):
|
|
|
|
BaseRepository.__init__(self, reposname, account)
|
2011-09-16 10:54:25 +02:00
|
|
|
# Root directory in which the LocalStatus folders reside
|
|
|
|
self.root = os.path.join(account.getaccountmeta(), 'LocalStatus')
|
|
|
|
# statusbackend can be 'plain' or 'sqlite'
|
2011-05-07 11:00:57 +02:00
|
|
|
backend = self.account.getconf('status_backend', 'plain')
|
|
|
|
if backend == 'sqlite':
|
|
|
|
self._backend = 'sqlite'
|
|
|
|
self.LocalStatusFolderClass = LocalStatusSQLiteFolder
|
2011-09-16 10:54:25 +02:00
|
|
|
self.root += '-sqlite'
|
2011-05-07 11:00:57 +02:00
|
|
|
elif backend == 'plain':
|
|
|
|
self._backend = 'plain'
|
|
|
|
self.LocalStatusFolderClass = LocalStatusFolder
|
|
|
|
else:
|
|
|
|
raise SyntaxWarning("Unknown status_backend '%s' for account '%s'" \
|
|
|
|
% (backend, account.name))
|
|
|
|
|
2011-09-16 10:54:25 +02:00
|
|
|
if not os.path.exists(self.root):
|
2012-02-05 11:31:54 +01:00
|
|
|
os.mkdir(self.root, 0o700)
|
2011-05-07 11:00:57 +02:00
|
|
|
|
Reuse LocalStatus() folders rather than recreate instances
If we ask twice for a LocalStatusFolder via getfolder(), we would
get a newly created instance each time. This can lead to problems,
as e.g. write locks protecting files only work within the same Folder
instance. Make it so, that we cache all Folder instances that we have
asked for and hand back the existing one if we ask again for it,
rather than recreate a new instance.
Also, make getfolders() a noop for LocalStatus. We attempted to
derive the foldername from the name of the LocalStatusfile. However,
this is not really possible, as we do file name mangling
(".$" -> "dot", "/" -> ".") and there is no way to get the original folder
name from the LocalStatus file name anyway.
This commit could potentially solve the "file not found" errors, that people
have been seeing with their LocalStatusCache files. If we have 2
instances of a LocalStatusFolder pointing to the same file, our locking
system would not work.
Signed-off-by: Sebastian Spaeth <Sebastian@SSpaeth.de>
2012-09-01 01:02:20 +02:00
|
|
|
# self._folders is a dict of name:LocalStatusFolders()
|
|
|
|
self._folders = {}
|
2002-06-21 03:20:42 +01:00
|
|
|
|
|
|
|
def getsep(self):
|
|
|
|
return '.'
|
|
|
|
|
|
|
|
def getfolderfilename(self, foldername):
|
2011-08-17 16:11:00 +02:00
|
|
|
"""Return the full path of the status file
|
|
|
|
|
|
|
|
This mimics the path that Folder().getfolderbasename() would return"""
|
|
|
|
if not foldername:
|
|
|
|
basename = '.'
|
|
|
|
else: #avoid directory hierarchies and file names such as '/'
|
|
|
|
basename = foldername.replace('/', '.')
|
|
|
|
# replace with literal 'dot' if final path name is '.' as '.' is
|
|
|
|
# an invalid file name.
|
|
|
|
basename = re.sub('(^|\/)\.$','\\1dot', basename)
|
2011-09-16 10:54:25 +02:00
|
|
|
return os.path.join(self.root, basename)
|
2002-06-21 03:20:42 +01:00
|
|
|
|
|
|
|
def makefolder(self, foldername):
|
2011-05-07 11:00:57 +02:00
|
|
|
"""Create a LocalStatus Folder
|
|
|
|
|
|
|
|
Empty Folder for plain backend. NoOp for sqlite backend as those
|
|
|
|
are created on demand."""
|
|
|
|
if self._backend == 'sqlite':
|
2011-09-15 15:37:52 +02:00
|
|
|
return # noop for sqlite which creates on-demand
|
|
|
|
|
|
|
|
if self.account.dryrun:
|
|
|
|
return # bail out in dry-run mode
|
2011-05-07 11:00:57 +02:00
|
|
|
|
2008-05-06 18:23:48 -05:00
|
|
|
filename = self.getfolderfilename(foldername)
|
2008-04-17 00:09:11 -05:00
|
|
|
file = open(filename + ".tmp", "wt")
|
2011-05-25 21:09:09 +02:00
|
|
|
file.write(magicline + '\n')
|
2002-06-21 03:20:42 +01:00
|
|
|
file.close()
|
2008-03-03 12:59:40 -06:00
|
|
|
os.rename(filename + ".tmp", filename)
|
2002-06-21 03:20:42 +01:00
|
|
|
# Invalidate the cache.
|
2012-09-01 02:58:14 +02:00
|
|
|
self._folders = {}
|
2002-06-21 03:20:42 +01:00
|
|
|
|
2011-05-05 15:59:28 +02:00
|
|
|
def getfolder(self, foldername):
|
|
|
|
"""Return the Folder() object for a foldername"""
|
Reuse LocalStatus() folders rather than recreate instances
If we ask twice for a LocalStatusFolder via getfolder(), we would
get a newly created instance each time. This can lead to problems,
as e.g. write locks protecting files only work within the same Folder
instance. Make it so, that we cache all Folder instances that we have
asked for and hand back the existing one if we ask again for it,
rather than recreate a new instance.
Also, make getfolders() a noop for LocalStatus. We attempted to
derive the foldername from the name of the LocalStatusfile. However,
this is not really possible, as we do file name mangling
(".$" -> "dot", "/" -> ".") and there is no way to get the original folder
name from the LocalStatus file name anyway.
This commit could potentially solve the "file not found" errors, that people
have been seeing with their LocalStatusCache files. If we have 2
instances of a LocalStatusFolder pointing to the same file, our locking
system would not work.
Signed-off-by: Sebastian Spaeth <Sebastian@SSpaeth.de>
2012-09-01 01:02:20 +02:00
|
|
|
if foldername in self._folders:
|
|
|
|
return self._folders[foldername]
|
|
|
|
|
|
|
|
folder = self.LocalStatusFolderClass(foldername, self)
|
|
|
|
self._folders[foldername] = folder
|
|
|
|
return folder
|
2011-05-05 15:59:28 +02:00
|
|
|
|
2002-06-21 03:20:42 +01:00
|
|
|
def getfolders(self):
|
2013-07-21 23:00:23 +04:00
|
|
|
"""Returns a list of all cached folders.
|
2002-06-21 04:40:58 +01:00
|
|
|
|
Reuse LocalStatus() folders rather than recreate instances
If we ask twice for a LocalStatusFolder via getfolder(), we would
get a newly created instance each time. This can lead to problems,
as e.g. write locks protecting files only work within the same Folder
instance. Make it so, that we cache all Folder instances that we have
asked for and hand back the existing one if we ask again for it,
rather than recreate a new instance.
Also, make getfolders() a noop for LocalStatus. We attempted to
derive the foldername from the name of the LocalStatusfile. However,
this is not really possible, as we do file name mangling
(".$" -> "dot", "/" -> ".") and there is no way to get the original folder
name from the LocalStatus file name anyway.
This commit could potentially solve the "file not found" errors, that people
have been seeing with their LocalStatusCache files. If we have 2
instances of a LocalStatusFolder pointing to the same file, our locking
system would not work.
Signed-off-by: Sebastian Spaeth <Sebastian@SSpaeth.de>
2012-09-01 01:02:20 +02:00
|
|
|
Does nothing for this backend. We mangle the folder file names
|
|
|
|
(see getfolderfilename) so we can not derive folder names from
|
|
|
|
the file names that we have available. TODO: need to store a
|
|
|
|
list of folder names somehow?"""
|
|
|
|
pass
|
2002-06-21 03:20:42 +01:00
|
|
|
|
2011-05-05 15:59:28 +02:00
|
|
|
def forgetfolders(self):
|
|
|
|
"""Forgets the cached list of folders, if any. Useful to run
|
|
|
|
after a sync run."""
|
Reuse LocalStatus() folders rather than recreate instances
If we ask twice for a LocalStatusFolder via getfolder(), we would
get a newly created instance each time. This can lead to problems,
as e.g. write locks protecting files only work within the same Folder
instance. Make it so, that we cache all Folder instances that we have
asked for and hand back the existing one if we ask again for it,
rather than recreate a new instance.
Also, make getfolders() a noop for LocalStatus. We attempted to
derive the foldername from the name of the LocalStatusfile. However,
this is not really possible, as we do file name mangling
(".$" -> "dot", "/" -> ".") and there is no way to get the original folder
name from the LocalStatus file name anyway.
This commit could potentially solve the "file not found" errors, that people
have been seeing with their LocalStatusCache files. If we have 2
instances of a LocalStatusFolder pointing to the same file, our locking
system would not work.
Signed-off-by: Sebastian Spaeth <Sebastian@SSpaeth.de>
2012-09-01 01:02:20 +02:00
|
|
|
self._folders = {}
|