Невозможно добавить имя пользователя в запись в журнале с помощью Middleware - PullRequest
0 голосов
/ 26 апреля 2020

Я пытаюсь войти (по умолчанию) username и project (что можно выбрать из request объекта). Я не хочу добавлять context к каждому журналу вручную.

Проблема в том, что я не могу Django добавить request или прямые username и project к LogRecord. Я пробовал десятки способов.

Это мой код:

middlewares.py

import threading
local = threading.local()

class LoggingRequestMiddleware:
    def __init__(self, get_response):
        self.get_response = get_response
        # One-time configuration and initialization.

    def __call__(self, request):
        # Code to be executed for each request before
        # the view (and later middleware) are called.
        setattr(local, 'request', request)
        response = self.get_response(request)

        # Code to be executed for each request/response after
        # the view is called.

        return response

settings.py

def add_username_to_log(record):
local = threading.local()
record.username = '-'
request = getattr(local,'request',None)
print(request)

return True

LOGGING = {
    'version': 1,
    'disable_existing_loggers': False,
    'formatters': {
        'verbose': {
            'format': LOGGING_VERBOSE_FORMAT,
            'style': '{',
        },
    },
    'filters': {
        'context_filter': {
            '()': 'django.utils.log.CallbackFilter',
            'callback': add_username_to_log,
        },

    },
    'handlers': {
        'console': {
            'level': DEFAULT_LOG_LEVEL,
            'class': 'logging.StreamHandler',
            'formatter': 'verbose',
            'filters': ['context_filter'],
        },
        'file_main': {
            'level': DEFAULT_LOG_LEVEL,
            'class': 'logging.handlers.RotatingFileHandler',
            'filename': os.path.join(LOG_PATH, 'main.log'),
            'maxBytes': DEFAULT_LOG_SIZE,
            'formatter': 'verbose',
            'filters': ['context_filter'],
            'backupCount': 0,
        },

    },
    'loggers': {
        '': {
            'handlers': ['file_main'],
            'level': DEFAULT_LOG_LEVEL,
            'propagate': False,
        },

    },
}

Но request объект всегда None. Знаете почему?

1 Ответ

1 голос
/ 26 апреля 2020

threading.local() возвращает новый объект каждый раз, когда вы должны читать и писать в один и тот же объект.

locals_a = threading.local()
locals_a.foo = 1
hasattr(locals_a, 'foo')  # True
locals_b = threading.local()
hasattr(locals_b, 'foo')  # False

Вам необходимо определить свой объект locals в 1 месте, которое вы можете затем импортировать везде, где вы необходимо получить доступ к запросу и читать и писать в этот объект каждый раз. В качестве базового c примера это должно работать

def add_username_to_log(record):
    from middleware import local
    request = getattr(local,'request',None)
...