Merge branch 'next' of github.com:OfflineIMAP/offlineimap into next

This commit is contained in:
Eygene Ryabinkin 2015-01-31 10:08:07 +03:00
commit fd79baade5
6 changed files with 94 additions and 45 deletions

@ -5,13 +5,67 @@ ChangeLog
:website: http://offlineimap.org
OfflineIMAP v6.5.7-rc3 (2015- - )
===================================
OfflineIMAP v6.5.7-rc2 (2015-01-18)
===================================
Notes
-----
This release candidate should be minor for most users.
The best points are about SSL not falling back on other authentication methods
when failing, better RAM footprint and reduced I/O access.
Documentation had our attention, too.
There's some code cleanups and code refactoring, as usual.
Features
--------
* Do not keep reloading pyhtonfile, make it stateful.
* HACKING: how to create tags.
* MANUAL: add minor sample on how to retrieve a password with a helper python file.
Fixes
-----
* Make OS-default CA certificate file to be requested explicitely.
* SSL: do not fallback on other authentication mode if it fails.
* Fix regression introduced while style patching.
* API documentation: properly auto-document main class, fixes.
* ui: Machine: remove offending param for a _printData() call.
* Drop caches after having processed folders.
Changes
-------
* Fix unexpected garbage code.
* Properly re-raise exception to save original tracebacks.
* Refactoring: avoid redefining various Python keywords.
* Code: improvements of comments and more style consistency.
* Configuration file: better design and other small improvements.
* nametrans documentation: fix minor error.
* Unused import removal.
* Add a note about the incorrect rendering of the docstring with Sphinx.
* Errors handling: log the messages with level ERROR.
* MAINTAINERS: add mailing list maintainers.
* Fixed copyright statement.
* COPYING: fix unexpected characters.
OfflineIMAP v6.5.7-rc1 (2015-01-07)
===================================
Notes
-----
I think it's time for a new release candidate. Our release cycle are long
I think it's time for a new release candidate. Our release cycles are long
enough and users are asked to use the current TIP of the next branch to test
our recent patches.
@ -114,7 +168,7 @@ OfflineIMAP v6.5.6 (2014-05-14)
to Tomasz Żok)
OfflineIMAP v6.5.6-RC1 (2014-05-14)
OfflineIMAP v6.5.6-rc1 (2014-05-14)
===================================
* Add knob to invoke folderfilter dynamically on each sync (GitHub#73)

@ -2,7 +2,7 @@ __all__ = ['OfflineImap']
__productname__ = 'OfflineIMAP'
__version__ = "6.5.7"
__revision__ = "-rc1"
__revision__ = "-rc2"
__bigversion__ = __version__ + __revision__
__copyright__ = "Copyright 2002-2015 John Goerzen & contributors"
__author__ = "John Goerzen"

@ -45,10 +45,6 @@ class LocalStatusFolder(BaseFolder):
def isnewfolder(self):
return not os.path.exists(self.filename)
# Interface from BaseFolder
def getname(self):
return self.name
# Interface from BaseFolder
def getfullname(self):
return self.filename
@ -117,33 +113,33 @@ class LocalStatusFolder(BaseFolder):
self.messagelist = {}
return
# loop as many times as version, and update format
for i in range(1, self.cur_version+1):
file = open(self.filename, "rt")
# Loop as many times as version, and update format.
for i in range(1, self.cur_version + 1):
self.messagelist = {}
line = file.readline().strip()
cache = open(self.filename, "rt")
line = cache.readline().strip()
# convert from format v1
if line == (self.magicline % 1):
self.ui._msg('Upgrading LocalStatus cache from version 1 to version 2 for %s:%s' %\
(self.repository, self))
self.readstatus_v1(file)
file.close()
# Format is up to date. break.
if line == (self.magicline % self.cur_version):
break
# Convert from format v1.
elif line == (self.magicline % 1):
self.ui._msg('Upgrading LocalStatus cache from version 1'
'to version 2 for %s:%s'% (self.repository, self))
self.readstatus_v1(cache)
cache.close()
self.save()
# NOTE: Add other format transitions here in the future.
# elif line == (self.magicline % 2):
# self.ui._msg('Upgrading LocalStatus cache from version 2 to version 3 for %s:%s' %\
# (self.repository, self))
# self.readstatus_v2(file)
# file.close()
# file.save()
# self.ui._msg(u'Upgrading LocalStatus cache from version 2'
# 'to version 3 for %s:%s'% (self.repository, self))
# self.readstatus_v2(cache)
# cache.close()
# cache.save()
# format is up to date. break
elif line == (self.magicline % self.cur_version):
break
# something is wrong
# Something is wrong.
else:
errstr = "Unrecognized cache magicline in '%s'" % self.filename
self.ui.warn(errstr)
@ -152,14 +148,14 @@ class LocalStatusFolder(BaseFolder):
if not line:
# The status file is empty - should not have happened,
# but somehow did.
errstr = "Cache file '%s' is empty. Closing..." % self.filename
errstr = "Cache file '%s' is empty."% self.filename
self.ui.warn(errstr)
file.close()
cache.close()
return
assert(line == (self.magicline % self.cur_version))
self.readstatus(file)
file.close()
self.readstatus(cache)
cache.close()
def dropmessagelistcache(self):
self.messagelist = None

@ -15,15 +15,15 @@
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
import os
import re
from sys import exc_info
from threading import Lock
from .Base import BaseFolder
try:
import sqlite3 as sqlite
except:
pass #fail only if needed later on, not on import
from .Base import BaseFolder
class LocalStatusSQLiteFolder(BaseFolder):
"""LocalStatus backend implemented with an SQLite database
@ -56,7 +56,8 @@ class LocalStatusSQLiteFolder(BaseFolder):
if not os.path.exists(dirname):
os.makedirs(dirname)
if not os.path.isdir(dirname):
raise UserWarning("SQLite database path '%s' is not a directory." % dirname)
raise UserWarning("SQLite database path '%s' is not a directory."%
dirname)
# dblock protects against concurrent writes in same connection
self._dblock = Lock()
@ -67,14 +68,15 @@ class LocalStatusSQLiteFolder(BaseFolder):
except NameError:
# sqlite import had failed
raise UserWarning('SQLite backend chosen, but no sqlite python '
'bindings available. Please install.'), None, exc_info()[2]
'bindings available. Please install.'), None, exc_info()[2]
#Make sure sqlite is in multithreading SERIALIZE mode
assert sqlite.threadsafety == 1, 'Your sqlite is not multithreading safe.'
#Test if db version is current enough and if db is readable.
try:
cursor = self.connection.execute("SELECT value from metadata WHERE key='db_version'")
cursor = self.connection.execute(
"SELECT value from metadata WHERE key='db_version'")
except sqlite.DatabaseError:
#db file missing or corrupt, recreate it.
self.__create_db()
@ -88,9 +90,6 @@ class LocalStatusSQLiteFolder(BaseFolder):
def storesmessages(self):
return False
def getname(self):
return self.name
def getfullname(self):
return self.filename

@ -44,13 +44,13 @@ class Repository(object):
name = account.getconf('remoterepository')
# We don't support Maildirs on the remote side.
typemap = {'IMAP': IMAPRepository,
'Gmail': GmailRepository}
'Gmail': GmailRepository}
elif reqtype == 'local':
name = account.getconf('localrepository')
typemap = {'IMAP': MappedIMAPRepository,
'Maildir': MaildirRepository,
'GmailMaildir': GmailMaildirRepository}
'Maildir': MaildirRepository,
'GmailMaildir': GmailMaildirRepository}
elif reqtype == 'status':
# create and return a LocalStatusRepository
@ -69,7 +69,7 @@ class Repository(object):
errstr = ("Could not find section '%s' in configuration. Required "
"for account '%s'." % ('Repository %s' % name, account))
raise OfflineImapError(errstr, OfflineImapError.ERROR.REPO), \
None, exc_info()[2]
None, exc_info()[2]
try:
repo = typemap[repostype]
@ -77,7 +77,7 @@ class Repository(object):
errstr = "'%s' repository not supported for '%s' repositories." \
% (repostype, reqtype)
raise OfflineImapError(errstr, OfflineImapError.ERROR.REPO), \
None, exc_info()[2]
None, exc_info()[2]
return repo(name, account)

@ -539,7 +539,7 @@ class UIBase(object):
abortsleep = False
while sleepsecs > 0 and not abortsleep:
if account.get_abort_event():
abortsleep = True
abortsleep = True
else:
abortsleep = self.sleeping(10, sleepsecs)
sleepsecs -= 10