Django rest框架在不同的文件上记录不同的级别

Tha*_*nos 9 python django logging error-logging

我正在研究Django REST 框架,我希望有单独的文件来记录数据。

我想要一个用于简单事务的文件,例如 GET、PUT、POST 等,以及一个包含错误的文件,我将在发生错误时收集这些错误。

我一直在阅读Django Logging Documentation并且我想出了一些关于如何记录信息数据的配置。配置示例如下:

设置.py

STATIC_URL = '/static/'
PROJECT_ROOT = os.path.dirname(os.path.abspath(__file__))
STATIC_ROOT = os.path.join(PROJECT_ROOT, 'static')
LOGGING_ROOT = os.path.join(STATIC_ROOT, 'logging')

LOGGING = {
    'version': 1,
    'disable_existing_loggers': False,
    'handlers': {
        'file': {
            'level': 'INFO',
            'class': 'logging.FileHandler',
            'filename': LOGGING_ROOT + "/info.log",
        },
    },
    'loggers': {
        'django': {
            'handlers': ['file'],
            'level': 'INFO',
            'propagate': True,
        },
    },
}
Run Code Online (Sandbox Code Playgroud)

它作为文件中的预期数据样本工作:

"PUT /upload/dat.txt HTTP/1.1" 204 14
"OPTIONS / HTTP/1.1" 200 10020
"GET / HTTP/1.1" 200 9916
Run Code Online (Sandbox Code Playgroud)

我尝试在 settings.py 文件中应用另一个处理程序,例如:

LOGGING = {
    'version': 1,
    'disable_existing_loggers': False,
    'handlers': {
        'fileInfo': {
            'level': 'INFO',
            'class': 'logging.FileHandler',
            'filename': LOGGING_ROOT + "/info.log",
        },
        'fileDebug': {
            'level': 'DEBUG',
            'class': 'logging.FileHandler',
            'filename': LOGGING_ROOT + "/debbug.log",
        },
    },
    'loggers': {
        'django': {
            'handlers': ['fileInfo'],
            'level': 'INFO',
            'propagate': True,
        },
        'django': {
            'handlers': ['fileDebug'],
            'level': 'DEBUG',
            'propagate': True,
        },
    },
}
Run Code Online (Sandbox Code Playgroud)

我没有按预期工作,现在我在同一个文件 debug.log 上获取所有数据(INFO 和 DEBUG)。所以我决定通过日志库采用另一种方法。我无法弄清楚的是如何将这些日志错误的输出通过管道传输到目录中,例如(path/errors.log)。

从文档:

# import the logging library
import logging

# Get an instance of a logger
logger = logging.getLogger(__name__)

def my_view(request, arg1, arg):
    ...
    if bad_mojo:
        # Log an error message
        logger.error('Something went wrong!')
Run Code Online (Sandbox Code Playgroud)

我发现了这个类似的问题No handlers can be found for logger如果我应用以下内容,它就可以工作:

import logging
logging.basicConfig()
logger = logging.getLogger(__name__)
logger.error('Something went wrong!')
Run Code Online (Sandbox Code Playgroud)

现在一切都记录在标准输出(信息和错误)上。有没有办法组合这些配置?

我想将所有请求记录到 info.log 文件中,并在我选择文件时将错误记录到 error.log 文件中。

更新:

bruno desthuilliers提供的解决方案示例如下:

设置.py

LOGGING = {
    'version': 1,
    'disable_existing_loggers': False,
    'handlers': {
        'fileInfo': {
            'level': 'INFO',
            'class': 'logging.FileHandler',
            'filename': os.path.join(LOGGING_ROOT, "info.log"),
        },
        'fileDebug': {
            'level': 'DEBUG',
            'class': 'logging.FileHandler',
            'filename': os.path.join(LOGGING_ROOT, "debug.log")
        },
        'console': {
            'class': 'logging.StreamHandler',
        },
    },
    'loggers': {
        'django': {
            'handlers': ['console', 'fileInfo', 'fileDebug'],
            'level': 'DEBUG',
            'propagate': True,
        },
    },
}
Run Code Online (Sandbox Code Playgroud)

我还发现Make Python loggers 除了 log 之外,还可以将所有消息输出到 stdout。我当前修改的示例:

import logging
from thanosTest import settings
logging.basicConfig(filename=os.path.join(settings.LOGGING_ROOT, "error.log"))
stderrLogger = logging.StreamHandler()
stderrLogger.setFormatter(logging.Formatter(logging.BASIC_FORMAT))
logging.getLogger().addHandler(stderrLogger)
logging.error('Something went wrong!')
Run Code Online (Sandbox Code Playgroud)

现在一切都记录在 error.log 上。我想我需要应用一些过滤或其他东西。我会想办法解决的。

bru*_*ers 8

这里:

'loggers': {
    'django': {
        'handlers': ['fileInfo'],
        'level': 'INFO',
        'propagate': True,
    },
    'django': {
        'handlers': ['fileDebug'],
        'level': 'DEBUG',
        'propagate': True,
    },
Run Code Online (Sandbox Code Playgroud)

您定义了 'django' 键两次,所以第二个会覆盖第一个。

作为一般规则,如果您想要给定记录器的特定设置,请将其配置为不同的记录器(即每个包或 django 应用程序一个记录器)。另请注意,记录器的“处理程序”是列表,因此每个记录器可以有多个处理程序(即一个用于调试,一个用于信息)。这个logging库有点复杂,但花时间阅读完整的文档并进行一些试验是值得的,真的。

其他一些注意事项:

  1. 为多进程应用程序使用文件处理程序(在生产 django 中最常由多进程前端服务器提供服务)通常是一个坏主意(对文件的并发写入访问永远不会真正起作用)。

  2. 'filename': LOGGING_ROOT + "/info.log"种失败的使用整点os.path-你想os.path.join(LOGGING_ROOT, "info.log"),而不是