Files
mongo/buildscripts/resmokelib/logging/loggers.py
Max Hirschhorn 611220640b SERVER-31556 Use fixture logger as parent of hook logger.
This enables the hook logger to log messages to logkeeper when
--log=buildlogger is specified without needing to start a new dynamic
test case.
2017-11-26 12:14:00 -05:00

345 lines
14 KiB
Python

"""
Module to hold the logger instances themselves.
"""
from __future__ import absolute_import
import logging
import sys
from . import buildlogger
from . import formatters
_DEFAULT_FORMAT = "[%(name)s] %(message)s"
EXECUTOR_LOGGER_NAME = "executor"
FIXTURE_LOGGER_NAME = "fixture"
TESTS_LOGGER_NAME = "tests"
EXECUTOR_LOGGER = None
def _build_logger_server(logging_config):
"""Create and return a new BuildloggerServer if "buildlogger" is configured as
one of the handler class in the configuration, return None otherwise.
"""
for logger_name in (FIXTURE_LOGGER_NAME, TESTS_LOGGER_NAME):
logger_info = logging_config[logger_name]
for handler_info in logger_info["handlers"]:
if handler_info["class"] == "buildlogger":
return buildlogger.BuildloggerServer()
def configure_loggers(logging_config):
buildlogger.BUILDLOGGER_FALLBACK = BaseLogger("buildlogger")
# The 'buildlogger' prefix is not added to the fallback logger since the prefix of the original
# logger will be there as part of the logged message.
buildlogger.BUILDLOGGER_FALLBACK.addHandler(
_fallback_buildlogger_handler(include_logger_name=False))
build_logger_server = _build_logger_server(logging_config)
fixture_logger = FixtureRootLogger(logging_config, build_logger_server)
tests_logger = TestsRootLogger(logging_config, build_logger_server)
global EXECUTOR_LOGGER
EXECUTOR_LOGGER = ExecutorRootLogger(logging_config, build_logger_server,
fixture_logger, tests_logger)
class BaseLogger(logging.Logger):
"""Base class for the custom loggers used in this library.
Custom loggers share access to the logging configuration and provide methods
to create other loggers.
"""
def __init__(self, name, logging_config=None, build_logger_server=None, parent=None):
"""Initialize a BaseLogger.
:param name: the logger name.
:param logging_config: the logging configuration.
:param build_logger_server: the build logger server (e.g. logkeeper).
:param parent: the parent logger.
"""
logging.Logger.__init__(self, name, level=logging.DEBUG)
self._logging_config = logging_config
self._build_logger_server = build_logger_server
if parent:
self.parent = parent
self.propagate = True
@property
def build_logger_server(self):
"""The configured BuildloggerServer instance, or None."""
if self._build_logger_server:
return self._build_logger_server
elif self.parent:
# Fetching the value from parent
return getattr(self.parent, "build_logger_server", None)
return None
@property
def logging_config(self):
"""The logging configuration."""
if self._logging_config:
return self._logging_config
elif self.parent:
# Fetching the value from parent
return getattr(self.parent, "logging_config", None)
return None
@staticmethod
def get_formatter(logger_info):
log_format = logger_info.get("format", _DEFAULT_FORMAT)
return formatters.ISO8601Formatter(fmt=log_format)
class RootLogger(BaseLogger):
"""A custom class for top-level loggers (executor, fixture, tests)."""
def __init__(self, name, logging_config, build_logger_server):
"""Initialize a RootLogger.
:param name: the logger name.
:param logging_config: the logging configuration.
:param build_logger_server: the build logger server, if one is configured.
"""
BaseLogger.__init__(self, name, logging_config, build_logger_server)
self._configure()
def _configure(self):
if self.name not in self.logging_config:
raise ValueError("Logging configuration should contain the %s component", self.name)
logger_info = self.logging_config[self.name]
formatter = self.get_formatter(logger_info)
for handler_info in logger_info.get("handlers", []):
self._add_handler(handler_info, formatter)
def _add_handler(self, handler_info, formatter):
handler_class = handler_info["class"]
if handler_class == "logging.FileHandler":
handler = logging.FileHandler(filename=handler_info["filename"],
mode=handler_info.get("mode", "w"))
elif handler_class == "logging.NullHandler":
handler = logging.NullHandler()
elif handler_class == "logging.StreamHandler":
handler = logging.StreamHandler(sys.stdout)
elif handler_class == "buildlogger":
return # Buildlogger handlers are applied when creating specific child loggers
else:
raise ValueError("Unknown handler class '%s'" % handler_class)
handler.setFormatter(formatter)
self.addHandler(handler)
class ExecutorRootLogger(RootLogger):
"""Class for the "executor" top-level logger."""
def __init__(self, logging_config, build_logger_server, fixture_root_logger, tests_root_logger):
"""Initialize an ExecutorRootLogger."""
RootLogger.__init__(self, EXECUTOR_LOGGER_NAME, logging_config, build_logger_server)
self.fixture_root_logger = fixture_root_logger
self.tests_root_logger = tests_root_logger
def new_resmoke_logger(self):
"""Create a child logger of this logger with the name "resmoke"."""
return BaseLogger("resmoke", parent=self)
def new_job_logger(self, test_kind, job_num):
"""Create a new child JobLogger."""
return JobLogger(test_kind, job_num, self, self.fixture_root_logger)
def new_testqueue_logger(self, test_kind):
"""Create a new TestQueueLogger that will be a child of the "tests" root logger."""
return TestQueueLogger(test_kind, self.tests_root_logger)
def new_hook_logger(self, behavior_class, job_num, fixture_logger):
"""Create a new child hook logger."""
return BaseLogger("%s:job%d" % (behavior_class, job_num), parent=fixture_logger)
class JobLogger(BaseLogger):
def __init__(self, test_kind, job_num, parent, fixture_root_logger):
"""Initialize a JobLogger.
:param test_kind: the test kind (e.g. js_test, db_test, etc.).
:param job_num: a job number.
:param fixture_root_logger: the root logger for the fixture logs.
"""
name = "executor:%s:job%d" % (test_kind, job_num)
BaseLogger.__init__(self, name, parent=parent)
self.job_num = job_num
self.fixture_root_logger = fixture_root_logger
# create build_id if it should
if self.build_logger_server:
self.build_id = self.build_logger_server.new_build_id("job%d" % job_num)
if self.build_id:
url = self.build_logger_server.get_build_log_url(self.build_id)
parent.info("Writing output of job #%d to %s.", job_num, url)
else:
parent.info("Encountered an error configuring buildlogger for job #%d, falling"
" back to stderr.", job_num)
else:
self.build_id = None
def new_fixture_logger(self, fixture_class):
"""Create a new fixture logger that will be a child of the "fixture" root logger."""
return FixtureLogger(fixture_class, self.job_num, self.build_id, self.fixture_root_logger)
def new_test_logger(self, test_shortname, test_basename, command, parent):
"""Create a new test logger that will be a child of the given parent."""
if self.build_id:
test_id = self.build_logger_server.new_test_id(self.build_id, test_basename, command)
if test_id:
url = self.build_logger_server.get_test_log_url(self.build_id, test_id)
self.info("Writing output of %s to %s.", test_shortname, url)
return TestLogger(test_shortname, parent, self.build_id, test_id, url)
else:
self.info("Encountered an error configuring buildlogger for test %s, falling"
" back to stderr.", test_shortname)
return TestLogger(test_shortname, parent)
class TestLogger(BaseLogger):
def __init__(self, test_name, parent, build_id=None, test_id=None, url=None):
"""Initialize a TestLogger.
:param test_name: the test name.
:param parent: the parent logger.
:param build_id: the build logger build id.
:param test_id: the build logger test id.
:param url: the build logger URL endpoint for the test.
"""
name = "%s:%s" % (parent.name, test_name)
BaseLogger.__init__(self, name, parent=parent)
self.url_endpoint = url
self._add_build_logger_handler(build_id, test_id)
def _add_build_logger_handler(self, build_id, test_id):
logger_info = self.logging_config[TESTS_LOGGER_NAME]
handler_info = _get_buildlogger_handler_info(logger_info)
if handler_info is not None:
if build_id and test_id:
handler = self.build_logger_server.get_test_handler(build_id, test_id, handler_info)
handler.setFormatter(self.get_formatter(logger_info))
self.addHandler(handler)
else:
# The build_id or test_id could not be obtained from the build logger server, so
# we're falling back to stderr.
self.addHandler(_fallback_buildlogger_handler())
def new_test_thread_logger(self, test_kind, thread_id):
"""Create a new child test thread logger."""
return BaseLogger("%s:%s" % (test_kind, thread_id), parent=self)
class FixtureRootLogger(RootLogger):
"""Class for the "fixture" top-level logger."""
def __init__(self, logging_config, build_logger_server):
"""Initialize a FixtureRootLogger.
:param logging_config: the logging configuration.
:param build_logger_server: the build logger server, if one is configured.
"""
RootLogger.__init__(self, FIXTURE_LOGGER_NAME, logging_config, build_logger_server)
class FixtureLogger(BaseLogger):
def __init__(self, fixture_class, job_num, build_id, fixture_root_logger):
"""Initialize a FixtureLogger.
:param fixture_class: the name of the fixture class.
:param job_num: the number of the job the fixture is running on.
:param build_id: the build logger build id, if any.
:param fixture_root_logger: the root logger for the fixture logs.
"""
BaseLogger.__init__(self, "%s:job%d" % (fixture_class, job_num), parent=fixture_root_logger)
self.fixture_class = fixture_class
self.job_num = job_num
self._add_build_logger_handler(build_id)
def _add_build_logger_handler(self, build_id):
logger_info = self.logging_config[FIXTURE_LOGGER_NAME]
handler_info = _get_buildlogger_handler_info(logger_info)
if handler_info is not None:
if build_id:
handler = self.build_logger_server.get_global_handler(build_id, handler_info)
handler.setFormatter(self.get_formatter(logger_info))
self.addHandler(handler)
else:
# The build_id could not be obtained from the build logger server, so we're
# falling back to stderr.
self.addHandler(_fallback_buildlogger_handler())
def new_fixture_node_logger(self, node_name):
"""Create a new child FixtureNodeLogger."""
return FixtureNodeLogger(self.fixture_class, self.job_num, node_name, self)
class FixtureNodeLogger(BaseLogger):
def __init__(self, fixture_class, job_num, node_name, fixture_logger):
"""Initialize a FixtureNodeLogger.
:param fixture_class: the name of the fixture implementation class.
:param job_num: the number of the job the fixture is running on.
:param node_name: the node display name.
:param fixture_logger: the parent fixture logger.
"""
BaseLogger.__init__(self, "%s:job%d:%s" % (fixture_class, job_num, node_name),
parent=fixture_logger)
self.fixture_class = fixture_class
self.job_num = job_num
self.node_name = node_name
def new_fixture_node_logger(self, node_name):
"""Create a new child FixtureNodeLogger."""
return FixtureNodeLogger(self.fixture_class, self.job_num,
"%s:%s" % (self.node_name, node_name), self)
class TestsRootLogger(RootLogger):
"""Class for the "tests" top-level logger."""
def __init__(self, logging_config, build_logger_server):
"""Initialize a TestsRootLogger.
:param logging_config: the logging configuration.
:param build_logger_server: the build logger server, if one is configured.
"""
RootLogger.__init__(self, TESTS_LOGGER_NAME, logging_config, build_logger_server)
class TestQueueLogger(BaseLogger):
def __init__(self, test_kind, tests_root_logger):
"""Initialize a TestQueueLogger.
:param test_kind: the test kind (e.g. js_test, db_test, cpp_unit_test, etc.).
:param tests_root_logger: the root logger for the tests logs.
"""
BaseLogger.__init__(self, test_kind, parent=tests_root_logger)
# Util methods
def _fallback_buildlogger_handler(include_logger_name=True):
"""
Returns a handler that writes to stderr.
"""
if include_logger_name:
log_format = "[fallback] [%(name)s] %(message)s"
else:
log_format = "[fallback] %(message)s"
formatter = formatters.ISO8601Formatter(fmt=log_format)
handler = logging.StreamHandler(sys.stderr)
handler.setFormatter(formatter)
return handler
def _get_buildlogger_handler_info(logger_info):
"""
Returns the buildlogger handler information if it exists, and None
otherwise.
"""
for handler_info in logger_info["handlers"]:
handler_info = handler_info.copy()
if handler_info.pop("class") == "buildlogger":
return handler_info
return None