Python / Django: войти в консоль на сервере запуска, войти в файл в Apache


114

Как я могу отправлять сообщения трассировки на консоль (например, print), когда я запускаю свое приложение Django под manage.py runserver, но отправлять эти сообщения в файл журнала, когда я запускаю приложение под Apache?

Я изучил ведение журнала Django, и хотя меня впечатлила его гибкость и возможность настройки для расширенного использования, я все еще не понимаю, как справиться с моим простым вариантом использования.


Ответы:


84

Текст, напечатанный в stderr, будет отображаться в журнале ошибок httpd при запуске под mod_wsgi. Вы можете использовать его printнапрямую или loggingвместо него.

print >>sys.stderr, 'Goodbye, cruel world!'

2
Однако технически это недопустимый WSGI и вызовет ошибки в более строгих средах.
Пол Макмиллан,

13
С точки зрения WSGI, нет ничего плохого в использовании print с sys.stderr, и это не должно вызывать ошибок.
Грэм Дамплтон

Я импортировал sys, но, похоже, у меня это не работает.
Hack-R

17
Это не работает в Python 3, посмотрите здесь . You needprint("Goodbye cruel world!", file=sys.stderr)
кардамон

103

Вот решение на основе журналов Django. Он использует настройку DEBUG, а не фактически проверяет, запущен ли вы сервер разработки, но если вы найдете лучший способ проверить это, его будет легко адаптировать.

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/ .


8
также попробуйтеLOGGING['loggers'][logger]['handlers'] += ['console']
Нир Леви

@ m01: После настройки этого в settings.py, как использовать это для печати? Спасибо
Niks Jain

Я помещаю код из своего ответа в свой settings.pyвнизу и устанавливаю DEBUG = True(ищите этот параметр вверху в том же файле). Затем я запускаю python manage.py runserverс терминала (подробности см. В документации django), и сообщения журнала появятся в окне терминала. В производстве я бы использовал другой файл settings.py, где DEBUG = False- сообщения журнала /path/to/your/file.log.
m01

От вашего вдавливания у меня заболела голова. Спасибо за информацию, это работает!
ioan

Спасибо! Я внес некоторые изменения в отступ, надеюсь, теперь он лучше
m01

27

Вы можете настроить вход в свой settings.py файле.

Один пример:

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'
    )

Однако это зависит от настройки DEBUG, и, возможно, вам не нужно беспокоиться о том, как он настроен. См. Этот ответ в разделе Как узнать, работает ли мое приложение Django на сервере разработки или нет? для лучшего способа написания этого условного. Изменить: приведенный выше пример взят из проекта Django 1.1, конфигурация ведения журнала в Django несколько изменилась с той версии.


Я не хочу полагаться на DEBUG; Я бы предпочел полагаться на механизм обнаружения сервера-разработчика, связанный с этим другим постом. Но механизм обнаружения другого поста зависит от наличия доступа к экземпляру запроса. Как получить экземпляр запроса в settings.py?
Джастин Грант,

4

Я использую это:

logging.conf:

[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,)

testapp.py:

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()

0

Вы можете сделать это довольно легко с помощью tagalog (https://github.com/dorkitude/tagalog)

Например, в то время как стандартный модуль python записывает в объект файла, открытый в режиме добавления, модуль App Engine (https://github.com/dorkitude/tagalog/blob/master/tagalog_appengine.py) отменяет это поведение и вместо этого использует logging.INFO .

Чтобы добиться такого поведения в проекте App Engine, можно просто сделать:

import tagalog.tagalog_appengine as tagalog
tagalog.log('whatever message', ['whatever','tags'])

Вы можете самостоятельно расширить модуль и без особого труда перезаписать функцию журнала.


0

Это довольно хорошо работает в моем local.py, избавляя меня от испорченного обычного ведения журнала:

from .settings import *

LOGGING['handlers']['console'] = {
    'level': 'DEBUG',
    'class': 'logging.StreamHandler',
    'formatter': 'verbose'
}
LOGGING['loggers']['foo.bar'] = {
    'handlers': ['console'],
    'propagate': False,
    'level': 'DEBUG',
}
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.