mongo/buildscripts/resmokelib/logging/loggers.py

443 lines
17 KiB
Python

"""Module to hold the logger instances themselves."""
import logging
import os
import sys
import yaml
from . import buildlogger
from . import formatters
from . import handlers
from .. import config
from .. import errors
_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.
This occurs 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()
return None
def configure_loggers(logging_config):
"""Configure the loggers."""
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 # pylint: disable=global-statement
EXECUTOR_LOGGER = ExecutorRootLogger(logging_config, build_logger_server, fixture_logger,
tests_logger)
_write_evergreen_log_spec()
def get_evergreen_log_name(job_num, test_id=None):
"""Return the log name, relative to the reserved test log directory, on the Evergreen task host."""
return f"job{job_num}/" + (f"{test_id}.log" if test_id else "global.log")
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):
"""Get 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):
"""Get 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):
"""Return formatter."""
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
elif handler_class == "evergreen":
return # Evergreen 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, hook_class, fixture_logger):
"""Create a new child hook logger."""
return HookLogger(hook_class, fixture_logger, self.tests_root_logger)
class JobLogger(BaseLogger):
"""JobLogger class."""
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
if self.build_logger_server:
# If we're configured to log messages to the buildlogger server, then request a new
# build_id for this job.
self.build_id = self.build_logger_server.new_build_id("job%d" % job_num)
if not self.build_id:
buildlogger.set_log_output_incomplete()
raise errors.LoggerRuntimeConfigError(
"Encountered an error configuring buildlogger for job #{:d}: Failed to get a"
" new build_id".format(job_num))
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:
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_id, test_shortname, test_basename, command, parent): # pylint: disable=too-many-arguments
"""Create a new test logger that will be a child of the given parent."""
if self.build_id:
# If we're configured to log messages to the buildlogger server, then request a new
# test id for this test.
build_test_id = self.build_logger_server.new_test_id(self.build_id, test_basename,
command)
if not build_test_id:
buildlogger.set_log_output_incomplete()
raise errors.LoggerRuntimeConfigError(
"Encountered an error configuring buildlogger for test {}: Failed to get a new"
" test_id".format(test_basename))
url = self.build_logger_server.get_test_log_url(self.build_id, build_test_id)
self.info("Writing output of %s to %s.", test_basename, url)
return TestLogger(test_shortname, parent, self.job_num, self.build_id, build_test_id,
url)
return TestLogger(test_shortname, parent, job_num=self.job_num, test_id=test_id)
class TestLogger(BaseLogger):
"""TestLogger class."""
def __init__( # pylint: disable=too-many-arguments
self, test_name, parent, job_num=None, build_id=None, test_id=None, url=None):
"""Initialize a TestLogger.
:param test_name: the test name.
:param parent: the parent logger.
:param job_num: the number of the job the test is running on.
:param build_id: the build logger build id.
:param test_id: the test id for the logger handler.
: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)
_add_evergreen_handler(self, self.logging_config, job_num, 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:
handler = self.build_logger_server.get_test_handler(build_id, test_id, handler_info)
handler.setFormatter(self.get_formatter(logger_info))
self.addHandler(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):
"""FixtureLogger class."""
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)
_add_evergreen_handler(self, self.logging_config, job_num)
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:
handler = self.build_logger_server.get_global_handler(build_id, handler_info)
handler.setFormatter(self.get_formatter(logger_info))
self.addHandler(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):
"""FixtureNodeLogger class."""
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):
"""TestQueueLogger class."""
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)
class HookLogger(BaseLogger):
"""HookLogger class."""
def __init__(self, hook_class, fixture_logger, tests_root_logger):
"""Initialize a HookLogger.
:param hook_class: the hook's name (e.g. CheckReplDBHash, ValidateCollections, etc.).
:param fixture_logger: the logger for the fixtures logs.
:param tests_root_logger: the root logger for the tests logs.
"""
logger_name = "{}:job{:d}".format(hook_class, fixture_logger.job_num)
BaseLogger.__init__(self, logger_name, parent=fixture_logger)
self.test_case_logger = BaseLogger(logger_name, parent=tests_root_logger)
# Util methods
def _fallback_buildlogger_handler(include_logger_name=True):
"""Return 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):
"""Return 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
# Utility functions for Evergreen file system logging.
# See `https://docs.devprod.prod.corp.mongodb.com/evergreen/Project-Configuration/Task-Output-Directory#test-logs`
# for more information.
def _write_evergreen_log_spec():
"""Configure file system logging for Evergreen tasks."""
if not config.EVERGREEN_WORK_DIR:
return
fp = f"{_get_evergreen_log_dirname()}/log_spec.yaml"
os.makedirs(os.path.dirname(fp), exist_ok=True)
EXECUTOR_LOGGER.info("Writing Evergreen test log spec to %s.", fp)
log_spec = {
"version": 0,
"format": "text-timestamp",
}
with open(fp, "w") as fd:
yaml.dump(log_spec, fd)
def _add_evergreen_handler(logger, logging_config, job_num, test_id=None):
"""Add a new evergreen handler to a logger."""
logger_info = logging_config[TESTS_LOGGER_NAME]
evergreen_handler_info = None
for handler_info in logger_info["handlers"]:
if handler_info["class"] == "evergreen":
evergreen_handler_info = handler_info
break
if evergreen_handler_info:
fp = f"{_get_evergreen_log_dirname()}/{get_evergreen_log_name(job_num, test_id)}"
os.makedirs(os.path.dirname(fp), exist_ok=True)
handler = handlers.BufferedFileHandler(fp)
handler.setFormatter(
formatters.EvergreenLogFormatter(fmt=logger_info.get("format", _DEFAULT_FORMAT)))
logger.addHandler(handler)
if test_id:
EXECUTOR_LOGGER.info("Writing output of %s to %s.", test_id, fp)
else:
EXECUTOR_LOGGER.info("Writing output of job #%d to %s.", job_num, fp)
def _get_evergreen_log_dirname():
"""Return the reserved directory for test logs on the Evergreen task host."""
return f"{config.EVERGREEN_WORK_DIR}/build/TestLogs"