123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960 |
- from __future__ import absolute_import
- from __future__ import unicode_literals
- import functools
- import logging
- import pprint
- from itertools import chain
- import six
- def format_call(args, kwargs):
- args = (repr(a) for a in args)
- kwargs = ("{0!s}={1!r}".format(*item) for item in six.iteritems(kwargs))
- return "({0})".format(", ".join(chain(args, kwargs)))
- def format_return(result, max_lines):
- if isinstance(result, (list, tuple, set)):
- return "({0} with {1} items)".format(type(result).__name__, len(result))
- if result:
- lines = pprint.pformat(result).split('\n')
- extra = '\n...' if len(lines) > max_lines else ''
- return '\n'.join(lines[:max_lines]) + extra
- return result
- class VerboseProxy(object):
- """Proxy all function calls to another class and log method name, arguments
- and return values for each call.
- """
- def __init__(self, obj_name, obj, log_name=None, max_lines=10):
- self.obj_name = obj_name
- self.obj = obj
- self.max_lines = max_lines
- self.log = logging.getLogger(log_name or __name__)
- def __getattr__(self, name):
- attr = getattr(self.obj, name)
- if not six.callable(attr):
- return attr
- return functools.partial(self.proxy_callable, name)
- def proxy_callable(self, call_name, *args, **kwargs):
- self.log.info("%s %s <- %s",
- self.obj_name,
- call_name,
- format_call(args, kwargs))
- result = getattr(self.obj, call_name)(*args, **kwargs)
- self.log.info("%s %s -> %s",
- self.obj_name,
- call_name,
- format_return(result, self.max_lines))
- return result
|