1""" fontTools.misc.loggingTools.py -- tools for interfacing with the Python 2logging package. 3""" 4 5from __future__ import print_function, absolute_import 6from fontTools.misc.py23 import * 7import sys 8import logging 9import timeit 10from functools import wraps 11try: 12 from collections.abc import Mapping, Callable 13except ImportError: # python < 3.3 14 from collections import Mapping, Callable 15import warnings 16 17try: 18 from logging import PercentStyle 19except ImportError: 20 PercentStyle = None 21 22 23# default logging level used by Timer class 24TIME_LEVEL = logging.DEBUG 25 26# per-level format strings used by the default formatter 27# (the level name is not printed for INFO and DEBUG messages) 28DEFAULT_FORMATS = { 29 "*": "%(levelname)s: %(message)s", 30 "INFO": "%(message)s", 31 "DEBUG": "%(message)s", 32 } 33 34 35class LevelFormatter(logging.Formatter): 36 """ Formatter class which optionally takes a dict of logging levels to 37 format strings, allowing to customise the log records appearance for 38 specific levels. 39 The '*' key identifies the default format string. 40 41 >>> import sys 42 >>> handler = logging.StreamHandler(sys.stdout) 43 >>> formatter = LevelFormatter( 44 ... fmt={ 45 ... '*': '[%(levelname)s] %(message)s', 46 ... 'DEBUG': '%(name)s [%(levelname)s] %(message)s', 47 ... 'INFO': '%(message)s', 48 ... }) 49 >>> handler.setFormatter(formatter) 50 >>> log = logging.getLogger('test') 51 >>> log.setLevel(logging.DEBUG) 52 >>> log.addHandler(handler) 53 >>> log.debug('this uses a custom format string') 54 test [DEBUG] this uses a custom format string 55 >>> log.info('this also uses a custom format string') 56 this also uses a custom format string 57 >>> log.warning("this one uses the default format string") 58 [WARNING] this one uses the default format string 59 """ 60 61 def __init__(self, fmt=None, datefmt=None, style="%"): 62 if style != '%': 63 raise ValueError( 64 "only '%' percent style is supported in both python 2 and 3") 65 if fmt is None: 66 fmt = DEFAULT_FORMATS 67 if isinstance(fmt, basestring): 68 default_format = fmt 69 custom_formats = {} 70 elif isinstance(fmt, Mapping): 71 custom_formats = dict(fmt) 72 default_format = custom_formats.pop("*", None) 73 else: 74 raise TypeError('fmt must be a str or a dict of str: %r' % fmt) 75 super(LevelFormatter, self).__init__(default_format, datefmt) 76 self.default_format = self._fmt 77 self.custom_formats = {} 78 for level, fmt in custom_formats.items(): 79 level = logging._checkLevel(level) 80 self.custom_formats[level] = fmt 81 82 def format(self, record): 83 if self.custom_formats: 84 fmt = self.custom_formats.get(record.levelno, self.default_format) 85 if self._fmt != fmt: 86 self._fmt = fmt 87 # for python >= 3.2, _style needs to be set if _fmt changes 88 if PercentStyle: 89 self._style = PercentStyle(fmt) 90 return super(LevelFormatter, self).format(record) 91 92 93def configLogger(**kwargs): 94 """ Do basic configuration for the logging system. This is more or less 95 the same as logging.basicConfig with some additional options and defaults. 96 97 The default behaviour is to create a StreamHandler which writes to 98 sys.stderr, set a formatter using the DEFAULT_FORMATS strings, and add 99 the handler to the top-level library logger ("fontTools"). 100 101 A number of optional keyword arguments may be specified, which can alter 102 the default behaviour. 103 104 logger Specifies the logger name or a Logger instance to be configured. 105 (it defaults to "fontTools" logger). Unlike basicConfig, this 106 function can be called multiple times to reconfigure a logger. 107 If the logger or any of its children already exists before the 108 call is made, they will be reset before the new configuration 109 is applied. 110 filename Specifies that a FileHandler be created, using the specified 111 filename, rather than a StreamHandler. 112 filemode Specifies the mode to open the file, if filename is specified 113 (if filemode is unspecified, it defaults to 'a'). 114 format Use the specified format string for the handler. This argument 115 also accepts a dictionary of format strings keyed by level name, 116 to allow customising the records appearance for specific levels. 117 The special '*' key is for 'any other' level. 118 datefmt Use the specified date/time format. 119 level Set the logger level to the specified level. 120 stream Use the specified stream to initialize the StreamHandler. Note 121 that this argument is incompatible with 'filename' - if both 122 are present, 'stream' is ignored. 123 handlers If specified, this should be an iterable of already created 124 handlers, which will be added to the logger. Any handler 125 in the list which does not have a formatter assigned will be 126 assigned the formatter created in this function. 127 filters If specified, this should be an iterable of already created 128 filters, which will be added to the handler(s), if the latter 129 do(es) not already have filters assigned. 130 propagate All loggers have a "propagate" attribute initially set to True, 131 which determines whether to continue searching for handlers up 132 the logging hierarchy. By default, this arguments sets the 133 "propagate" attribute to False. 134 """ 135 # using kwargs to enforce keyword-only arguments in py2. 136 handlers = kwargs.pop("handlers", None) 137 if handlers is None: 138 if "stream" in kwargs and "filename" in kwargs: 139 raise ValueError("'stream' and 'filename' should not be " 140 "specified together") 141 else: 142 if "stream" in kwargs or "filename" in kwargs: 143 raise ValueError("'stream' or 'filename' should not be " 144 "specified together with 'handlers'") 145 if handlers is None: 146 filename = kwargs.pop("filename", None) 147 mode = kwargs.pop("filemode", 'a') 148 if filename: 149 h = logging.FileHandler(filename, mode) 150 else: 151 stream = kwargs.pop("stream", None) 152 h = logging.StreamHandler(stream) 153 handlers = [h] 154 # By default, the top-level library logger is configured. 155 logger = kwargs.pop("logger", "fontTools") 156 if not logger or isinstance(logger, basestring): 157 # empty "" or None means the 'root' logger 158 logger = logging.getLogger(logger) 159 # before (re)configuring, reset named logger and its children (if exist) 160 _resetExistingLoggers(parent=logger.name) 161 # use DEFAULT_FORMATS if 'format' is None 162 fs = kwargs.pop("format", None) 163 dfs = kwargs.pop("datefmt", None) 164 # XXX: '%' is the only format style supported on both py2 and 3 165 style = kwargs.pop("style", '%') 166 fmt = LevelFormatter(fs, dfs, style) 167 filters = kwargs.pop("filters", []) 168 for h in handlers: 169 if h.formatter is None: 170 h.setFormatter(fmt) 171 if not h.filters: 172 for f in filters: 173 h.addFilter(f) 174 logger.addHandler(h) 175 if logger.name != "root": 176 # stop searching up the hierarchy for handlers 177 logger.propagate = kwargs.pop("propagate", False) 178 # set a custom severity level 179 level = kwargs.pop("level", None) 180 if level is not None: 181 logger.setLevel(level) 182 if kwargs: 183 keys = ', '.join(kwargs.keys()) 184 raise ValueError('Unrecognised argument(s): %s' % keys) 185 186 187def _resetExistingLoggers(parent="root"): 188 """ Reset the logger named 'parent' and all its children to their initial 189 state, if they already exist in the current configuration. 190 """ 191 root = logging.root 192 # get sorted list of all existing loggers 193 existing = sorted(root.manager.loggerDict.keys()) 194 if parent == "root": 195 # all the existing loggers are children of 'root' 196 loggers_to_reset = [parent] + existing 197 elif parent not in existing: 198 # nothing to do 199 return 200 elif parent in existing: 201 loggers_to_reset = [parent] 202 # collect children, starting with the entry after parent name 203 i = existing.index(parent) + 1 204 prefixed = parent + "." 205 pflen = len(prefixed) 206 num_existing = len(existing) 207 while i < num_existing: 208 if existing[i][:pflen] == prefixed: 209 loggers_to_reset.append(existing[i]) 210 i += 1 211 for name in loggers_to_reset: 212 if name == "root": 213 root.setLevel(logging.WARNING) 214 for h in root.handlers[:]: 215 root.removeHandler(h) 216 for f in root.filters[:]: 217 root.removeFilters(f) 218 root.disabled = False 219 else: 220 logger = root.manager.loggerDict[name] 221 logger.level = logging.NOTSET 222 logger.handlers = [] 223 logger.filters = [] 224 logger.propagate = True 225 logger.disabled = False 226 227 228class Timer(object): 229 """ Keeps track of overall time and split/lap times. 230 231 >>> import time 232 >>> timer = Timer() 233 >>> time.sleep(0.01) 234 >>> print("First lap:", timer.split()) 235 First lap: ... 236 >>> time.sleep(0.02) 237 >>> print("Second lap:", timer.split()) 238 Second lap: ... 239 >>> print("Overall time:", timer.time()) 240 Overall time: ... 241 242 Can be used as a context manager inside with-statements. 243 244 >>> with Timer() as t: 245 ... time.sleep(0.01) 246 >>> print("%0.3f seconds" % t.elapsed) 247 0... seconds 248 249 If initialised with a logger, it can log the elapsed time automatically 250 upon exiting the with-statement. 251 252 >>> import logging 253 >>> log = logging.getLogger("my-fancy-timer-logger") 254 >>> configLogger(logger=log, level="DEBUG", format="%(message)s", stream=sys.stdout) 255 >>> with Timer(log, 'do something'): 256 ... time.sleep(0.01) 257 Took ... to do something 258 259 The same Timer instance, holding a reference to a logger, can be reused 260 in multiple with-statements, optionally with different messages or levels. 261 262 >>> timer = Timer(log) 263 >>> with timer(): 264 ... time.sleep(0.01) 265 elapsed time: ...s 266 >>> with timer('redo it', level=logging.INFO): 267 ... time.sleep(0.02) 268 Took ... to redo it 269 270 It can also be used as a function decorator to log the time elapsed to run 271 the decorated function. 272 273 >>> @timer() 274 ... def test1(): 275 ... time.sleep(0.01) 276 >>> @timer('run test 2', level=logging.INFO) 277 ... def test2(): 278 ... time.sleep(0.02) 279 >>> test1() 280 Took ... to run 'test1' 281 >>> test2() 282 Took ... to run test 2 283 """ 284 285 # timeit.default_timer choses the most accurate clock for each platform 286 _time = timeit.default_timer 287 default_msg = "elapsed time: %(time).3fs" 288 default_format = "Took %(time).3fs to %(msg)s" 289 290 def __init__(self, logger=None, msg=None, level=None, start=None): 291 self.reset(start) 292 if logger is None: 293 for arg in ('msg', 'level'): 294 if locals().get(arg) is not None: 295 raise ValueError( 296 "'%s' can't be specified without a 'logger'" % arg) 297 self.logger = logger 298 self.level = level if level is not None else TIME_LEVEL 299 self.msg = msg 300 301 def reset(self, start=None): 302 """ Reset timer to 'start_time' or the current time. """ 303 if start is None: 304 self.start = self._time() 305 else: 306 self.start = start 307 self.last = self.start 308 self.elapsed = 0.0 309 310 def time(self): 311 """ Return the overall time (in seconds) since the timer started. """ 312 return self._time() - self.start 313 314 def split(self): 315 """ Split and return the lap time (in seconds) in between splits. """ 316 current = self._time() 317 self.elapsed = current - self.last 318 self.last = current 319 return self.elapsed 320 321 def formatTime(self, msg, time): 322 """ Format 'time' value in 'msg' and return formatted string. 323 If 'msg' contains a '%(time)' format string, try to use that. 324 Otherwise, use the predefined 'default_format'. 325 If 'msg' is empty or None, fall back to 'default_msg'. 326 """ 327 if not msg: 328 msg = self.default_msg 329 if msg.find("%(time)") < 0: 330 msg = self.default_format % {"msg": msg, "time": time} 331 else: 332 try: 333 msg = msg % {"time": time} 334 except (KeyError, ValueError): 335 pass # skip if the format string is malformed 336 return msg 337 338 def __enter__(self): 339 """ Start a new lap """ 340 self.last = self._time() 341 self.elapsed = 0.0 342 return self 343 344 def __exit__(self, exc_type, exc_value, traceback): 345 """ End the current lap. If timer has a logger, log the time elapsed, 346 using the format string in self.msg (or the default one). 347 """ 348 time = self.split() 349 if self.logger is None or exc_type: 350 # if there's no logger attached, or if any exception occurred in 351 # the with-statement, exit without logging the time 352 return 353 message = self.formatTime(self.msg, time) 354 # Allow log handlers to see the individual parts to facilitate things 355 # like a server accumulating aggregate stats. 356 msg_parts = { 'msg': self.msg, 'time': time } 357 self.logger.log(self.level, message, msg_parts) 358 359 def __call__(self, func_or_msg=None, **kwargs): 360 """ If the first argument is a function, return a decorator which runs 361 the wrapped function inside Timer's context manager. 362 Otherwise, treat the first argument as a 'msg' string and return an updated 363 Timer instance, referencing the same logger. 364 A 'level' keyword can also be passed to override self.level. 365 """ 366 if isinstance(func_or_msg, Callable): 367 func = func_or_msg 368 # use the function name when no explicit 'msg' is provided 369 if not self.msg: 370 self.msg = "run '%s'" % func.__name__ 371 372 @wraps(func) 373 def wrapper(*args, **kwds): 374 with self: 375 return func(*args, **kwds) 376 return wrapper 377 else: 378 msg = func_or_msg or kwargs.get("msg") 379 level = kwargs.get("level", self.level) 380 return self.__class__(self.logger, msg, level) 381 382 def __float__(self): 383 return self.elapsed 384 385 def __int__(self): 386 return int(self.elapsed) 387 388 def __str__(self): 389 return "%.3f" % self.elapsed 390 391 392class ChannelsFilter(logging.Filter): 393 """ Filter out records emitted from a list of enabled channel names, 394 including their children. It works the same as the logging.Filter class, 395 but allows to specify multiple channel names. 396 397 >>> import sys 398 >>> handler = logging.StreamHandler(sys.stdout) 399 >>> handler.setFormatter(logging.Formatter("%(message)s")) 400 >>> filter = ChannelsFilter("A.B", "C.D") 401 >>> handler.addFilter(filter) 402 >>> root = logging.getLogger() 403 >>> root.addHandler(handler) 404 >>> root.setLevel(level=logging.DEBUG) 405 >>> logging.getLogger('A.B').debug('this record passes through') 406 this record passes through 407 >>> logging.getLogger('A.B.C').debug('records from children also pass') 408 records from children also pass 409 >>> logging.getLogger('C.D').debug('this one as well') 410 this one as well 411 >>> logging.getLogger('A.B.').debug('also this one') 412 also this one 413 >>> logging.getLogger('A.F').debug('but this one does not!') 414 >>> logging.getLogger('C.DE').debug('neither this one!') 415 """ 416 417 def __init__(self, *names): 418 self.names = names 419 self.num = len(names) 420 self.lenghts = {n: len(n) for n in names} 421 422 def filter(self, record): 423 if self.num == 0: 424 return True 425 for name in self.names: 426 nlen = self.lenghts[name] 427 if name == record.name: 428 return True 429 elif (record.name.find(name, 0, nlen) == 0 430 and record.name[nlen] == "."): 431 return True 432 return False 433 434 435class CapturingLogHandler(logging.Handler): 436 def __init__(self, logger, level): 437 super(CapturingLogHandler, self).__init__(level=level) 438 self.records = [] 439 if isinstance(logger, basestring): 440 self.logger = logging.getLogger(logger) 441 else: 442 self.logger = logger 443 444 def __enter__(self): 445 self.original_disabled = self.logger.disabled 446 self.original_level = self.logger.level 447 self.original_propagate = self.logger.propagate 448 449 self.logger.addHandler(self) 450 self.logger.setLevel(self.level) 451 self.logger.disabled = False 452 self.logger.propagate = False 453 454 return self 455 456 def __exit__(self, type, value, traceback): 457 self.logger.removeHandler(self) 458 self.logger.setLevel(self.original_level) 459 self.logger.disabled = self.original_disabled 460 self.logger.propagate = self.original_propagate 461 462 return self 463 464 def emit(self, record): 465 self.records.append(record) 466 467 def assertRegex(self, regexp, msg=None): 468 import re 469 pattern = re.compile(regexp) 470 for r in self.records: 471 if pattern.search(r.getMessage()): 472 return True 473 if msg is None: 474 msg = "Pattern '%s' not found in logger records" % regexp 475 assert 0, msg 476 477 478class LogMixin(object): 479 """ Mixin class that adds logging functionality to another class. 480 You can define a new class that subclasses from LogMixin as well as 481 other base classes through multiple inheritance. 482 All instances of that class will have a 'log' property that returns 483 a logging.Logger named after their respective <module>.<class>. 484 For example: 485 486 >>> class BaseClass(object): 487 ... pass 488 >>> class MyClass(LogMixin, BaseClass): 489 ... pass 490 >>> a = MyClass() 491 >>> isinstance(a.log, logging.Logger) 492 True 493 >>> print(a.log.name) 494 fontTools.misc.loggingTools.MyClass 495 >>> class AnotherClass(MyClass): 496 ... pass 497 >>> b = AnotherClass() 498 >>> isinstance(b.log, logging.Logger) 499 True 500 >>> print(b.log.name) 501 fontTools.misc.loggingTools.AnotherClass 502 """ 503 504 @property 505 def log(self): 506 if not hasattr(self, "_log"): 507 name = ".".join( 508 (self.__class__.__module__, self.__class__.__name__) 509 ) 510 self._log = logging.getLogger(name) 511 return self._log 512 513 514def deprecateArgument(name, msg, category=UserWarning): 515 """ Raise a warning about deprecated function argument 'name'. """ 516 warnings.warn( 517 "%r is deprecated; %s" % (name, msg), category=category, stacklevel=3) 518 519 520def deprecateFunction(msg, category=UserWarning): 521 """ Decorator to raise a warning when a deprecated function is called. """ 522 def decorator(func): 523 @wraps(func) 524 def wrapper(*args, **kwargs): 525 warnings.warn( 526 "%r is deprecated; %s" % (func.__name__, msg), 527 category=category, stacklevel=2) 528 return func(*args, **kwargs) 529 return wrapper 530 return decorator 531 532 533if __name__ == "__main__": 534 import doctest 535 sys.exit(doctest.testmod(optionflags=doctest.ELLIPSIS).failed) 536