Module sverchok.utils.sv_logging
Expand source code
from pathlib import Path
from typing import Type, Dict, Optional
import bpy
import inspect
import traceback
import logging
import logging.handlers
from contextlib import contextmanager
import sverchok
if not sverchok.reload_event: # otherwise it leeds to infinite recursion
old_factory = logging.getLogRecordFactory()
def add_relative_path_factory(name, *args, **kwargs):
record = old_factory(name, *args, **kwargs)
if name.startswith('sverchok'):
path = Path(record.pathname)
# search root path of the add-on
for root in path.parents:
if root.parent.name == 'addons': # add-ons are not always in the folder
break
else:
root = None
if root is not None:
record.relative_path = path.relative_to(root)
else: # it can if there is several instances of sverchok (as add-on and a separate folder)
record.relative_path = path
return record
if not sverchok.reload_event: # otherwise it leeds to infinite recursion
logging.setLogRecordFactory(add_relative_path_factory)
log_format = "%(asctime)s.%(msecs)03d [%(levelname)-5s] %(name)s %(relative_path)s:%(lineno)d - %(message)s"
sv_logger = logging.getLogger('sverchok') # root logger
# set any level whatever you desire,
# it will be overridden by the add-on settings after the last one will be registered
if not sverchok.reload_event:
sv_logger.setLevel(logging.ERROR)
class ColorFormatter(logging.Formatter):
START_COLOR = '\033[{}m'
RESET_COLOR = '\033[0m'
COLORS = {
'DEBUG': '1;30', # grey
'INFO': 32, # green
'WARNING': 33, # yellow
'ERROR': 31, # red
'CRITICAL': 41, # white on red bg
}
def format(self, record):
color = self.START_COLOR.format(self.COLORS[record.levelname])
color_format = color + self._fmt + self.RESET_COLOR
formatter = logging.Formatter(color_format, datefmt=self.datefmt)
return formatter.format(record)
console_handler = logging.StreamHandler()
console_handler.setFormatter(ColorFormatter(log_format, datefmt='%H:%M:%S'))
sv_logger.addHandler(console_handler)
def add_node_error_location(record: logging.LogRecord):
# https://docs.python.org/3/howto/logging-cookbook.html#using-filters-to-impart-contextual-information
# should be called with logger.error(msg, exc_info=True)
frame_info = inspect.getinnerframes(record.exc_info[-1])[-1]
record.relative_path = Path(frame_info.filename).name
record.lineno = frame_info.lineno
if not is_enabled_for('DEBUG'): # show traceback only in DEBUG mode
record.exc_info = None
return True
node_error_logger = logging.getLogger('sverchok.node_error')
node_error_logger.addFilter(add_node_error_location)
def add_file_handler(file_path):
sv_logger.debug(f'Logging to file="{file_path}"')
handler = logging.handlers.RotatingFileHandler(file_path,
maxBytes=10 * 1024 * 1024,
backupCount=3)
handler.setFormatter(logging.Formatter(log_format, datefmt="%Y-%m-%d %H:%M:%S"))
sv_logger.addHandler(handler)
def remove_console_handler():
# Remove console output handler.
logging.debug("Log output to console is disabled. Further messages will"
" be available only in text buffer and file (if configured).")
sv_logger.removeHandler(console_handler)
# https://docs.python.org/3/howto/logging.html#configuring-logging-for-a-library
sv_logger.addHandler(logging.NullHandler())
@contextmanager
def catch_log_error():
"""Catch logging errors"""
try:
yield
except Exception as e:
frame, _, line, *_ = inspect.trace()[-1]
module = inspect.getmodule(frame)
name = module.__name__ or "<Unknown Module>"
_logger = logging.getLogger(f'{name} {line}')
_logger.error(e)
if _logger.isEnabledFor(logging.DEBUG):
traceback.print_exc()
@contextmanager
def fix_error_msg(msgs: Dict[Type[Exception], str]):
try:
yield
except Exception as e:
err_class = type(e)
if err_class in msgs:
e.args = (msgs[err_class], )
raise
class TextBufferHandler(logging.Handler):
"""
A handler class which writes logging records, appropriately formatted,
to Blender's internal text buffer.
"""
terminator = '\n'
def __init__(self, name):
"""
Initialize the handler.
"""
super().__init__()
self.buffer_name = name
if self.buffer is None:
raise RuntimeError("Can't create TextBufferHandler, "
"most likely because Blender is not fully loaded")
def emit(self, record):
"""
Emit a record.
If a formatter is specified, it is used to format the record.
The record is then written to the buffer with a trailing newline. If
exception information is present, it is formatted using
traceback.print_exception and appended to the stream. If the stream
has an 'encoding' attribute, it is used to determine how to do the
output to the stream.
"""
# wen user enables a Sverchok extension it seems disables all Blender
# collections until the extension will be registered
# for now ignore such cases
if self.buffer is None:
return
try:
msg = self.format(record)
self.buffer.write(msg)
self.buffer.write(self.terminator)
self.flush()
except Exception:
self.handleError(record)
def clear(self):
"""Clear all records"""
self.buffer.clear()
sv_logger.debug("Internal text buffer cleared")
@property
def buffer(self) -> Optional:
"""
Get internal blender text buffer for logging.
"""
try:
return bpy.data.texts.get(self.buffer_name) \
or bpy.data.texts.new(name=self.buffer_name)
except AttributeError as e:
# logging.debug("Can't initialize logging to internal buffer: get_log_buffer is called too early: {}".format(e))
return None
@classmethod
def add_to_main_logger(cls):
"""This handler can work only after Blender is fully loaded"""
addon = bpy.context.preferences.addons.get(sverchok.__name__)
prefs = addon.preferences
if prefs.log_to_buffer:
sv_logger.debug(f'Logging to Blender text editor="{prefs.log_buffer_name}"')
handler = cls(prefs.log_buffer_name)
handler.setFormatter(logging.Formatter(log_format, datefmt="%Y-%m-%d %H:%M:%S"))
sv_logger.addHandler(handler)
def __repr__(self):
level = logging.getLevelName(self.level)
name = getattr(self.buffer, 'name', '')
if name:
name += ' '
return '<%s %s(%s)>' % (self.__class__.__name__, name, level)
# Convenience functions
def get_logger():
"""Get Logger instance. Logger name is obtained from caller module name."""
frame, *_ = inspect.stack()[1]
module = inspect.getmodule(frame)
name = module.__name__
return logging.getLogger(name)
def is_enabled_for(log_level="DEBUG") -> bool:
"""This check should be used for improving performance of calling disabled loggers"""
addon = bpy.context.preferences.addons.get(sverchok.__name__)
current_level = getattr(logging, addon.preferences.log_level)
given_level = getattr(logging, log_level)
return given_level >= current_level
Functions
def add_file_handler(file_path)
-
Expand source code
def add_file_handler(file_path): sv_logger.debug(f'Logging to file="{file_path}"') handler = logging.handlers.RotatingFileHandler(file_path, maxBytes=10 * 1024 * 1024, backupCount=3) handler.setFormatter(logging.Formatter(log_format, datefmt="%Y-%m-%d %H:%M:%S")) sv_logger.addHandler(handler)
def add_node_error_location(record: logging.LogRecord)
-
Expand source code
def add_node_error_location(record: logging.LogRecord): # https://docs.python.org/3/howto/logging-cookbook.html#using-filters-to-impart-contextual-information # should be called with logger.error(msg, exc_info=True) frame_info = inspect.getinnerframes(record.exc_info[-1])[-1] record.relative_path = Path(frame_info.filename).name record.lineno = frame_info.lineno if not is_enabled_for('DEBUG'): # show traceback only in DEBUG mode record.exc_info = None return True
def add_relative_path_factory(name, *args, **kwargs)
-
Expand source code
def add_relative_path_factory(name, *args, **kwargs): record = old_factory(name, *args, **kwargs) if name.startswith('sverchok'): path = Path(record.pathname) # search root path of the add-on for root in path.parents: if root.parent.name == 'addons': # add-ons are not always in the folder break else: root = None if root is not None: record.relative_path = path.relative_to(root) else: # it can if there is several instances of sverchok (as add-on and a separate folder) record.relative_path = path return record
def catch_log_error()
-
Catch logging errors
Expand source code
@contextmanager def catch_log_error(): """Catch logging errors""" try: yield except Exception as e: frame, _, line, *_ = inspect.trace()[-1] module = inspect.getmodule(frame) name = module.__name__ or "<Unknown Module>" _logger = logging.getLogger(f'{name} {line}') _logger.error(e) if _logger.isEnabledFor(logging.DEBUG): traceback.print_exc()
def fix_error_msg(msgs: Dict[Type[Exception], str])
-
Expand source code
@contextmanager def fix_error_msg(msgs: Dict[Type[Exception], str]): try: yield except Exception as e: err_class = type(e) if err_class in msgs: e.args = (msgs[err_class], ) raise
def get_logger()
-
Get Logger instance. Logger name is obtained from caller module name.
Expand source code
def get_logger(): """Get Logger instance. Logger name is obtained from caller module name.""" frame, *_ = inspect.stack()[1] module = inspect.getmodule(frame) name = module.__name__ return logging.getLogger(name)
def is_enabled_for(log_level='DEBUG') ‑> bool
-
This check should be used for improving performance of calling disabled loggers
Expand source code
def is_enabled_for(log_level="DEBUG") -> bool: """This check should be used for improving performance of calling disabled loggers""" addon = bpy.context.preferences.addons.get(sverchok.__name__) current_level = getattr(logging, addon.preferences.log_level) given_level = getattr(logging, log_level) return given_level >= current_level
def remove_console_handler()
-
Expand source code
def remove_console_handler(): # Remove console output handler. logging.debug("Log output to console is disabled. Further messages will" " be available only in text buffer and file (if configured).") sv_logger.removeHandler(console_handler) # https://docs.python.org/3/howto/logging.html#configuring-logging-for-a-library sv_logger.addHandler(logging.NullHandler())
Classes
class ColorFormatter (fmt=None, datefmt=None, style='%', validate=True)
-
Formatter instances are used to convert a LogRecord to text.
Formatters need to know how a LogRecord is constructed. They are responsible for converting a LogRecord to (usually) a string which can be interpreted by either a human or an external system. The base Formatter allows a formatting string to be specified. If none is supplied, the style-dependent default value, "%(message)s", "{message}", or "${message}", is used.
The Formatter can be initialized with a format string which makes use of knowledge of the LogRecord attributes - e.g. the default value mentioned above makes use of the fact that the user's message and arguments are pre- formatted into a LogRecord's message attribute. Currently, the useful attributes in a LogRecord are described by:
%(name)s Name of the logger (logging channel) %(levelno)s Numeric logging level for the message (DEBUG, INFO, WARNING, ERROR, CRITICAL) %(levelname)s Text logging level for the message ("DEBUG", "INFO", "WARNING", "ERROR", "CRITICAL") %(pathname)s Full pathname of the source file where the logging call was issued (if available) %(filename)s Filename portion of pathname %(module)s Module (name portion of filename) %(lineno)d Source line number where the logging call was issued (if available) %(funcName)s Function name %(created)f Time when the LogRecord was created (time.time() return value) %(asctime)s Textual time when the LogRecord was created %(msecs)d Millisecond portion of the creation time %(relativeCreated)d Time in milliseconds when the LogRecord was created, relative to the time the logging module was loaded (typically at application startup time) %(thread)d Thread ID (if available) %(threadName)s Thread name (if available) %(process)d Process ID (if available) %(message)s The result of record.getMessage(), computed just as the record is emitted
Initialize the formatter with specified format strings.
Initialize the formatter either with the specified format string, or a default as described above. Allow for specialized date formatting with the optional datefmt argument. If datefmt is omitted, you get an ISO8601-like (or RFC 3339-like) format.
Use a style parameter of '%', '{' or '$' to specify that you want to use one of %-formatting, :meth:
str.format
({}
) formatting or :class:string.Template
formatting in your format string.Changed in version: 3.2
Added the
style
parameter.Expand source code
class ColorFormatter(logging.Formatter): START_COLOR = '\033[{}m' RESET_COLOR = '\033[0m' COLORS = { 'DEBUG': '1;30', # grey 'INFO': 32, # green 'WARNING': 33, # yellow 'ERROR': 31, # red 'CRITICAL': 41, # white on red bg } def format(self, record): color = self.START_COLOR.format(self.COLORS[record.levelname]) color_format = color + self._fmt + self.RESET_COLOR formatter = logging.Formatter(color_format, datefmt=self.datefmt) return formatter.format(record)
Ancestors
- logging.Formatter
Class variables
var COLORS
var RESET_COLOR
var START_COLOR
Methods
def format(self, record)
-
Format the specified record as text.
The record's attribute dictionary is used as the operand to a string formatting operation which yields the returned string. Before formatting the dictionary, a couple of preparatory steps are carried out. The message attribute of the record is computed using LogRecord.getMessage(). If the formatting string uses the time (as determined by a call to usesTime(), formatTime() is called to format the event time. If there is exception information, it is formatted using formatException() and appended to the message.
Expand source code
def format(self, record): color = self.START_COLOR.format(self.COLORS[record.levelname]) color_format = color + self._fmt + self.RESET_COLOR formatter = logging.Formatter(color_format, datefmt=self.datefmt) return formatter.format(record)
class TextBufferHandler (name)
-
A handler class which writes logging records, appropriately formatted, to Blender's internal text buffer.
Initialize the handler.
Expand source code
class TextBufferHandler(logging.Handler): """ A handler class which writes logging records, appropriately formatted, to Blender's internal text buffer. """ terminator = '\n' def __init__(self, name): """ Initialize the handler. """ super().__init__() self.buffer_name = name if self.buffer is None: raise RuntimeError("Can't create TextBufferHandler, " "most likely because Blender is not fully loaded") def emit(self, record): """ Emit a record. If a formatter is specified, it is used to format the record. The record is then written to the buffer with a trailing newline. If exception information is present, it is formatted using traceback.print_exception and appended to the stream. If the stream has an 'encoding' attribute, it is used to determine how to do the output to the stream. """ # wen user enables a Sverchok extension it seems disables all Blender # collections until the extension will be registered # for now ignore such cases if self.buffer is None: return try: msg = self.format(record) self.buffer.write(msg) self.buffer.write(self.terminator) self.flush() except Exception: self.handleError(record) def clear(self): """Clear all records""" self.buffer.clear() sv_logger.debug("Internal text buffer cleared") @property def buffer(self) -> Optional: """ Get internal blender text buffer for logging. """ try: return bpy.data.texts.get(self.buffer_name) \ or bpy.data.texts.new(name=self.buffer_name) except AttributeError as e: # logging.debug("Can't initialize logging to internal buffer: get_log_buffer is called too early: {}".format(e)) return None @classmethod def add_to_main_logger(cls): """This handler can work only after Blender is fully loaded""" addon = bpy.context.preferences.addons.get(sverchok.__name__) prefs = addon.preferences if prefs.log_to_buffer: sv_logger.debug(f'Logging to Blender text editor="{prefs.log_buffer_name}"') handler = cls(prefs.log_buffer_name) handler.setFormatter(logging.Formatter(log_format, datefmt="%Y-%m-%d %H:%M:%S")) sv_logger.addHandler(handler) def __repr__(self): level = logging.getLevelName(self.level) name = getattr(self.buffer, 'name', '') if name: name += ' ' return '<%s %s(%s)>' % (self.__class__.__name__, name, level)
Ancestors
- logging.Handler
- logging.Filterer
Class variables
var terminator
Static methods
def add_to_main_logger()
-
This handler can work only after Blender is fully loaded
Expand source code
@classmethod def add_to_main_logger(cls): """This handler can work only after Blender is fully loaded""" addon = bpy.context.preferences.addons.get(sverchok.__name__) prefs = addon.preferences if prefs.log_to_buffer: sv_logger.debug(f'Logging to Blender text editor="{prefs.log_buffer_name}"') handler = cls(prefs.log_buffer_name) handler.setFormatter(logging.Formatter(log_format, datefmt="%Y-%m-%d %H:%M:%S")) sv_logger.addHandler(handler)
Instance variables
var buffer : Optional
-
Get internal blender text buffer for logging.
Expand source code
@property def buffer(self) -> Optional: """ Get internal blender text buffer for logging. """ try: return bpy.data.texts.get(self.buffer_name) \ or bpy.data.texts.new(name=self.buffer_name) except AttributeError as e: # logging.debug("Can't initialize logging to internal buffer: get_log_buffer is called too early: {}".format(e)) return None
Methods
def clear(self)
-
Clear all records
Expand source code
def clear(self): """Clear all records""" self.buffer.clear() sv_logger.debug("Internal text buffer cleared")
def emit(self, record)
-
Emit a record. If a formatter is specified, it is used to format the record. The record is then written to the buffer with a trailing newline. If exception information is present, it is formatted using traceback.print_exception and appended to the stream. If the stream has an 'encoding' attribute, it is used to determine how to do the output to the stream.
Expand source code
def emit(self, record): """ Emit a record. If a formatter is specified, it is used to format the record. The record is then written to the buffer with a trailing newline. If exception information is present, it is formatted using traceback.print_exception and appended to the stream. If the stream has an 'encoding' attribute, it is used to determine how to do the output to the stream. """ # wen user enables a Sverchok extension it seems disables all Blender # collections until the extension will be registered # for now ignore such cases if self.buffer is None: return try: msg = self.format(record) self.buffer.write(msg) self.buffer.write(self.terminator) self.flush() except Exception: self.handleError(record)