zulip/zerver/lib/logging_util.py

275 lines
9.6 KiB
Python

# System documented in https://zulip.readthedocs.io/en/latest/subsystems/logging.html
import hashlib
import logging
import threading
import traceback
from contextlib import suppress
from datetime import datetime, timedelta, timezone
from logging import Logger
import orjson
from django.conf import settings
from django.core.cache import cache
from django.http import HttpRequest
from django.utils.timezone import now as timezone_now
from typing_extensions import override
class _RateLimitFilter:
"""This class is designed to rate-limit Django error reporting
notifications so that it won't send thousands of emails if the
database or cache is completely down. It uses a remote shared
cache (shared by all Django processes) for its default behavior
(so that the deduplication is global, not per-process), and a
local in-process cache for when it can't access the remote cache.
This is critical code because it is called every time
`logging.error` or `logging.exception` (or an exception) happens
in the codebase.
Adapted from https://djangosnippets.org/snippets/2242/.
"""
last_error = datetime.min.replace(tzinfo=timezone.utc)
# This thread-local variable is used to detect recursive
# exceptions during exception handling (primarily intended for
# when accessing the shared cache throws an exception).
handling_exception = threading.local()
should_reset_handling_exception = False
def can_use_remote_cache(self) -> tuple[bool, bool]:
if getattr(self.handling_exception, "value", False):
# If we're processing an exception that occurred
# while handling an exception, this almost
# certainly was because interacting with the
# remote cache is failing (e.g. because the cache
# is down). Fall back to tracking duplicate
# exceptions in memory without the remote shared cache.
return False, False
# Now we test if the remote cache is accessible.
#
# This code path can only be reached if we are not potentially
# handling a recursive exception, so here we set
# self.handling_exception (in case the cache access we're
# about to do triggers a `logging.error` or exception that
# might recurse into this filter class), and actually record
# that this is the main exception handler thread.
try:
self.handling_exception.value = True
cache.set("RLF_TEST_KEY", 1, 1)
return cache.get("RLF_TEST_KEY") == 1, True
except Exception:
return False, True
def filter(self, record: logging.LogRecord) -> bool:
# When the original filter() call finishes executing, it's
# going to change handling_exception.value to False. The
# local variable below tracks whether the *current*,
# potentially recursive, filter() call is allowed to touch
# that value (only the original will find this to be True
# at the end of its execution)
should_reset_handling_exception = False
try:
# Track duplicate errors
duplicate = False
rate = getattr(settings, f"{type(self).__name__.upper()}_LIMIT", 600) # seconds
if rate > 0:
(use_cache, should_reset_handling_exception) = self.can_use_remote_cache()
if use_cache:
if record.exc_info is not None:
tb = "\n".join(traceback.format_exception(*record.exc_info))
else:
tb = str(record)
key = type(self).__name__.upper() + hashlib.sha1(tb.encode()).hexdigest()
duplicate = cache.get(key) == 1
if not duplicate:
cache.set(key, 1, rate)
else:
min_date = timezone_now() - timedelta(seconds=rate)
duplicate = self.last_error >= min_date
if not duplicate:
self.last_error = timezone_now()
return not duplicate
finally:
if should_reset_handling_exception:
self.handling_exception.value = False
class ZulipLimiter(_RateLimitFilter):
pass
class EmailLimiter(_RateLimitFilter):
pass
class ReturnTrue(logging.Filter):
@override
def filter(self, record: logging.LogRecord) -> bool:
return True
class RequireReallyDeployed(logging.Filter):
@override
def filter(self, record: logging.LogRecord) -> bool:
return settings.PRODUCTION
def find_log_caller_module(record: logging.LogRecord) -> str | None:
"""Find the module name corresponding to where this record was logged.
Sadly `record.module` is just the innermost component of the full
module name, so we have to go reconstruct this ourselves.
"""
# Repeat a search similar to that in logging.Logger.findCaller.
# The logging call should still be on the stack somewhere; search until
# we find something in the same source file, and that should give the
# right module name.
f = logging.currentframe()
while True:
if f.f_code.co_filename == record.pathname:
return f.f_globals.get("__name__")
if f.f_back is None:
return None
f = f.f_back
logger_nicknames = {
"root": "", # This one is more like undoing a nickname.
"zulip.requests": "zr", # Super common.
}
def find_log_origin(record: logging.LogRecord) -> str:
logger_name = logger_nicknames.get(record.name, record.name)
if settings.LOGGING_SHOW_MODULE:
module_name = find_log_caller_module(record)
if module_name in (logger_name, record.name):
# Abbreviate a bit.
pass
else:
logger_name = "{}/{}".format(logger_name, module_name or "?")
if settings.RUNNING_INSIDE_TORNADO:
# In multi-sharded Tornado, it's often valuable to have which shard is
# responsible for the request in the logs.
from zerver.tornado.ioloop_logging import logging_data
shard = logging_data.get("port", "unknown")
logger_name = f"{logger_name}:{shard}"
return logger_name
log_level_abbrevs = {
"DEBUG": "DEBG",
"INFO": "INFO",
"WARNING": "WARN",
"ERROR": "ERR",
"CRITICAL": "CRIT",
}
def abbrev_log_levelname(levelname: str) -> str:
# It's unlikely someone will set a custom log level with a custom name,
# but it's an option, so we shouldn't crash if someone does.
return log_level_abbrevs.get(levelname, levelname[:4])
class ZulipFormatter(logging.Formatter):
# Used in the base implementation. Default uses `,`.
default_msec_format = "%s.%03d"
def __init__(self) -> None:
super().__init__(fmt=self._compute_fmt())
def _compute_fmt(self) -> str:
pieces = ["%(asctime)s", "%(zulip_level_abbrev)-4s"]
if settings.LOGGING_SHOW_PID:
pieces.append("pid:%(process)d")
pieces.extend(["[%(zulip_origin)s]", "%(message)s"])
return " ".join(pieces)
@override
def format(self, record: logging.LogRecord) -> str:
if not hasattr(record, "zulip_decorated"):
record.zulip_level_abbrev = abbrev_log_levelname(record.levelname)
record.zulip_origin = find_log_origin(record)
record.zulip_decorated = True
return super().format(record)
class ZulipWebhookFormatter(ZulipFormatter):
@override
def _compute_fmt(self) -> str:
basic = super()._compute_fmt()
multiline = [
basic,
"user: %(user)s",
"client: %(client)s",
"url: %(url)s",
"content_type: %(content_type)s",
"custom_headers:",
"%(custom_headers)s",
"payload:",
"%(payload)s",
]
return "\n".join(multiline)
@override
def format(self, record: logging.LogRecord) -> str:
request: HttpRequest | None = getattr(record, "request", None)
if request is None:
record.user = None
record.client = None
record.url = None
record.content_type = None
record.custom_headers = None
record.payload = None
return super().format(record)
if request.content_type == "application/json":
payload: str | bytes = request.body
else:
payload = request.POST["payload"]
with suppress(orjson.JSONDecodeError):
payload = orjson.dumps(orjson.loads(payload), option=orjson.OPT_INDENT_2).decode()
header_text = "".join(
f"{header}: {value}\n"
for header, value in request.headers.items()
if header.lower().startswith("x-")
)
from zerver.lib.request import RequestNotes
client = RequestNotes.get_notes(request).client
assert client is not None
assert request.user.is_authenticated
record.user = f"{request.user.delivery_email} ({request.user.realm.string_id})"
record.client = client.name
record.url = request.META.get("PATH_INFO", None)
record.content_type = request.content_type
record.custom_headers = header_text or None
record.payload = payload
return super().format(record)
def log_to_file(
logger: Logger,
filename: str,
log_format: str = "%(asctime)s %(levelname)-8s %(message)s",
) -> None:
"""Note: `filename` should be declared in zproject/computed_settings.py with zulip_path."""
formatter = logging.Formatter(log_format)
handler = logging.FileHandler(filename)
handler.setFormatter(formatter)
logger.addHandler(handler)