Python/Django: log to console under runserver, log to file under Apache
当我在
我回顾了Django日志记录,虽然它的灵活性和可配置性给我留下了深刻的印象,但我仍然无法理解如何处理我的简单用例。
这是一个基于django日志记录的解决方案。它使用调试设置,而不是实际检查您是否在运行开发服务器,但是如果您找到一种更好的方法来检查它,那么应该很容易适应。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 | LOGGING = { 'version': 1, 'formatters': { 'verbose': { 'format': '%(levelname)s %(asctime)s %(module)s %(process)d %(thread)d %(message)s' }, 'simple': { 'format': '%(levelname)s %(message)s' }, }, 'handlers': { 'console': { 'level': 'DEBUG', 'class': 'logging.StreamHandler', 'formatter': 'simple' }, 'file': { 'level': 'DEBUG', 'class': 'logging.FileHandler', 'filename': '/path/to/your/file.log', 'formatter': 'simple' }, }, 'loggers': { 'django': { 'handlers': ['file'], 'level': 'DEBUG', 'propagate': True, }, } } if DEBUG: # make all loggers use the console. for logger in LOGGING['loggers']: LOGGING['loggers'][logger]['handlers'] = ['console'] |
有关详细信息,请参阅https://docs.djangoproject.com/en/dev/topics/logging/。
在mod wsgi下运行时,打印到stderr的文本将显示在httpd的错误日志中。您可以直接使用
1 | print >>sys.stderr, 'Goodbye, cruel world!' |
您可以在
一个例子:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | if DEBUG: # will output to your console logging.basicConfig( level = logging.DEBUG, format = '%(asctime)s %(levelname)s %(message)s', ) else: # will output to logging file logging.basicConfig( level = logging.DEBUG, format = '%(asctime)s %(levelname)s %(message)s', filename = '/my_log_file.log', filemode = 'a' ) |
然而,这取决于设置调试,也许您不想担心它是如何设置的。请参阅此答案,了解如何判断Django应用程序是否在开发服务器上运行?以更好的方式编写条件。编辑:上面的示例来自Django1.1项目,自该版本以来,Django中的日志配置发生了一些变化。
我用这个:
记录:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 | [loggers] keys=root,applog [handlers] keys=rotateFileHandler,rotateConsoleHandler [formatters] keys=applog_format,console_format [formatter_applog_format] format=%(asctime)s-[%(levelname)-8s]:%(message)s [formatter_console_format] format=%(asctime)s-%(filename)s%(lineno)d[%(levelname)s]:%(message)s [logger_root] level=DEBUG handlers=rotateFileHandler,rotateConsoleHandler [logger_applog] level=DEBUG handlers=rotateFileHandler qualname=simple_example [handler_rotateFileHandler] class=handlers.RotatingFileHandler level=DEBUG formatter=applog_format args=('applog.log', 'a', 10000, 9) [handler_rotateConsoleHandler] class=StreamHandler level=DEBUG formatter=console_format args=(sys.stdout,) |
测试:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 | import logging import logging.config def main(): logging.config.fileConfig('logging.conf') logger = logging.getLogger('applog') logger.debug('debug message') logger.info('info message') logger.warn('warn message') logger.error('error message') logger.critical('critical message') #logging.shutdown() if __name__ == '__main__': main() |
这在我的local.py中非常有效,避免了我对常规日志的干扰:
1 2 3 4 5 6 7 8 9 10 11 12 | from .settings import * LOGGING['handlers']['console'] = { 'level': 'DEBUG', 'class': 'logging.StreamHandler', 'formatter': 'verbose' } LOGGING['loggers']['foo.bar'] = { 'handlers': ['console'], 'propagate': False, 'level': 'DEBUG', } |
使用
例如,当标准的python模块写入以append模式打开的文件对象时,app engine模块(https://github.com/dorkitude/tagalog/blob/master/tagalog-app engine.py)会覆盖此行为,而使用
要在应用程序引擎项目中获得此行为,可以简单地执行以下操作:
1 2 | import tagalog.tagalog_appengine as tagalog tagalog.log('whatever message', ['whatever','tags']) |
您可以自己扩展模块并覆盖日志函数,而不需要太大的困难。