added client_ip to all proxy log lines not otherwise containing it

This commit is contained in:
John Dickinson 2011-08-12 16:27:27 +00:00 committed by Tarmac
commit b144d10d3d
3 changed files with 44 additions and 6 deletions

View File

@ -294,10 +294,12 @@ class LoggerFileObject(object):
class LogAdapter(logging.LoggerAdapter, object): class LogAdapter(logging.LoggerAdapter, object):
""" """
A Logger like object which performs some reformatting on calls to A Logger like object which performs some reformatting on calls to
:meth:`exception`. Can be used to store a threadlocal transaction id. :meth:`exception`. Can be used to store a threadlocal transaction id and
client ip.
""" """
_txn_id = threading.local() _txn_id = threading.local()
_client_ip = threading.local()
def __init__(self, logger, server): def __init__(self, logger, server):
logging.LoggerAdapter.__init__(self, logger, {}) logging.LoggerAdapter.__init__(self, logger, {})
@ -313,6 +315,15 @@ class LogAdapter(logging.LoggerAdapter, object):
def txn_id(self, value): def txn_id(self, value):
self._txn_id.value = value self._txn_id.value = value
@property
def client_ip(self):
if hasattr(self._client_ip, 'value'):
return self._client_ip.value
@client_ip.setter
def client_ip(self, value):
self._client_ip.value = value
def getEffectiveLevel(self): def getEffectiveLevel(self):
return self.logger.getEffectiveLevel() return self.logger.getEffectiveLevel()
@ -320,7 +331,8 @@ class LogAdapter(logging.LoggerAdapter, object):
""" """
Add extra info to message Add extra info to message
""" """
kwargs['extra'] = {'server': self.server, 'txn_id': self.txn_id} kwargs['extra'] = {'server': self.server, 'txn_id': self.txn_id,
'client_ip': self.client_ip}
return msg, kwargs return msg, kwargs
def notice(self, msg, *args, **kwargs): def notice(self, msg, *args, **kwargs):
@ -365,7 +377,7 @@ class LogAdapter(logging.LoggerAdapter, object):
call('%s: %s' % (msg, emsg), *args, **kwargs) call('%s: %s' % (msg, emsg), *args, **kwargs)
class TxnFormatter(logging.Formatter): class SwiftLogFormatter(logging.Formatter):
""" """
Custom logging.Formatter will append txn_id to a log message if the record Custom logging.Formatter will append txn_id to a log message if the record
has one and the message does not. has one and the message does not.
@ -376,6 +388,9 @@ class TxnFormatter(logging.Formatter):
if (record.txn_id and record.levelno != logging.INFO and if (record.txn_id and record.levelno != logging.INFO and
record.txn_id not in msg): record.txn_id not in msg):
msg = "%s (txn: %s)" % (msg, record.txn_id) msg = "%s (txn: %s)" % (msg, record.txn_id)
if (record.client_ip and record.levelno != logging.INFO and
record.client_ip not in msg):
msg = "%s (client_ip: %s)" % (msg, record.client_ip)
return msg return msg
@ -406,7 +421,7 @@ def get_logger(conf, name=None, log_to_console=False, log_route=None,
logger = logging.getLogger(log_route) logger = logging.getLogger(log_route)
logger.propagate = False logger.propagate = False
# all new handlers will get the same formatter # all new handlers will get the same formatter
formatter = TxnFormatter(fmt) formatter = SwiftLogFormatter(fmt)
# get_logger will only ever add one SysLog Handler to a logger # get_logger will only ever add one SysLog Handler to a logger
if not hasattr(get_logger, 'handler4logger'): if not hasattr(get_logger, 'handler4logger'):

View File

@ -1645,6 +1645,7 @@ class BaseApplication(object):
controller = controller(self, **path_parts) controller = controller(self, **path_parts)
controller.trans_id = req.headers.get('x-trans-id', '-') controller.trans_id = req.headers.get('x-trans-id', '-')
self.logger.txn_id = req.headers.get('x-trans-id', None) self.logger.txn_id = req.headers.get('x-trans-id', None)
self.logger.client_ip = get_remote_client(req)
try: try:
handler = getattr(controller, req.method) handler = getattr(controller, req.method)
if not getattr(handler, 'publicly_accessible'): if not getattr(handler, 'publicly_accessible'):

View File

@ -409,12 +409,12 @@ class TestUtils(unittest.TestCase):
logger.logger.removeHandler(handler) logger.logger.removeHandler(handler)
reset_loggers() reset_loggers()
def test_txn_formatter(self): def test_swift_log_formatter(self):
# setup stream logging # setup stream logging
sio = StringIO() sio = StringIO()
logger = utils.get_logger(None) logger = utils.get_logger(None)
handler = logging.StreamHandler(sio) handler = logging.StreamHandler(sio)
handler.setFormatter(utils.TxnFormatter()) handler.setFormatter(utils.SwiftLogFormatter())
logger.logger.addHandler(handler) logger.logger.addHandler(handler)
def strip_value(sio): def strip_value(sio):
@ -443,6 +443,28 @@ class TestUtils(unittest.TestCase):
self.assertEquals(logger.txn_id, '12345') self.assertEquals(logger.txn_id, '12345')
logger.warn('test 12345 test') logger.warn('test 12345 test')
self.assertEquals(strip_value(sio), 'test 12345 test\n') self.assertEquals(strip_value(sio), 'test 12345 test\n')
# test client_ip
self.assertFalse(logger.client_ip)
logger.error('my error message')
log_msg = strip_value(sio)
self.assert_('my error message' in log_msg)
self.assert_('client_ip' not in log_msg)
logger.client_ip = '1.2.3.4'
logger.error('test')
log_msg = strip_value(sio)
self.assert_('client_ip' in log_msg)
self.assert_('1.2.3.4' in log_msg)
# test no client_ip on info message
self.assertEquals(logger.client_ip, '1.2.3.4')
logger.info('test')
log_msg = strip_value(sio)
self.assert_('client_ip' not in log_msg)
self.assert_('1.2.3.4' not in log_msg)
# test client_ip (and txn) already in message
self.assertEquals(logger.client_ip, '1.2.3.4')
logger.warn('test 1.2.3.4 test 12345')
self.assertEquals(strip_value(sio), 'test 1.2.3.4 test 12345\n')
finally: finally:
logger.logger.removeHandler(handler) logger.logger.removeHandler(handler)
reset_loggers() reset_loggers()