Split offlineimap.run()
Was getting too large, split into an parse_cmd_options and a sync() function. Moving config and ui to self.config and self.ui to make them available through the OfflineImap instance. Signed-off-by: Sebastian Spaeth <Sebastian@SSpaeth.de>
This commit is contained in:
parent
a5eebd4b6b
commit
6b3f429c81
@ -44,7 +44,11 @@ class OfflineImap:
|
|||||||
"""
|
"""
|
||||||
def run(self):
|
def run(self):
|
||||||
"""Parse the commandline and invoke everything"""
|
"""Parse the commandline and invoke everything"""
|
||||||
|
# next line also sets self.config
|
||||||
|
options = self.parse_cmd_options()
|
||||||
|
self.sync(options)
|
||||||
|
|
||||||
|
def parse_cmd_options(self):
|
||||||
parser = OptionParser(version=offlineimap.__version__,
|
parser = OptionParser(version=offlineimap.__version__,
|
||||||
description="%s.\n\n%s" %
|
description="%s.\n\n%s" %
|
||||||
(offlineimap.__copyright__,
|
(offlineimap.__copyright__,
|
||||||
@ -187,19 +191,19 @@ class OfflineImap:
|
|||||||
'of %s' % ', '.join(UI_LIST.keys()))
|
'of %s' % ', '.join(UI_LIST.keys()))
|
||||||
try:
|
try:
|
||||||
# create the ui class
|
# create the ui class
|
||||||
ui = UI_LIST[ui_type.lower()](config)
|
self.ui = UI_LIST[ui_type.lower()](config)
|
||||||
except KeyError:
|
except KeyError:
|
||||||
logging.error("UI '%s' does not exist, choose one of: %s" % \
|
logging.error("UI '%s' does not exist, choose one of: %s" % \
|
||||||
(ui_type,', '.join(UI_LIST.keys())))
|
(ui_type,', '.join(UI_LIST.keys())))
|
||||||
sys.exit(1)
|
sys.exit(1)
|
||||||
setglobalui(ui)
|
setglobalui(self.ui)
|
||||||
|
|
||||||
#set up additional log files
|
#set up additional log files
|
||||||
if options.logfile:
|
if options.logfile:
|
||||||
ui.setlogfd(open(options.logfile, 'wt'))
|
self.ui.setlogfd(open(options.logfile, 'wt'))
|
||||||
|
|
||||||
#welcome blurb
|
#welcome blurb
|
||||||
ui.init_banner()
|
self.ui.init_banner()
|
||||||
|
|
||||||
if options.debugtype:
|
if options.debugtype:
|
||||||
if options.debugtype.lower() == 'all':
|
if options.debugtype.lower() == 'all':
|
||||||
@ -207,13 +211,13 @@ class OfflineImap:
|
|||||||
#force single threading?
|
#force single threading?
|
||||||
if not ('thread' in options.debugtype.split(',') \
|
if not ('thread' in options.debugtype.split(',') \
|
||||||
and not options.singlethreading):
|
and not options.singlethreading):
|
||||||
ui._msg("Debug mode: Forcing to singlethreaded.")
|
self.ui._msg("Debug mode: Forcing to singlethreaded.")
|
||||||
options.singlethreading = True
|
options.singlethreading = True
|
||||||
|
|
||||||
debugtypes = options.debugtype.split(',') + ['']
|
debugtypes = options.debugtype.split(',') + ['']
|
||||||
for type in debugtypes:
|
for type in debugtypes:
|
||||||
type = type.strip()
|
type = type.strip()
|
||||||
ui.add_debug(type)
|
self.ui.add_debug(type)
|
||||||
if type.lower() == 'imap':
|
if type.lower() == 'imap':
|
||||||
imaplib.Debug = 5
|
imaplib.Debug = 5
|
||||||
|
|
||||||
@ -241,47 +245,64 @@ class OfflineImap:
|
|||||||
config.set(section, "folderfilter", folderfilter)
|
config.set(section, "folderfilter", folderfilter)
|
||||||
config.set(section, "folderincludes", folderincludes)
|
config.set(section, "folderincludes", folderincludes)
|
||||||
|
|
||||||
self.config = config
|
if options.logfile:
|
||||||
|
sys.stderr = self.ui.logfile
|
||||||
|
|
||||||
|
socktimeout = config.getdefaultint("general", "socktimeout", 0)
|
||||||
|
if socktimeout > 0:
|
||||||
|
socket.setdefaulttimeout(socktimeout)
|
||||||
|
|
||||||
|
threadutil.initInstanceLimit('ACCOUNTLIMIT',
|
||||||
|
config.getdefaultint('general', 'maxsyncaccounts', 1))
|
||||||
|
|
||||||
|
for reposname in config.getsectionlist('Repository'):
|
||||||
|
for instancename in ["FOLDER_" + reposname,
|
||||||
|
"MSGCOPY_" + reposname]:
|
||||||
|
if options.singlethreading:
|
||||||
|
threadutil.initInstanceLimit(instancename, 1)
|
||||||
|
else:
|
||||||
|
threadutil.initInstanceLimit(instancename,
|
||||||
|
config.getdefaultint('Repository ' + reposname,
|
||||||
|
'maxconnections', 2))
|
||||||
|
self.config = config
|
||||||
|
return options
|
||||||
|
|
||||||
|
def sync(self, options):
|
||||||
|
"""Invoke the correct single/multithread syncing
|
||||||
|
|
||||||
|
self.config is supposed to have been correctly initialized
|
||||||
|
already."""
|
||||||
def sigterm_handler(signum, frame):
|
def sigterm_handler(signum, frame):
|
||||||
# die immediately
|
# die immediately
|
||||||
ui = getglobalui()
|
self.ui.terminate(errormsg="terminating...")
|
||||||
ui.terminate(errormsg="terminating...")
|
signal.signal(signal.SIGTERM, sigterm_handler)
|
||||||
|
|
||||||
signal.signal(signal.SIGTERM,sigterm_handler)
|
|
||||||
|
|
||||||
try:
|
try:
|
||||||
pidfd = open(config.getmetadatadir() + "/pid", "w")
|
pidfd = open(self.config.getmetadatadir() + "/pid", "w")
|
||||||
pidfd.write(str(os.getpid()) + "\n")
|
pidfd.write(str(os.getpid()) + "\n")
|
||||||
pidfd.close()
|
pidfd.close()
|
||||||
except:
|
except:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
try:
|
try:
|
||||||
if options.logfile:
|
activeaccounts = self.config.get("general", "accounts")
|
||||||
sys.stderr = ui.logfile
|
|
||||||
|
|
||||||
socktimeout = config.getdefaultint("general", "socktimeout", 0)
|
|
||||||
if socktimeout > 0:
|
|
||||||
socket.setdefaulttimeout(socktimeout)
|
|
||||||
|
|
||||||
activeaccounts = config.get("general", "accounts")
|
|
||||||
if options.accounts:
|
if options.accounts:
|
||||||
activeaccounts = options.accounts
|
activeaccounts = options.accounts
|
||||||
activeaccounts = activeaccounts.replace(" ", "")
|
activeaccounts = activeaccounts.replace(" ", "")
|
||||||
activeaccounts = activeaccounts.split(",")
|
activeaccounts = activeaccounts.split(",")
|
||||||
allaccounts = accounts.AccountHashGenerator(config)
|
allaccounts = accounts.AccountHashGenerator(self.config)
|
||||||
|
|
||||||
syncaccounts = []
|
syncaccounts = []
|
||||||
for account in activeaccounts:
|
for account in activeaccounts:
|
||||||
if account not in allaccounts:
|
if account not in allaccounts:
|
||||||
if len(allaccounts) == 0:
|
if len(allaccounts) == 0:
|
||||||
errormsg = 'The account "%s" does not exist because no accounts are defined!'%account
|
errormsg = "The account '%s' does not exist because no"\
|
||||||
|
" accounts are defined!" % account
|
||||||
else:
|
else:
|
||||||
errormsg = 'The account "%s" does not exist. Valid accounts are:'%account
|
errormsg = "The account '%s' does not exist. Valid ac"\
|
||||||
for name in allaccounts.keys():
|
"counts are: " % account
|
||||||
errormsg += '\n%s'%name
|
errormsg += ", ".join(allaccounts.keys())
|
||||||
ui.terminate(1, errortitle = 'Unknown Account "%s"'%account, errormsg = errormsg)
|
self.ui.terminate(1, errormsg = errormsg)
|
||||||
if account not in syncaccounts:
|
if account not in syncaccounts:
|
||||||
syncaccounts.append(account)
|
syncaccounts.append(account)
|
||||||
|
|
||||||
@ -289,19 +310,6 @@ class OfflineImap:
|
|||||||
remoterepos = None
|
remoterepos = None
|
||||||
localrepos = None
|
localrepos = None
|
||||||
|
|
||||||
threadutil.initInstanceLimit('ACCOUNTLIMIT',
|
|
||||||
config.getdefaultint('general',
|
|
||||||
'maxsyncaccounts', 1))
|
|
||||||
|
|
||||||
for reposname in config.getsectionlist('Repository'):
|
|
||||||
for instancename in ["FOLDER_" + reposname,
|
|
||||||
"MSGCOPY_" + reposname]:
|
|
||||||
if options.singlethreading:
|
|
||||||
threadutil.initInstanceLimit(instancename, 1)
|
|
||||||
else:
|
|
||||||
threadutil.initInstanceLimit(instancename,
|
|
||||||
config.getdefaultint('Repository ' + reposname,
|
|
||||||
'maxconnections', 2))
|
|
||||||
def sig_handler(sig, frame):
|
def sig_handler(sig, frame):
|
||||||
if sig == signal.SIGUSR1 or sig == signal.SIGHUP:
|
if sig == signal.SIGUSR1 or sig == signal.SIGHUP:
|
||||||
# tell each account to stop sleeping
|
# tell each account to stop sleeping
|
||||||
@ -315,7 +323,7 @@ class OfflineImap:
|
|||||||
signal.signal(signal.SIGUSR2,sig_handler)
|
signal.signal(signal.SIGUSR2,sig_handler)
|
||||||
|
|
||||||
#various initializations that need to be performed:
|
#various initializations that need to be performed:
|
||||||
offlineimap.mbnames.init(config, syncaccounts)
|
offlineimap.mbnames.init(self.config, syncaccounts)
|
||||||
|
|
||||||
#TODO: keep legacy lock for a few versions, then remove.
|
#TODO: keep legacy lock for a few versions, then remove.
|
||||||
self._legacy_lock = open(self.config.getmetadatadir() + "/lock",
|
self._legacy_lock = open(self.config.getmetadatadir() + "/lock",
|
||||||
@ -331,20 +339,20 @@ class OfflineImap:
|
|||||||
|
|
||||||
if options.singlethreading:
|
if options.singlethreading:
|
||||||
#singlethreaded
|
#singlethreaded
|
||||||
self.sync_singlethreaded(syncaccounts, config)
|
self.sync_singlethreaded(syncaccounts)
|
||||||
else:
|
else:
|
||||||
# multithreaded
|
# multithreaded
|
||||||
t = threadutil.ExitNotifyThread(target=syncmaster.syncitall,
|
t = threadutil.ExitNotifyThread(target=syncmaster.syncitall,
|
||||||
name='Sync Runner',
|
name='Sync Runner',
|
||||||
kwargs = {'accounts': syncaccounts,
|
kwargs = {'accounts': syncaccounts,
|
||||||
'config': config})
|
'config': self.config})
|
||||||
t.setDaemon(1)
|
t.setDaemon(1)
|
||||||
t.start()
|
t.start()
|
||||||
threadutil.exitnotifymonitorloop(threadutil.threadexited)
|
threadutil.exitnotifymonitorloop(threadutil.threadexited)
|
||||||
|
|
||||||
ui.terminate()
|
self.ui.terminate()
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
ui.terminate(1, errormsg = 'CTRL-C pressed, aborting...')
|
self.ui.terminate(1, errormsg = 'CTRL-C pressed, aborting...')
|
||||||
return
|
return
|
||||||
except (SystemExit):
|
except (SystemExit):
|
||||||
raise
|
raise
|
||||||
@ -352,13 +360,13 @@ class OfflineImap:
|
|||||||
ui.error(e)
|
ui.error(e)
|
||||||
ui.terminate()
|
ui.terminate()
|
||||||
|
|
||||||
def sync_singlethreaded(self, accs, config):
|
def sync_singlethreaded(self, accs):
|
||||||
"""Executed if we do not want a separate syncmaster thread
|
"""Executed if we do not want a separate syncmaster thread
|
||||||
|
|
||||||
:param accs: A list of accounts that should be synced
|
:param accs: A list of accounts that should be synced
|
||||||
:param config: The CustomConfig object
|
|
||||||
"""
|
"""
|
||||||
for accountname in accs:
|
for accountname in accs:
|
||||||
account = offlineimap.accounts.SyncableAccount(config, accountname)
|
account = offlineimap.accounts.SyncableAccount(self.config,
|
||||||
|
accountname)
|
||||||
threading.currentThread().name = "Account sync %s" % accountname
|
threading.currentThread().name = "Account sync %s" % accountname
|
||||||
account.syncrunner()
|
account.syncrunner()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user