2015-06-17 21:46:32 +00:00
|
|
|
import json
|
2015-06-08 17:20:30 +00:00
|
|
|
import logging
|
2015-06-17 21:46:32 +00:00
|
|
|
from os import path
|
|
|
|
import random
|
|
|
|
import threading
|
2015-06-08 17:20:30 +00:00
|
|
|
import unittest
|
2015-06-17 21:46:32 +00:00
|
|
|
import uuid
|
2015-06-08 17:20:30 +00:00
|
|
|
|
2015-06-17 21:46:32 +00:00
|
|
|
import mock
|
2015-06-08 17:20:30 +00:00
|
|
|
import sprockets.logging
|
|
|
|
|
2015-06-17 21:46:32 +00:00
|
|
|
LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2015-06-08 17:20:30 +00:00
|
|
|
|
|
|
|
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]')
|
2015-06-17 21:46:32 +00:00
|
|
|
|
|
|
|
|
|
|
|
class MockRequest(object):
|
|
|
|
|
|
|
|
headers = {u'Accept': u'application/msgpack',
|
|
|
|
u'Correlation-ID': str(uuid.uuid4()).decode('utf-8')}
|
|
|
|
method = u'GET'
|
|
|
|
path = u'/test'
|
|
|
|
protocol = u'http'
|
|
|
|
remote_ip = u'127.0.0.1'
|
|
|
|
query_arguments = {u'mock': True}
|
|
|
|
|
|
|
|
def __init__(self):
|
|
|
|
self.duration = random.randint(10, 200)
|
|
|
|
|
|
|
|
def request_time(self):
|
|
|
|
return self.duration
|
|
|
|
|
|
|
|
|
|
|
|
class MockHandler(object):
|
|
|
|
|
|
|
|
def __init__(self, status_code=200):
|
|
|
|
self.status_code = status_code
|
|
|
|
self.request = MockRequest()
|
|
|
|
|
|
|
|
def get_status(self):
|
|
|
|
return self.status_code
|
|
|
|
|
|
|
|
|
|
|
|
class TornadoLogFunctionTestCase(unittest.TestCase):
|
|
|
|
|
|
|
|
@mock.patch('tornado.log.access_log')
|
|
|
|
def test_log_function_return_value(self, access_log):
|
|
|
|
handler = MockHandler()
|
|
|
|
expectation = ('', {u'correlation_id':
|
|
|
|
handler.request.headers['Correlation-ID'],
|
|
|
|
u'duration': handler.request.duration * 1000.0,
|
|
|
|
u'headers': handler.request.headers,
|
|
|
|
u'method': handler.request.method,
|
|
|
|
u'path': handler.request.path,
|
|
|
|
u'protocol': handler.request.protocol,
|
|
|
|
u'query_args': handler.request.query_arguments,
|
|
|
|
u'remote_ip': handler.request.remote_ip,
|
|
|
|
u'status_code': handler.status_code})
|
|
|
|
sprockets.logging.tornado_log_function(handler)
|
|
|
|
access_log.assertCalledOnceWith(expectation)
|
|
|
|
|
|
|
|
|
|
|
|
class JSONRequestHandlerTestCase(unittest.TestCase):
|
|
|
|
|
2015-06-17 21:55:39 +00:00
|
|
|
def setUp(self):
|
|
|
|
self.maxDiff = 32768
|
|
|
|
|
2015-06-17 21:46:32 +00:00
|
|
|
def test_log_function_return_value(self):
|
|
|
|
class LoggingHandler(logging.Handler):
|
|
|
|
def __init__(self, level):
|
|
|
|
super(LoggingHandler, self).__init__(level)
|
|
|
|
self.formatter = sprockets.logging.JSONRequestFormatter()
|
|
|
|
self.records = []
|
|
|
|
self.results = []
|
|
|
|
|
|
|
|
def handle(self, value):
|
|
|
|
self.records.append(value)
|
|
|
|
self.results.append(self.formatter.format(value))
|
|
|
|
|
|
|
|
logging_handler = LoggingHandler(logging.INFO)
|
|
|
|
LOGGER.addHandler(logging_handler)
|
|
|
|
|
|
|
|
handler = MockHandler()
|
|
|
|
args = {u'correlation_id':
|
|
|
|
handler.request.headers['Correlation-ID'],
|
|
|
|
u'duration': handler.request.duration * 1000.0,
|
|
|
|
u'headers': handler.request.headers,
|
|
|
|
u'method': handler.request.method,
|
|
|
|
u'path': handler.request.path,
|
|
|
|
u'protocol': handler.request.protocol,
|
|
|
|
u'query_args': handler.request.query_arguments,
|
|
|
|
u'remote_ip': handler.request.remote_ip,
|
|
|
|
u'status_code': handler.status_code}
|
|
|
|
|
|
|
|
LOGGER.info('', args)
|
|
|
|
record = logging_handler.records.pop(0)
|
|
|
|
result = logging_handler.results.pop(0)
|
|
|
|
expectation = \
|
|
|
|
{u'line_number': 130,
|
|
|
|
u'file': path.basename(globals()['__file__']),
|
|
|
|
u'level': u'INFO',
|
|
|
|
u'module': globals()['__name__'].decode('utf-8'),
|
|
|
|
u'name': globals()['__name__'].decode('utf-8'),
|
|
|
|
u'process': u'MainProcess',
|
|
|
|
u'thread': threading.current_thread().name.decode('utf-8'),
|
|
|
|
u'timestamp':
|
|
|
|
logging_handler.formatter.formatTime(record).decode('utf-8'),
|
|
|
|
u'request': args}
|
|
|
|
self.assertDictEqual(json.loads(result), expectation)
|