mirror of
https://github.com/sprockets/sprockets.logging.git
synced 2024-11-28 11:19:51 +00:00
Merge pull request #1 from sprockets/implement-context-filter
Implement context filter
This commit is contained in:
commit
2e553ebcfa
15 changed files with 205 additions and 59 deletions
|
@ -6,9 +6,8 @@ python:
|
||||||
- pypy
|
- pypy
|
||||||
- 3.4
|
- 3.4
|
||||||
before_install:
|
before_install:
|
||||||
- pip install codecov
|
- pip install nose coverage codecov
|
||||||
install:
|
install:
|
||||||
- pip install -r test-requirements.txt
|
|
||||||
- pip install -e .
|
- pip install -e .
|
||||||
script: nosetests
|
script: nosetests
|
||||||
after_success:
|
after_success:
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
include LICENSE
|
include LICENSE
|
||||||
include README.rst
|
include README.rst
|
||||||
include HISTORY.rst
|
include HISTORY.rst
|
||||||
include *requirements.txt
|
|
||||||
include tests.py
|
include tests.py
|
||||||
graft docs
|
graft docs
|
||||||
global-exclude __pycache__
|
global-exclude __pycache__
|
||||||
|
|
16
README.rst
16
README.rst
|
@ -2,7 +2,7 @@ sprockets.logging
|
||||||
=================
|
=================
|
||||||
Making logs nicer since 2015!
|
Making logs nicer since 2015!
|
||||||
|
|
||||||
|Version| |Downloads| |Status| |Coverage| |License|
|
|Version| |Downloads| |Travis| |CodeCov| |ReadTheDocs|
|
||||||
|
|
||||||
Installation
|
Installation
|
||||||
------------
|
------------
|
||||||
|
@ -24,7 +24,7 @@ Requirements
|
||||||
|
|
||||||
Example
|
Example
|
||||||
-------
|
-------
|
||||||
This examples demonstrates how to use ``sprockets.logging`` by ...
|
This examples demonstrates the most basic usage of ``sprockets.logging``
|
||||||
|
|
||||||
.. code-block:: python
|
.. code-block:: python
|
||||||
|
|
||||||
|
@ -33,13 +33,15 @@ This examples demonstrates how to use ``sprockets.logging`` by ...
|
||||||
|
|
||||||
import sprockets.logging
|
import sprockets.logging
|
||||||
|
|
||||||
|
|
||||||
formatter = logging.Formatter('%(levelname)s %(message)s {%(context)s}')
|
formatter = logging.Formatter('%(levelname)s %(message)s {%(context)s}')
|
||||||
handler = logging.StreamHandler(sys.stdout)
|
handler = logging.StreamHandler(sys.stdout)
|
||||||
handler.setFormatter(formatter)
|
handler.setFormatter(formatter)
|
||||||
handler.addFilter(sprockets.logging.ContextFilter(properties=['context']))
|
handler.addFilter(sprockets.logging.ContextFilter(properties=['context']))
|
||||||
logging.Logger.root.addHandler(handler)
|
logging.Logger.root.addHandler(handler)
|
||||||
|
logging.Logger.root.setLevel(logging.DEBUG)
|
||||||
|
|
||||||
# Outputs: INFO Hi there {}
|
# Outputs: INFO Hi there {None}
|
||||||
logging.info('Hi there')
|
logging.info('Hi there')
|
||||||
|
|
||||||
# Outputs: INFO No KeyError {bah}
|
# Outputs: INFO No KeyError {bah}
|
||||||
|
@ -47,7 +49,7 @@ This examples demonstrates how to use ``sprockets.logging`` by ...
|
||||||
|
|
||||||
# Outputs: INFO Now with context! {foo}
|
# Outputs: INFO Now with context! {foo}
|
||||||
adapted = logging.LoggerAdapter(logging.Logger.root, extra={'context': 'foo'})
|
adapted = logging.LoggerAdapter(logging.Logger.root, extra={'context': 'foo'})
|
||||||
adapter.info('Now with context!')
|
adapted.info('Now with context!')
|
||||||
|
|
||||||
Source
|
Source
|
||||||
------
|
------
|
||||||
|
@ -61,14 +63,14 @@ License
|
||||||
.. |Version| image:: https://badge.fury.io/py/sprockets.logging.svg?
|
.. |Version| image:: https://badge.fury.io/py/sprockets.logging.svg?
|
||||||
:target: http://badge.fury.io/py/sprockets.logging
|
:target: http://badge.fury.io/py/sprockets.logging
|
||||||
|
|
||||||
.. |Status| image:: https://travis-ci.org/sprockets/sprockets.logging.svg?branch=master
|
.. |Travis| image:: https://travis-ci.org/sprockets/sprockets.logging.svg?branch=master
|
||||||
:target: https://travis-ci.org/sprockets/sprockets.logging
|
:target: https://travis-ci.org/sprockets/sprockets.logging
|
||||||
|
|
||||||
.. |Coverage| image:: http://codecov.io/github/sprockets/sprockets.logging/coverage.svg?branch=master
|
.. |CodeCov| image:: http://codecov.io/github/sprockets/sprockets.logging/coverage.svg?branch=master
|
||||||
:target: https://codecov.io/github/sprockets/sprockets.logging?branch=master
|
:target: https://codecov.io/github/sprockets/sprockets.logging?branch=master
|
||||||
|
|
||||||
.. |Downloads| image:: https://pypip.in/d/sprockets.logging/badge.svg?
|
.. |Downloads| image:: https://pypip.in/d/sprockets.logging/badge.svg?
|
||||||
:target: https://pypi.python.org/pypi/sprockets.logging
|
:target: https://pypi.python.org/pypi/sprockets.logging
|
||||||
|
|
||||||
.. |License| image:: https://pypip.in/license/sprockets.logging/badge.svg?
|
.. |ReadTheDocs| image:: https://readthedocs.org/projects/sprocketslogging/badge/
|
||||||
:target: https://sprocketslogging.readthedocs.org
|
:target: https://sprocketslogging.readthedocs.org
|
||||||
|
|
|
@ -1,6 +0,0 @@
|
||||||
-r requirements.txt
|
|
||||||
-r test-requirements.txt
|
|
||||||
flake8>=2.1,<3
|
|
||||||
sphinx>=1.2,<2
|
|
||||||
sphinx-rtd-theme>=0.1,<1.0
|
|
||||||
sphinxcontrib-httpdomain>=1.2,<2
|
|
|
@ -3,29 +3,17 @@ Examples
|
||||||
|
|
||||||
Simple Usage
|
Simple Usage
|
||||||
------------
|
------------
|
||||||
The following snippet uses :class:`sprockets.logging.filters.ContextFilter`
|
The following snippet uses :class:`sprockets.logging.ContextFilter`
|
||||||
to insert context information into a message using a
|
to insert context information into a message using a
|
||||||
:class:`logging.LoggerAdapter` instance.
|
:class:`logging.LoggerAdapter` instance.
|
||||||
|
|
||||||
.. code-block:: python
|
.. literalinclude:: ../examples/simple.py
|
||||||
|
|
||||||
import logging
|
Dictionary-based Configuration
|
||||||
import sys
|
------------------------------
|
||||||
|
This package begins to shine if you use the dictionary-based logging
|
||||||
|
configuration offered by :func:`logging.config.dictConfig`. You can insert
|
||||||
|
the custom filter and format string into the logging infrastructure and
|
||||||
|
insert context easily with :class:`logging.LoggerAdapter`.
|
||||||
|
|
||||||
import sprockets.logging
|
.. literalinclude:: ../examples/tornado-app.py
|
||||||
|
|
||||||
formatter = logging.Formatter('%(levelname)s %(message)s {%(context)s}')
|
|
||||||
handler = logging.StreamHandler(sys.stdout)
|
|
||||||
handler.setFormatter(formatter)
|
|
||||||
handler.addFilter(sprockets.logging.ContextFilter(properties=['context']))
|
|
||||||
logging.Logger.root.addHandler(handler)
|
|
||||||
|
|
||||||
# Outputs: INFO Hi there {}
|
|
||||||
logging.info('Hi there')
|
|
||||||
|
|
||||||
# Outputs: INFO No KeyError {bah}
|
|
||||||
logging.info('No KeyError', extra={'context': 'bah'})
|
|
||||||
|
|
||||||
# Outputs: INFO Now with context! {foo}
|
|
||||||
adapted = logging.LoggerAdapter(logging.Logger.root, extra={'context': 'foo'})
|
|
||||||
adapter.info('Now with context!')
|
|
||||||
|
|
|
@ -3,4 +3,4 @@ Version History
|
||||||
|
|
||||||
Next Release
|
Next Release
|
||||||
------------
|
------------
|
||||||
- implement greatness
|
- Added :class:`sprockets.logging.ContextFilter`
|
||||||
|
|
2
docs/requirements.txt
Normal file
2
docs/requirements.txt
Normal file
|
@ -0,0 +1,2 @@
|
||||||
|
sphinx-rtd-theme
|
||||||
|
sphinxcontrib-httpdomain
|
22
examples/simple.py
Normal file
22
examples/simple.py
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
import logging
|
||||||
|
import sys
|
||||||
|
|
||||||
|
import sprockets.logging
|
||||||
|
|
||||||
|
|
||||||
|
formatter = logging.Formatter('%(levelname)s %(message)s {%(context)s}')
|
||||||
|
handler = logging.StreamHandler(sys.stdout)
|
||||||
|
handler.setFormatter(formatter)
|
||||||
|
handler.addFilter(sprockets.logging.ContextFilter(properties=['context']))
|
||||||
|
logging.Logger.root.addHandler(handler)
|
||||||
|
logging.Logger.root.setLevel(logging.DEBUG)
|
||||||
|
|
||||||
|
# Outputs: INFO Hi there {None}
|
||||||
|
logging.info('Hi there')
|
||||||
|
|
||||||
|
# Outputs: INFO No KeyError {bah}
|
||||||
|
logging.info('No KeyError', extra={'context': 'bah'})
|
||||||
|
|
||||||
|
# Outputs: INFO Now with context! {foo}
|
||||||
|
adapted = logging.LoggerAdapter(logging.Logger.root, extra={'context': 'foo'})
|
||||||
|
adapted.info('Now with context!')
|
77
examples/tornado-app.py
Normal file
77
examples/tornado-app.py
Normal file
|
@ -0,0 +1,77 @@
|
||||||
|
import logging.config
|
||||||
|
import signal
|
||||||
|
import uuid
|
||||||
|
|
||||||
|
from tornado import ioloop, web
|
||||||
|
import sprockets.logging
|
||||||
|
|
||||||
|
|
||||||
|
LOG_CONFIG = {
|
||||||
|
'version': 1,
|
||||||
|
'handlers': {
|
||||||
|
'console': {
|
||||||
|
'class': 'logging.StreamHandler',
|
||||||
|
'stream': 'ext://sys.stdout',
|
||||||
|
'formatter': 'simple',
|
||||||
|
'filters': ['context'],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
'formatters': {
|
||||||
|
'simple': {
|
||||||
|
'class': 'logging.Formatter',
|
||||||
|
'format': '%(levelname)s %(name)s: %(message)s [%(context)s]',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
'filters': {
|
||||||
|
'context': {
|
||||||
|
'()': 'sprockets.logging.ContextFilter',
|
||||||
|
'properties': ['context'],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
'loggers': {
|
||||||
|
'tornado': {
|
||||||
|
'level': 'DEBUG',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
'root': {
|
||||||
|
'handlers': ['console'],
|
||||||
|
'level': 'DEBUG',
|
||||||
|
},
|
||||||
|
'incremental': False,
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
class RequestHandler(web.RequestHandler):
|
||||||
|
|
||||||
|
def __init__(self, *args, **kwargs):
|
||||||
|
self.parent_log = kwargs.pop('parent_log')
|
||||||
|
super(RequestHandler, self).__init__(*args, **kwargs)
|
||||||
|
|
||||||
|
def prepare(self):
|
||||||
|
uniq_id = self.request.headers.get('X-UniqID', uuid.uuid4().hex)
|
||||||
|
self.logger = logging.LoggerAdapter(
|
||||||
|
self.parent_log.getChild('RequestHandler'),
|
||||||
|
extra={'context': uniq_id})
|
||||||
|
|
||||||
|
def get(self, object_id):
|
||||||
|
self.logger.debug('fetchin %s', object_id)
|
||||||
|
self.set_status(200)
|
||||||
|
return self.finish()
|
||||||
|
|
||||||
|
def sig_handler(signo, frame):
|
||||||
|
logging.info('caught signal %d, stopping IO loop', signo)
|
||||||
|
iol = ioloop.IOLoop.instance()
|
||||||
|
iol.add_callback_from_signal(iol.stop)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
logging.config.dictConfig(LOG_CONFIG)
|
||||||
|
logger = logging.getLogger('app')
|
||||||
|
app = web.Application([
|
||||||
|
web.url('/(?P<object_id>\w+)', RequestHandler,
|
||||||
|
kwargs={'parent_log': logger}),
|
||||||
|
])
|
||||||
|
app.listen(8000)
|
||||||
|
signal.signal(signal.SIGINT, sig_handler)
|
||||||
|
signal.signal(signal.SIGTERM, sig_handler)
|
||||||
|
ioloop.IOLoop.instance().start()
|
||||||
|
logger.info('IO loop stopped, exiting')
|
20
setup.py
20
setup.py
|
@ -7,23 +7,9 @@ import setuptools
|
||||||
import sprockets.logging
|
import sprockets.logging
|
||||||
|
|
||||||
|
|
||||||
def read_requirements_file(req_name):
|
install_requires = []
|
||||||
requirements = []
|
setup_requires = []
|
||||||
try:
|
tests_require = ['nose>=1.3,<2']
|
||||||
with codecs.open(req_name, encoding='utf-8') as req_file:
|
|
||||||
for req_line in req_file:
|
|
||||||
if '#' in req_line:
|
|
||||||
req_line = req_line[0:req_line.find('#')].strip()
|
|
||||||
if req_line:
|
|
||||||
requirements.append(req_line.strip())
|
|
||||||
except IOError:
|
|
||||||
pass
|
|
||||||
return requirements
|
|
||||||
|
|
||||||
|
|
||||||
install_requires = read_requirements_file('requirements.txt')
|
|
||||||
setup_requires = read_requirements_file('setup-requirements.txt')
|
|
||||||
tests_require = read_requirements_file('test-requirements.txt')
|
|
||||||
|
|
||||||
if sys.version_info < (3, 0):
|
if sys.version_info < (3, 0):
|
||||||
tests_require.append('mock')
|
tests_require.append('mock')
|
||||||
|
|
|
@ -1,5 +1,39 @@
|
||||||
"""
|
"""
|
||||||
Make good log output easier.
|
Make good log output easier.
|
||||||
|
|
||||||
|
- :class:`ContextFilter` adds fixed properties to a log record
|
||||||
|
|
||||||
"""
|
"""
|
||||||
|
from __future__ import absolute_import
|
||||||
|
|
||||||
|
import logging
|
||||||
|
|
||||||
|
|
||||||
version_info = (0, 0, 0)
|
version_info = (0, 0, 0)
|
||||||
__version__ = '.'.join(str(v) for v in version_info)
|
__version__ = '.'.join(str(v) for v in version_info)
|
||||||
|
|
||||||
|
|
||||||
|
class ContextFilter(logging.Filter):
|
||||||
|
"""
|
||||||
|
Ensures that properties exist on a LogRecord.
|
||||||
|
|
||||||
|
:param list|None properties: optional list of properties that
|
||||||
|
will be added to LogRecord instances if they are missing
|
||||||
|
|
||||||
|
This filter implementation will ensure that a set of properties
|
||||||
|
exists on every log record which means that you can always refer
|
||||||
|
to custom properties in a format string. Without this, referring
|
||||||
|
to a property that is not explicitly passed in will result in an
|
||||||
|
ugly ``KeyError`` exception.
|
||||||
|
|
||||||
|
"""
|
||||||
|
|
||||||
|
def __init__(self, name='', properties=None):
|
||||||
|
logging.Filter.__init__(self, name)
|
||||||
|
self.properties = list(properties) if properties else []
|
||||||
|
|
||||||
|
def filter(self, record):
|
||||||
|
for property_name in self.properties:
|
||||||
|
if not hasattr(record, property_name):
|
||||||
|
setattr(record, property_name, None)
|
||||||
|
return True
|
||||||
|
|
|
@ -1,2 +0,0 @@
|
||||||
coverage>=3.7,<4
|
|
||||||
nose>=1.3,<2
|
|
44
tests.py
44
tests.py
|
@ -0,0 +1,44 @@
|
||||||
|
import logging
|
||||||
|
import uuid
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
import sprockets.logging
|
||||||
|
|
||||||
|
|
||||||
|
class Prototype(object):
|
||||||
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
class RecordingHandler(logging.FileHandler):
|
||||||
|
def __init__(self):
|
||||||
|
logging.FileHandler.__init__(self, filename='/dev/null')
|
||||||
|
self.log_lines = []
|
||||||
|
|
||||||
|
def format(self, record):
|
||||||
|
log_line = logging.FileHandler.format(self, record)
|
||||||
|
self.log_lines.append(log_line)
|
||||||
|
return log_line
|
||||||
|
|
||||||
|
|
||||||
|
class ContextFilterTests(unittest.TestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(ContextFilterTests, self).setUp()
|
||||||
|
self.logger = logging.getLogger(uuid.uuid4().hex)
|
||||||
|
self.handler = RecordingHandler()
|
||||||
|
self.logger.addHandler(self.handler)
|
||||||
|
|
||||||
|
def test_that_filter_blocks_key_errors(self):
|
||||||
|
formatter = logging.Formatter('%(message)s [%(context)s]')
|
||||||
|
self.handler.setFormatter(formatter)
|
||||||
|
self.handler.addFilter(sprockets.logging.ContextFilter(
|
||||||
|
properties=['context']))
|
||||||
|
self.logger.info('hi there')
|
||||||
|
|
||||||
|
def test_that_filter_does_not_overwrite_extras(self):
|
||||||
|
formatter = logging.Formatter('%(message)s [%(context)s]')
|
||||||
|
self.handler.setFormatter(formatter)
|
||||||
|
self.handler.addFilter(sprockets.logging.ContextFilter(
|
||||||
|
properties=['context']))
|
||||||
|
self.logger.info('hi there', extra={'context': 'foo'})
|
||||||
|
self.assertEqual(self.handler.log_lines[-1], 'hi there [foo]')
|
3
tox.ini
3
tox.ini
|
@ -3,7 +3,8 @@ envlist = py27,py34
|
||||||
indexserver =
|
indexserver =
|
||||||
default = https://pypi.python.org/simple
|
default = https://pypi.python.org/simple
|
||||||
toxworkdir = build/tox
|
toxworkdir = build/tox
|
||||||
|
skip_missing_interpreters = true
|
||||||
|
|
||||||
[testenv]
|
[testenv]
|
||||||
commands = nosetests []
|
commands = nosetests []
|
||||||
deps = -rtest-requirements.txt
|
deps = nose
|
||||||
|
|
Loading…
Reference in a new issue