123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203 |
- # Copyright (C) 2007-2016 John Goerzen & contributors.
- #
- # 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
- # the Free Software Foundation; either version 2 of the License, or
- # (at your option) any later version.
- #
- # 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
- # Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
- try:
- from urllib import urlencode
- except ImportError: # python3
- from urllib.parse import urlencode
- import sys
- import time
- import logging
- from threading import currentThread
- import offlineimap
- from offlineimap.ui.UIBase import UIBase
- protocol = '7.2.0'
- class MachineLogFormatter(logging.Formatter):
- """urlencodes any outputted line, to avoid multi-line output"""
- def format(s, record):
- # Mapping of log levels to historic tag names
- severity_map = {
- 'info': 'msg',
- 'warning': 'warn',
- }
- line = super(MachineLogFormatter, s).format(record)
- severity = record.levelname.lower()
- if severity in severity_map:
- severity = severity_map[severity]
- if hasattr(record, "machineui"):
- command = record.machineui["command"]
- whoami = record.machineui["id"]
- else:
- command = ""
- whoami = currentThread().getName()
- prefix = "%s:%s"% (command, urlencode([('', whoami)])[1:])
- return "%s:%s:%s"% (severity, prefix, urlencode([('', line)])[1:])
- class MachineUI(UIBase):
- def __init__(s, config, loglevel=logging.INFO):
- super(MachineUI, s).__init__(config, loglevel)
- s._log_con_handler.createLock()
- """lock needed to block on password input"""
- # Set up the formatter that urlencodes the strings...
- s._log_con_handler.setFormatter(MachineLogFormatter())
- # Arguments:
- # - handler: must be method from s.logger that reflects
- # the severity of the passed message
- # - command: command that produced this message
- # - msg: the message itself
- def _printData(s, handler, command, msg):
- handler(msg,
- extra = {
- 'machineui': {
- 'command': command,
- 'id': currentThread().getName(),
- }
- })
- def _msg(s, msg):
- s._printData(s.logger.info, '_display', msg)
- def warn(s, msg, minor=0):
- # TODO, remove and cleanup the unused minor stuff
- s._printData(s.logger.warning, '', msg)
- def registerthread(s, account):
- super(MachineUI, s).registerthread(account)
- s._printData(s.logger.info, 'registerthread', account)
- def unregisterthread(s, thread):
- UIBase.unregisterthread(s, thread)
- s._printData(s.logger.info, 'unregisterthread', thread.getName())
- def debugging(s, debugtype):
- s._printData(s.logger.debug, 'debugging', debugtype)
- def acct(s, accountname):
- s._printData(s.logger.info, 'acct', accountname)
- def acctdone(s, accountname):
- s._printData(s.logger.info, 'acctdone', accountname)
- def validityproblem(s, folder):
- s._printData(s.logger.warning, 'validityproblem', "%s\n%s\n%s\n%s"%
- (folder.getname(), folder.getrepository().getname(),
- folder.get_saveduidvalidity(), folder.get_uidvalidity()))
- def connecting(s, reposname, hostname, port):
- s._printData(s.logger.info, 'connecting', "%s\n%s\n%s"% (hostname,
- str(port), reposname))
- def syncfolders(s, srcrepos, destrepos):
- s._printData(s.logger.info, 'syncfolders', "%s\n%s"% (s.getnicename(srcrepos),
- s.getnicename(destrepos)))
- def syncingfolder(s, srcrepos, srcfolder, destrepos, destfolder):
- s._printData(s.logger.info, 'syncingfolder', "%s\n%s\n%s\n%s\n"%
- (s.getnicename(srcrepos), srcfolder.getname(),
- s.getnicename(destrepos), destfolder.getname()))
- def loadmessagelist(s, repos, folder):
- s._printData(s.logger.info, 'loadmessagelist', "%s\n%s"% (s.getnicename(repos),
- folder.getvisiblename()))
- def messagelistloaded(s, repos, folder, count):
- s._printData(s.logger.info, 'messagelistloaded', "%s\n%s\n%d"%
- (s.getnicename(repos), folder.getname(), count))
- def syncingmessages(s, sr, sf, dr, df):
- s._printData(s.logger.info, 'syncingmessages', "%s\n%s\n%s\n%s\n"%
- (s.getnicename(sr), sf.getname(), s.getnicename(dr),
- df.getname()))
- def ignorecopyingmessage(s, uid, srcfolder, destfolder):
- s._printData(s.logger.info, 'ignorecopyingmessage', "%d\n%s\n%s\n%s[%s]"%
- (uid, s.getnicename(srcfolder), srcfolder.getname(),
- s.getnicename(destfolder), destfolder))
- def copyingmessage(s, uid, num, num_to_copy, srcfolder, destfolder):
- s._printData(s.logger.info, 'copyingmessage', "%d\n%s\n%s\n%s[%s]"%
- (uid, s.getnicename(srcfolder), srcfolder.getname(),
- s.getnicename(destfolder), destfolder))
- def folderlist(s, list):
- return ("\f".join(["%s\t%s"% (s.getnicename(x), x.getname()) for x in list]))
- def uidlist(s, list):
- return ("\f".join([str(u) for u in list]))
- def deletingmessages(s, uidlist, destlist):
- ds = s.folderlist(destlist)
- s._printData(s.logger.info, 'deletingmessages', "%s\n%s"% (s.uidlist(uidlist), ds))
- def addingflags(s, uidlist, flags, dest):
- s._printData(s.logger.info, "addingflags", "%s\n%s\n%s"% (s.uidlist(uidlist),
- "\f".join(flags),
- dest))
- def deletingflags(s, uidlist, flags, dest):
- s._printData(s.logger.info, 'deletingflags', "%s\n%s\n%s"% (s.uidlist(uidlist),
- "\f".join(flags),
- dest))
- def threadException(s, thread):
- s._printData(s.logger.warning, 'threadException', "%s\n%s"%
- (thread.getName(), s.getThreadExceptionString(thread)))
- s.delThreadDebugLog(thread)
- s.terminate(100)
- def terminate(s, exitstatus=0, errortitle='', errormsg=''):
- s._printData(s.logger.info, 'terminate', "%d\n%s\n%s"% (exitstatus, errortitle, errormsg))
- sys.exit(exitstatus)
- def mainException(s):
- s._printData(s.logger.warning, 'mainException', s.getMainExceptionString())
- def threadExited(s, thread):
- s._printData(s.logger.info, 'threadExited', thread.getName())
- UIBase.threadExited(s, thread)
- def sleeping(s, sleepsecs, remainingsecs):
- s._printData(s.logger.info, 'sleeping', "%d\n%d"% (sleepsecs, remainingsecs))
- if sleepsecs > 0:
- time.sleep(sleepsecs)
- return 0
- def getpass(s, accountname, config, errmsg=None):
- if errmsg:
- s._printData(s.logger.warning,
- 'getpasserror', "%s\n%s"% (accountname, errmsg),
- False)
- s._log_con_handler.acquire() # lock the console output
- try:
- s._printData(s.logger.info, 'getpass', accountname)
- return (sys.stdin.readline()[:-1])
- finally:
- s._log_con_handler.release()
- def init_banner(s):
- s._printData(s.logger.info, 'protocol', protocol)
- s._printData(s.logger.info, 'initbanner', offlineimap.banner)
- def callhook(s, msg):
- s._printData(s.logger.info, 'callhook', msg)
|