2002-10-07 21:59:02 +01:00
# OfflineIMAP initialization code
2007-07-04 18:53:48 +01:00
# Copyright (C) 2002-2007 John Goerzen
2002-10-07 21:59:02 +01:00
# <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-10-07 21:59:02 +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-10-07 21:59:02 +01:00
2007-07-04 18:53:48 +01:00
import imaplib
from offlineimap import imapserver , repository , folder , mbnames , threadutil , version , syncmaster , accounts
2002-10-07 21:59:02 +01:00
from offlineimap . localeval import LocalEval
from offlineimap . threadutil import InstanceLimitedThread , ExitNotifyThread
from offlineimap . ui import UIBase
2003-07-25 03:15:27 +01:00
import re , os , os . path , offlineimap , sys
2003-01-04 05:57:46 +01:00
from offlineimap . CustomConfig import CustomConfigParser
2002-10-07 21:59:02 +01:00
from threading import *
2007-07-06 17:37:58 +01:00
import threading , socket
2002-10-07 21:59:02 +01:00
from getopt import getopt
2003-07-25 03:15:27 +01:00
try :
import fcntl
hasfcntl = 1
except :
hasfcntl = 0
2003-01-30 02:19:53 +01:00
lockfd = None
def lock ( config , ui ) :
2003-07-25 03:15:27 +01:00
global lockfd , hasfcntl
if not hasfcntl :
return
2003-01-30 02:19:53 +01:00
lockfd = open ( config . getmetadatadir ( ) + " /lock " , " w " )
try :
fcntl . flock ( lockfd , fcntl . LOCK_EX | fcntl . LOCK_NB )
except IOError :
ui . locked ( )
ui . terminate ( 1 )
2002-10-07 22:30:35 +01:00
def startup ( versionno ) :
2003-07-25 03:15:27 +01:00
assert versionno == version . versionstr , " Revision of main program ( %s ) does not match that of library ( %s ). Please double-check your PYTHONPATH and installation locations. " % ( versionno , version . versionstr )
2002-10-07 21:59:02 +01:00
options = { }
if ' --help ' in sys . argv [ 1 : ] :
sys . stdout . write ( version . cmdhelp + " \n " )
sys . exit ( 0 )
2003-06-02 20:07:30 +01:00
for optlist in getopt ( sys . argv [ 1 : ] , ' P:1oa:c:d:l:u:h ' ) [ 0 ] :
2002-10-07 21:59:02 +01:00
options [ optlist [ 0 ] ] = optlist [ 1 ]
2003-07-26 03:01:25 +01:00
if options . has_key ( ' -h ' ) :
2002-10-07 21:59:02 +01:00
sys . stdout . write ( version . cmdhelp )
sys . stdout . write ( " \n " )
sys . exit ( 0 )
configfilename = os . path . expanduser ( " ~/.offlineimaprc " )
2003-07-26 03:01:25 +01:00
if options . has_key ( ' -c ' ) :
2002-10-07 21:59:02 +01:00
configfilename = options [ ' -c ' ]
2003-07-26 03:01:25 +01:00
if options . has_key ( ' -P ' ) :
if not options . has_key ( ' -1 ' ) :
2002-10-07 21:59:02 +01:00
sys . stderr . write ( " FATAL: profile mode REQUIRES -1 \n " )
sys . exit ( 100 )
profiledir = options [ ' -P ' ]
os . mkdir ( profiledir )
threadutil . setprofiledir ( profiledir )
sys . stderr . write ( " WARNING: profile mode engaged; \n Potentially large data will be created in " + profiledir + " \n " )
2003-01-04 05:57:46 +01:00
config = CustomConfigParser ( )
2002-10-07 21:59:02 +01:00
if not os . path . exists ( configfilename ) :
sys . stderr . write ( " *** Config file %s does not exist; aborting! \n " % configfilename )
sys . exit ( 1 )
config . read ( configfilename )
2003-01-04 05:57:46 +01:00
ui = offlineimap . ui . detector . findUI ( config , options . get ( ' -u ' ) )
2002-10-07 21:59:02 +01:00
UIBase . setglobalui ( ui )
2003-07-26 03:01:25 +01:00
if options . has_key ( ' -l ' ) :
2003-06-02 20:52:33 +01:00
ui . setlogfd ( open ( options [ ' -l ' ] , ' wt ' ) )
ui . init_banner ( )
2003-07-26 03:01:25 +01:00
if options . has_key ( ' -d ' ) :
2002-10-07 21:59:02 +01:00
for debugtype in options [ ' -d ' ] . split ( ' , ' ) :
ui . add_debug ( debugtype . strip ( ) )
if debugtype == ' imap ' :
imaplib . Debug = 5
2003-04-18 03:18:34 +01:00
if debugtype == ' thread ' :
threading . _VERBOSE = 1
2002-10-07 21:59:02 +01:00
2003-07-26 03:01:25 +01:00
if options . has_key ( ' -o ' ) :
2003-04-18 03:18:34 +01:00
# FIXME: maybe need a better
for section in accounts . getaccountlist ( config ) :
config . remove_option ( ' Account ' + section , " autorefresh " )
2002-10-07 21:59:02 +01:00
2003-01-30 02:19:53 +01:00
lock ( config , ui )
2007-07-07 01:50:43 +01:00
try :
pidfd = open ( config . getmetadatadir ( ) + " /pid " , " w " )
pidfd . write ( os . getpid ( ) )
pidfd . close ( )
except :
pass
2006-12-01 11:59:22 +01:00
try :
if options . has_key ( ' -l ' ) :
sys . stderr = ui . logfile
2007-07-06 17:37:58 +01:00
socktimeout = config . getdefaultint ( " general " , " socktimeout " , 0 )
if socktimeout > 0 :
socket . setdefaulttimeout ( socktimeout )
2006-12-01 11:59:22 +01:00
activeaccounts = config . get ( " general " , " accounts " )
if options . has_key ( ' -a ' ) :
activeaccounts = options [ ' -a ' ]
activeaccounts = activeaccounts . replace ( " " , " " )
activeaccounts = activeaccounts . split ( " , " )
allaccounts = accounts . AccountHashGenerator ( config )
syncaccounts = { }
for account in activeaccounts :
if account not in allaccounts :
if len ( allaccounts ) == 0 :
errormsg = ' The account " %s " does not exist because no accounts are defined! ' % account
else :
errormsg = ' The account " %s " does not exist. Valid accounts are: ' % account
for name in allaccounts . keys ( ) :
errormsg + = ' \n %s ' % name
ui . terminate ( 1 , errortitle = ' Unknown Account " %s " ' % account , errormsg = errormsg )
syncaccounts [ account ] = allaccounts [ account ]
server = None
remoterepos = None
localrepos = None
if options . has_key ( ' -1 ' ) :
threadutil . initInstanceLimit ( " ACCOUNTLIMIT " , 1 )
else :
threadutil . initInstanceLimit ( " ACCOUNTLIMIT " ,
config . getdefaultint ( " general " , " maxsyncaccounts " , 1 ) )
for reposname in config . getsectionlist ( ' Repository ' ) :
for instancename in [ " FOLDER_ " + reposname ,
" MSGCOPY_ " + reposname ] :
if options . has_key ( ' -1 ' ) :
threadutil . initInstanceLimit ( instancename , 1 )
else :
threadutil . initInstanceLimit ( instancename ,
config . getdefaultint ( ' Repository ' + reposname , " maxconnections " , 1 ) )
threadutil . initexitnotify ( )
t = ExitNotifyThread ( target = syncmaster . syncitall ,
name = ' Sync Runner ' ,
kwargs = { ' accounts ' : syncaccounts ,
' config ' : config } )
t . setDaemon ( 1 )
t . start ( )
except :
ui . mainException ( )
2002-10-07 21:59:02 +01:00
try :
2002-10-07 22:17:13 +01:00
threadutil . exitnotifymonitorloop ( threadutil . threadexited )
2002-10-07 21:59:02 +01:00
except SystemExit :
raise
except :
ui . mainException ( ) # Also expected to terminate.