X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/1f780e48af479e7bf9a07eaaa1ab6b41f1ffb17b..f5791b9983fb52a074500ecbd7e8ef7a14a90b22:/wxPython/samples/ide/activegrid/util/aglogging.py diff --git a/wxPython/samples/ide/activegrid/util/aglogging.py b/wxPython/samples/ide/activegrid/util/aglogging.py index 880ff822f2..17ab688e1c 100644 --- a/wxPython/samples/ide/activegrid/util/aglogging.py +++ b/wxPython/samples/ide/activegrid/util/aglogging.py @@ -14,71 +14,362 @@ import sys import os import re import traceback +import logging +import logging.config +from activegrid.util.lang import * +import activegrid.util.objutils as objutils +import activegrid.util.sysutils as sysutils +import activegrid.util.appdirs as appdirs + +LEVEL_FATAL = logging.FATAL +LEVEL_ERROR = logging.ERROR +LEVEL_WARN = logging.WARN +LEVEL_INFO = logging.INFO +LEVEL_DEBUG = logging.DEBUG + +EXCEPTION_INFO = 'exceptionInfo' +loggingInitialized = False + +LOG_MODE_IDE = 1 +LOG_MODE_TESTRUN = 2 +LOG_MODE_RUN = 3 +def initLogging(mode, force=False): + global ag_debugLogger, loggingInitialized + if (force or not loggingInitialized): + loggingInitialized = True + configFile = None + if (mode == LOG_MODE_IDE): + configFile = os.getenv("AG_LOGCONFIG_IDE") + elif (mode == LOG_MODE_TESTRUN): + configFile = os.getenv("AG_LOGCONFIG_PYTESTRUN") + else: + configFile = os.getenv("AG_LOGCONFIG_RUN") + if ((configFile == None) or not os.path.exists(configFile)): + if (mode == LOG_MODE_IDE): + configFile = "IDELog" + elif (mode == LOG_MODE_TESTRUN): + configFile = "TestRunLog" + else: + configFile = "RunLog" + configFile = os.path.join(appdirs.getSystemDir(appdirs.AG_LOGS_DIR), "py" + configFile + ".ini") + if (os.path.exists(configFile)): + print "Using logging configuration file: %s" % configFile + fileConfig(configFile) + else: + print "*** Cannot find logging configuration file (%s) -- setting default logging level to WARN ***" % (configFile) + defaultStream = sys.stderr + if (mode == LOG_MODE_RUN): + defaultStream = sys.stdout + handler = logging.StreamHandler(defaultStream) + handler.setLevel(logging.DEBUG) + handler.setFormatter(logging.Formatter("%(asctime)s %(name)s %(levelname)s: %(message)s")) + logging.getLogger().addHandler(handler) + logging.getLogger().setLevel(logging.WARN) + ag_debugLogger = logging.getLogger("activegrid.debug") + ag_debugLogger.setLevel(logging.DEBUG) + return configFile + +ag_debugLogger = logging.getLogger("activegrid.debug") + +def log(logger, level, msg, *params): + if (logger == None): + logger = ag_debugLogger + apply(logger.log, (level, msg) + params) + +def fatal(logger, msg, *params): + apply(logger.fatal, (msg,) + params) + +def error(logger, msg, *params): + apply(logger.error, (msg,) + params) + +def warn(logger, msg, *params): + apply(logger.warn, (msg,) + params) + +def info(logger, msg, *params): + apply(logger.info, (msg,) + params) + +def debug(logger, msg, *params): + if (logger == None): + logger = ag_debugLogger + apply(logger.debug, (msg,) + params) + +def setLevelFatal(logger): + logger.setLevel(LEVEL_FATAL) + +def setLevelError(logger): + logger.setLevel(LEVEL_ERROR) + +def setLevelWarn(logger): + logger.setLevel(LEVEL_WARN) + +def setLevelInfo(logger): + logger.setLevel(LEVEL_INFO) + +def setLevelDebug(logger): + logger.setLevel(LEVEL_DEBUG) + +def isEnabledForError(logger): + return logger.isEnabledFor(LEVEL_ERROR) + +def isEnabledForWarn(logger): + return logger.isEnabledFor(LEVEL_WARN) + +def isEnabledForInfo(logger): + return logger.isEnabledFor(LEVEL_INFO) + +def isEnabledForDebug(logger): + return logger.isEnabledFor(LEVEL_DEBUG) + +TEST_MODE_NONE = 0 +TEST_MODE_DETERMINISTIC = 1 +TEST_MODE_NON_DETERMINISTIC = 2 global agTestMode -agTestMode = False +agTestMode = TEST_MODE_NONE def setTestMode(mode): global agTestMode - if (mode): - agTestMode = True - else: - agTestMode = False + agTestMode = mode def getTestMode(): global agTestMode return agTestMode -def testMode(normalObj, testObj=None): - if getTestMode(): +def testMode(normalObj, testObj=None, nonDeterministicObj=None): + testMode = getTestMode() + if testMode > TEST_MODE_NONE: + if ((nonDeterministicObj != None) and (testMode == TEST_MODE_NON_DETERMINISTIC)): + return nonDeterministicObj return testObj return normalObj -def toDiffableString(value): - s = repr(value) - ds = "" - i = s.find(" at 0x") - start = 0 - while (i >= 0): - j = s.find(">", i) - if (j < i): - break - ds += s[start:i] - start = j - i = s.find(" at 0x", start) - return ds + s[start:] - +pythonFileRefPattern = asString(r'(?<=File ")[^"]*(#[^#]*")(, line )[0-9]*') +phpFileRefPattern = asString(r'( in ).*#([^#]*#[^ ]*)(?= on line )') +pathSepPattern = os.sep +if (pathSepPattern == "\\"): + pathSepPattern = "\\\\" +pythonFileRefPattern = pythonFileRefPattern.replace("#", pathSepPattern) +pythonFileRefPattern = re.compile(pythonFileRefPattern) +phpFileRefPattern = phpFileRefPattern.replace("#", pathSepPattern) +phpFileRefPattern = re.compile(phpFileRefPattern) + def removeFileRefs(str): - str = re.sub(r'(?<=File ")[^"]*(\\[^\\]*")(, line )[0-9]*', _fileNameReplacement, str) + str = pythonFileRefPattern.sub(_fileNameReplacement, str) + str = phpFileRefPattern.sub(_fileNameReplacementPHP, str) + return str + +def removePHPFileRefs(str): + str = phpFileRefPattern.sub(_fileNameReplacementPHP, str) return str def _fileNameReplacement(match): - return "...%s" % match.group(1) - -def getTraceback(): - extype, val, tb = sys.exc_info() - tbs = "\n" - for s in traceback.format_tb(tb): - tbs += s + return "...%s" % match.group(1).replace(os.sep, "/") + +def _fileNameReplacementPHP(match): + return "%s...%s" % (match.group(1), match.group(2).replace(os.sep, "/")) + +def formatTraceback(tb=None): + if (tb == None): + extype, val, tb = sys.exc_info() + tbs = "\n" + "".join(traceback.format_tb(tb)) return tbs -def reportException(out=None, stacktrace=False, diffable=False): - extype, val, t = sys.exc_info() +def formatExceptionCause(cause, stacktrace=False): + if (cause == None): + return "" + tbs = "" + if (stacktrace): + tbs = formatTraceback() + return "Caused by %s.%s: %s%s" % (cause.__module__, cause.__class__.__name__, str(cause), tbs) + +def addExceptionInfo(e, key, value): + if not hasattr(e, EXCEPTION_INFO): + try: + setattr(e, EXCEPTION_INFO, {}) + except: + return # Make sure we still report the real exception even if we can't add the extra info + if not e.exceptionInfo.has_key(key): # Never overwrite exception info since we assume earlier info is more specific + e.exceptionInfo[key] = value + +def reportException(exception, out=None, stacktrace=False, diffable=False): + exstr = exceptionToString(exception, stacktrace, diffable) + if (out == None): + print exstr + else: + print >> out, exstr + +def exceptionToString(exception, stacktrace=False, diffable=False): + extype = objutils.typeToString(exception) + val = exception + if (stacktrace): + e,v,t = sys.exc_info() if (diffable): exstr = removeFileRefs(str(val)) else: exstr = str(val) - if (out == None): - print "Got Exception = %s: %s" % (extype, exstr) - else: - print >> out, "Got Exception = %s: %s" % (extype, exstr) + if hasattr(val, EXCEPTION_INFO): + firstTime = True + for infoKey, infoValue in getattr(val, EXCEPTION_INFO).items(): + if firstTime: + prefix = " EXTRA INFO:" + firstTime = False + else: + prefix = "," + exstr += ("%s %s=%s" % (prefix, infoKey, infoValue)) + result = "Got Exception = %s: %s" % (extype, exstr) if (stacktrace): fmt = traceback.format_exception(extype, val, t) for s in fmt: if (diffable): s = removeFileRefs(s) - if (out == None): - print s + result = result + "\n" + s + return result + +def fileConfig(fname, defaults=None): + """ + This is copied from logging.config so that we could fix the class lookup of + handlers. Previously handlers had to be defined in logging.handlers and we + need to be able to define our own. + """ + import ConfigParser, string + + cp = ConfigParser.ConfigParser(defaults) + if hasattr(cp, 'readfp') and hasattr(fname, 'readline'): + cp.readfp(fname) + else: + cp.read(fname) + #first, do the formatters... + flist = cp.get("formatters", "keys") + if len(flist): + flist = string.split(flist, ",") + formatters = {} + for form in flist: + sectname = "formatter_%s" % form + opts = cp.options(sectname) + if "format" in opts: + fs = cp.get(sectname, "format", 1) else: - print >> out, s - + fs = None + if "datefmt" in opts: + dfs = cp.get(sectname, "datefmt", 1) + else: + dfs = None + f = logging.Formatter(fs, dfs) + formatters[form] = f + #next, do the handlers... + #critical section... + logging._acquireLock() + try: +## try: + #first, lose the existing handlers... + logging._handlers.clear() + #now set up the new ones... + hlist = cp.get("handlers", "keys") + if len(hlist): + hlist = string.split(hlist, ",") + handlers = {} + fixups = [] #for inter-handler references + for hand in hlist: +## try: + sectname = "handler_%s" % hand + classname = cp.get(sectname, "class") + opts = cp.options(sectname) + if "formatter" in opts: + fmt = cp.get(sectname, "formatter") + else: + fmt = "" + klass = None + try: + klass = eval(classname, vars(logging)) + except: + pass + if (klass == None): + klass = objutils.classForName(classname) + args = cp.get(sectname, "args") + args = eval(args, vars(logging)) + h = apply(klass, args) + if "level" in opts: + level = cp.get(sectname, "level") + h.setLevel(logging._levelNames[level]) + if len(fmt): + h.setFormatter(formatters[fmt]) + #temporary hack for FileHandler and MemoryHandler. + if klass == logging.handlers.MemoryHandler: + if "target" in opts: + target = cp.get(sectname,"target") + else: + target = "" + if len(target): #the target handler may not be loaded yet, so keep for later... + fixups.append((h, target)) + handlers[hand] = h +## except Exception, e: #if an error occurs when instantiating a handler, too bad +## pass #this could happen e.g. because of lack of privileges + #now all handlers are loaded, fixup inter-handler references... + for fixup in fixups: + h = fixup[0] + t = fixup[1] + h.setTarget(handlers[t]) + #at last, the loggers...first the root... + llist = cp.get("loggers", "keys") + llist = string.split(llist, ",") + llist.remove("root") + sectname = "logger_root" + root = logging.root + log = root + opts = cp.options(sectname) + if "level" in opts: + level = cp.get(sectname, "level") + log.setLevel(logging._levelNames[level]) + for h in root.handlers[:]: + root.removeHandler(h) + hlist = cp.get(sectname, "handlers") + if len(hlist): + hlist = string.split(hlist, ",") + for hand in hlist: + log.addHandler(handlers[hand]) + #and now the others... + #we don't want to lose the existing loggers, + #since other threads may have pointers to them. + #existing is set to contain all existing loggers, + #and as we go through the new configuration we + #remove any which are configured. At the end, + #what's left in existing is the set of loggers + #which were in the previous configuration but + #which are not in the new configuration. + existing = root.manager.loggerDict.keys() + #now set up the new ones... + for log in llist: + sectname = "logger_%s" % log + qn = cp.get(sectname, "qualname") + opts = cp.options(sectname) + if "propagate" in opts: + propagate = cp.getint(sectname, "propagate") + else: + propagate = 1 + logger = logging.getLogger(qn) + if qn in existing: + existing.remove(qn) + if "level" in opts: + level = cp.get(sectname, "level") + logger.setLevel(logging._levelNames[level]) + for h in logger.handlers[:]: + logger.removeHandler(h) + logger.propagate = propagate + logger.disabled = 0 + hlist = cp.get(sectname, "handlers") + if len(hlist): + hlist = string.split(hlist, ",") + for hand in hlist: + logger.addHandler(handlers[hand]) + #Disable any old loggers. There's no point deleting + #them as other threads may continue to hold references + #and by disabling them, you stop them doing any logging. + for log in existing: + root.manager.loggerDict[log].disabled = 1 +## except: +## import traceback +## ei = sys.exc_info() +## traceback.print_exception(ei[0], ei[1], ei[2], None, sys.stderr) +## del ei + finally: + logging._releaseLock()