|
| 1 | +# -*- coding: utf-8 -*- |
| 2 | + |
| 3 | +""" |
| 4 | +Apache-like combined logging for WSGI Web Applications. |
| 5 | +
|
| 6 | +Homepage: https://github.com/pklaus/wsgi-request-logger |
| 7 | +
|
| 8 | +Copyright (c) 2013, Philipp Klaus. All rights reserved. |
| 9 | +Copyright (c) 2007-2011 L. C. Rees. All rights reserved. |
| 10 | +
|
| 11 | +License: BSD (see LICENSE for details) |
| 12 | +""" |
| 13 | + |
| 14 | +import time |
| 15 | +from datetime import datetime as dt |
| 16 | +import logging |
| 17 | + |
| 18 | +from .timehacks import Local |
| 19 | + |
| 20 | +__all__ = ['WSGILogger', 'ApacheFormatter', 'log'] |
| 21 | + |
| 22 | +class WSGILogger(object): |
| 23 | + ''' This is the generalized WSGI middleware for any style request logging. ''' |
| 24 | + |
| 25 | + def __init__(self, application, handlers, formatter=None, **kw): |
| 26 | + self.formatter = formatter or WSGILogger.standard_formatter |
| 27 | + self.logger = logging.getLogger('requestlogger') |
| 28 | + self.logger.setLevel(logging.DEBUG) |
| 29 | + for handler in handlers: |
| 30 | + self.logger.addHandler(handler) |
| 31 | + self.application = application |
| 32 | + |
| 33 | + def __call__(self, environ, start_response): |
| 34 | + start = time.clock() |
| 35 | + status_codes = [] |
| 36 | + content_lengths = [] |
| 37 | + def custom_start_response(status, response_headers, exc_info=None): |
| 38 | + status_codes.append(int(status.partition(' ')[0])) |
| 39 | + for name, value in response_headers: |
| 40 | + if name.lower() == 'content-length': |
| 41 | + content_lengths.append(int(value)) |
| 42 | + break |
| 43 | + return start_response(status, response_headers, exc_info) |
| 44 | + retval = self.application(environ, custom_start_response) |
| 45 | + runtime = int((time.clock() - start) * 10**6) |
| 46 | + content_length = content_lengths[0] if content_lengths else len(b''.join(retval)) |
| 47 | + msg = self.formatter(status_codes[0], environ, content_length, rt_ms=runtime) |
| 48 | + self.logger.info(msg) |
| 49 | + return retval |
| 50 | + |
| 51 | + @staticmethod |
| 52 | + def standard_formater(status_code, environ, content_length): |
| 53 | + return "{} {}".format(dt.now().isoformat(), status_code) |
| 54 | + |
| 55 | +def ApacheFormatter(with_response_time=True): |
| 56 | + ''' A factory that returns the wanted formatter ''' |
| 57 | + if with_response_time: |
| 58 | + return ApacheFormatters.format_with_response_time |
| 59 | + else: |
| 60 | + return ApacheFormatters.format_NCSA_log |
| 61 | + |
| 62 | +class ApacheFormatters(object): |
| 63 | + |
| 64 | + @staticmethod |
| 65 | + def format_NCSA_log(status_code, environ, content_length): |
| 66 | + """ |
| 67 | + Apache log format 'NCSA extended/combined log': |
| 68 | + "%h %l %u %t \"%r\" %>s %b \"%{Referer}i\" \"%{User-agent}i\"" |
| 69 | + see http://httpd.apache.org/docs/current/mod/mod_log_config.html#formats |
| 70 | + """ |
| 71 | + |
| 72 | + # Let's collect log values |
| 73 | + val = dict() |
| 74 | + val['host'] = environ.get('REMOTE_ADDR', '') |
| 75 | + val['logname'] = '-' |
| 76 | + val['user'] = '-' |
| 77 | + val['time'] = dt.now(tz=Local).strftime("%d/%b/%Y:%H:%M:%S %z") |
| 78 | + val['request'] = "{} {} {}".format( |
| 79 | + environ.get('REQUEST_METHOD', ''), |
| 80 | + environ.get('PATH_INFO', ''), |
| 81 | + environ.get('SERVER_PROTOCOL', '') |
| 82 | + ) |
| 83 | + val['status'] = status_code |
| 84 | + val['size'] = content_length |
| 85 | + val['referer'] = environ.get('HTTP_REFERER', '') |
| 86 | + val['agent'] = environ.get('HTTP_USER_AGENT', '') |
| 87 | + |
| 88 | + # see http://docs.python.org/3/library/string.html#format-string-syntax |
| 89 | + FORMAT = '{host} {logname} {user} [{time}] "{request}" ' |
| 90 | + FORMAT += '{status} {size} "{referer}" "{agent}"' |
| 91 | + return FORMAT.format(**val) |
| 92 | + |
| 93 | + @staticmethod |
| 94 | + def format_with_response_time(*args, **kw): |
| 95 | + """ |
| 96 | + The dict kw should contain 'rt_ms', the response time in milliseconds. |
| 97 | + This is the format for TinyLogAnalyzer: |
| 98 | + https://pypi.python.org/pypi/TinyLogAnalyzer |
| 99 | + """ |
| 100 | + rt_ms = kw.get('rt_ms') |
| 101 | + return ApacheFormatters.format_NCSA_log(*args) + " {}/{}".format(int(rt_ms/1000000), rt_ms) |
| 102 | + |
| 103 | +def log(handlers, formatter=ApacheFormatter(), **kw): |
| 104 | + '''Decorator for logging middleware.''' |
| 105 | + def decorator(application): |
| 106 | + return WSGILogger(application, handlers, **kw) |
| 107 | + return decorator |
| 108 | + |
0 commit comments