Fixing bug with x-trans-id. Will now be set on all incoming requests to proxy and trans-ids will not be reused.

Change-Id: Ieb59cadd9925a122566c98374b9736e77d3f98dd
This commit is contained in:
David Goetz 2011-10-13 09:36:17 -07:00
parent 30e71fa175
commit 215b3eb867
8 changed files with 50 additions and 39 deletions

View File

@ -22,7 +22,8 @@ from swift.common.utils import get_logger
class CatchErrorMiddleware(object): class CatchErrorMiddleware(object):
""" """
Middleware that provides high-level error handling. Middleware that provides high-level error handling and ensures that a
transaction id will be set for every request.
""" """
def __init__(self, app, conf): def __init__(self, app, conf):
@ -30,10 +31,12 @@ class CatchErrorMiddleware(object):
self.logger = get_logger(conf, log_route='catch-errors') self.logger = get_logger(conf, log_route='catch-errors')
def __call__(self, env, start_response): def __call__(self, env, start_response):
trans_id = env.get('HTTP_X_TRANS_ID') """
if not trans_id: If used, this should be the first middleware in pipeline.
"""
trans_id = 'tx' + uuid.uuid4().hex trans_id = 'tx' + uuid.uuid4().hex
env['HTTP_X_TRANS_ID'] = trans_id env['swift.trans_id'] = trans_id
self.logger.txn_id = trans_id
try: try:
def my_start_response(status, response_headers, exc_info=None): def my_start_response(status, response_headers, exc_info=None):

View File

@ -214,7 +214,7 @@ class StaticWeb(object):
""" """
new_env = {'REQUEST_METHOD': 'GET', new_env = {'REQUEST_METHOD': 'GET',
'HTTP_USER_AGENT': '%s StaticWeb' % env.get('HTTP_USER_AGENT')} 'HTTP_USER_AGENT': '%s StaticWeb' % env.get('HTTP_USER_AGENT')}
for name in ('eventlet.posthooks', 'HTTP_X_CF_TRANS_ID', 'REMOTE_USER', for name in ('eventlet.posthooks', 'swift.trans_id', 'REMOTE_USER',
'SCRIPT_NAME', 'SERVER_NAME', 'SERVER_PORT', 'SCRIPT_NAME', 'SERVER_NAME', 'SERVER_PORT',
'SERVER_PROTOCOL', 'swift.cache'): 'SERVER_PROTOCOL', 'swift.cache'):
if name in env: if name in env:
@ -532,7 +532,7 @@ class StaticWeb(object):
'-', '-',
'-', '-',
env.get('HTTP_ETAG', '-'), env.get('HTTP_ETAG', '-'),
env.get('HTTP_X_CF_TRANS_ID', '-'), env.get('swift.trans_id', '-'),
logged_headers or '-', logged_headers or '-',
trans_time))) trans_time)))

View File

@ -465,7 +465,7 @@ class TempAuth(object):
getattr(req, 'bytes_transferred', 0) or '-', getattr(req, 'bytes_transferred', 0) or '-',
getattr(response, 'bytes_transferred', 0) or '-', getattr(response, 'bytes_transferred', 0) or '-',
req.headers.get('etag', '-'), req.headers.get('etag', '-'),
req.headers.get('x-trans-id', '-'), logged_headers or '-', req.environ.get('swift.trans_id', '-'), logged_headers or '-',
trans_time))) trans_time)))

View File

@ -303,8 +303,7 @@ class LogAdapter(logging.LoggerAdapter, object):
client ip. client ip.
""" """
_txn_id = threading.local() _cls_thread_local = 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,21 +312,21 @@ class LogAdapter(logging.LoggerAdapter, object):
@property @property
def txn_id(self): def txn_id(self):
if hasattr(self._txn_id, 'value'): if hasattr(self._cls_thread_local, 'txn_id'):
return self._txn_id.value return self._cls_thread_local.txn_id
@txn_id.setter @txn_id.setter
def txn_id(self, value): def txn_id(self, value):
self._txn_id.value = value self._cls_thread_local.txn_id = value
@property @property
def client_ip(self): def client_ip(self):
if hasattr(self._client_ip, 'value'): if hasattr(self._cls_thread_local, 'client_ip'):
return self._client_ip.value return self._cls_thread_local.client_ip
@client_ip.setter @client_ip.setter
def client_ip(self, value): def client_ip(self, value):
self._client_ip.value = value self._cls_thread_local.client_ip = value
def getEffectiveLevel(self): def getEffectiveLevel(self):
return self.logger.getEffectiveLevel() return self.logger.getEffectiveLevel()

View File

@ -207,7 +207,7 @@ def make_pre_authed_request(env, method, path, body=None, headers=None,
(Stolen from Swauth: https://github.com/gholt/swauth) (Stolen from Swauth: https://github.com/gholt/swauth)
""" """
newenv = {'REQUEST_METHOD': method, 'HTTP_USER_AGENT': agent} newenv = {'REQUEST_METHOD': method, 'HTTP_USER_AGENT': agent}
for name in ('swift.cache', 'HTTP_X_TRANS_ID'): for name in ('swift.cache', 'swift.trans_id'):
if name in env: if name in env:
newenv[name] = env[name] newenv[name] = env[name]
newenv['swift.authorize'] = lambda req: None newenv['swift.authorize'] = lambda req: None

View File

@ -1632,8 +1632,13 @@ class BaseApplication(object):
return HTTPPreconditionFailed(request=req, body='Bad URL') return HTTPPreconditionFailed(request=req, body='Bad URL')
controller = controller(self, **path_parts) controller = controller(self, **path_parts)
controller.trans_id = req.headers.get('x-trans-id', '-') if 'swift.trans_id' not in req.environ:
self.logger.txn_id = req.headers.get('x-trans-id', None) # if this wasn't set by an earlier middleware, set it now
trans_id = 'tx' + uuid.uuid4().hex
req.environ['swift.trans_id'] = trans_id
self.logger.txn_id = trans_id
req.headers['x-trans-id'] = req.environ['swift.trans_id']
controller.trans_id = req.environ['swift.trans_id']
self.logger.client_ip = get_remote_client(req) self.logger.client_ip = get_remote_client(req)
try: try:
handler = getattr(controller, req.method) handler = getattr(controller, req.method)
@ -1708,10 +1713,12 @@ class Application(BaseApplication):
getattr(req, 'bytes_transferred', 0) or '-', getattr(req, 'bytes_transferred', 0) or '-',
getattr(response, 'bytes_transferred', 0) or '-', getattr(response, 'bytes_transferred', 0) or '-',
req.headers.get('etag', '-'), req.headers.get('etag', '-'),
req.headers.get('x-trans-id', '-'), req.environ.get('swift.trans_id', '-'),
logged_headers or '-', logged_headers or '-',
trans_time, trans_time,
))) )))
# done with this transaction
self.access_logger.txn_id = None
def app_factory(global_conf, **local_conf): def app_factory(global_conf, **local_conf):

View File

@ -15,29 +15,36 @@
import unittest import unittest
from webob import Request from webob import Request, Response
from swift.common.middleware import catch_errors from swift.common.middleware import catch_errors
from swift.common.utils import get_logger
class FakeApp(object): class FakeApp(object):
def __init__(self, error=False): def __init__(self, error=False):
self.error = error self.error = error
def __call__(self, env, start_response): def __call__(self, env, start_response):
if 'swift.trans_id' not in env:
raise Exception('Trans id should always be in env')
if self.error: if self.error:
raise Exception('augh!') raise Exception('An error occurred')
return "FAKE APP" return ["FAKE APP"]
def start_response(*args): def start_response(*args):
pass pass
class TestCatchErrors(unittest.TestCase): class TestCatchErrors(unittest.TestCase):
def setUp(self):
self.logger = get_logger({})
self.logger.txn_id = None
def test_catcherrors_passthrough(self): def test_catcherrors_passthrough(self):
app = catch_errors.CatchErrorMiddleware(FakeApp(), {}) app = catch_errors.CatchErrorMiddleware(FakeApp(), {})
req = Request.blank('/', environ={'REQUEST_METHOD': 'GET'}) req = Request.blank('/', environ={'REQUEST_METHOD': 'GET'})
resp = app(req.environ, start_response) resp = app(req.environ, start_response)
self.assertEquals(resp, 'FAKE APP') self.assertEquals(resp, ['FAKE APP'])
def test_catcherrors(self): def test_catcherrors(self):
app = catch_errors.CatchErrorMiddleware(FakeApp(True), {}) app = catch_errors.CatchErrorMiddleware(FakeApp(True), {})
@ -45,28 +52,23 @@ class TestCatchErrors(unittest.TestCase):
resp = app(req.environ, start_response) resp = app(req.environ, start_response)
self.assertEquals(resp, ['An error occurred']) self.assertEquals(resp, ['An error occurred'])
def test_trans_id_header(self): def test_trans_id_header_pass(self):
self.assertEquals(self.logger.txn_id, None)
def start_response(status, headers): def start_response(status, headers):
self.assert_('x-trans-id' in (x[0] for x in headers)) self.assert_('x-trans-id' in (x[0] for x in headers))
app = catch_errors.CatchErrorMiddleware(FakeApp(), {}) app = catch_errors.CatchErrorMiddleware(FakeApp(), {})
req = Request.blank('/v1/a')
app(req.environ, start_response)
app = catch_errors.CatchErrorMiddleware(FakeApp(), {})
req = Request.blank('/v1/a/c')
app(req.environ, start_response)
app = catch_errors.CatchErrorMiddleware(FakeApp(), {})
req = Request.blank('/v1/a/c/o') req = Request.blank('/v1/a/c/o')
app(req.environ, start_response) app(req.environ, start_response)
app = catch_errors.CatchErrorMiddleware(FakeApp(True), {}) self.assertEquals(len(self.logger.txn_id), 34) # 32 hex + 'tx'
req = Request.blank('/v1/a')
app(req.environ, start_response) def test_trans_id_header_fail(self):
app = catch_errors.CatchErrorMiddleware(FakeApp(True), {}) self.assertEquals(self.logger.txn_id, None)
req = Request.blank('/v1/a/c') def start_response(status, headers):
app(req.environ, start_response) self.assert_('x-trans-id' in (x[0] for x in headers))
app = catch_errors.CatchErrorMiddleware(FakeApp(True), {}) app = catch_errors.CatchErrorMiddleware(FakeApp(True), {})
req = Request.blank('/v1/a/c/o') req = Request.blank('/v1/a/c/o')
app(req.environ, start_response) app(req.environ, start_response)
self.assertEquals(len(self.logger.txn_id), 34)
if __name__ == '__main__': if __name__ == '__main__':
unittest.main() unittest.main()

View File

@ -179,7 +179,7 @@ class TestWSGI(unittest.TestCase):
@classmethod @classmethod
def fake_blank(cls, path, environ={}, body='', headers={}): def fake_blank(cls, path, environ={}, body='', headers={}):
self.assertEquals(environ['swift.authorize']('test'), None) self.assertEquals(environ['swift.authorize']('test'), None)
self.assertEquals(environ['HTTP_X_TRANS_ID'], '1234') self.assertFalse('HTTP_X_TRANS_ID' in environ)
was_blank = Request.blank was_blank = Request.blank
Request.blank = FakeReq.fake_blank Request.blank = FakeReq.fake_blank
wsgi.make_pre_authed_request({'HTTP_X_TRANS_ID': '1234'}, wsgi.make_pre_authed_request({'HTTP_X_TRANS_ID': '1234'},