Support x-openstack-request-id for Heat

This patch aims to support "x-openstack-request-id" for Heat.

Nova, Cinder and Neutron have realized "x-openstack-request-id" functionality
by using oslo-incubator.middleware. This functionality will enable effective
log tracing between Heat and Heat caller.

Change-Id: Ida2f3d33b02557c2117aade27ccac7d9351886c9
Closes-bug: #1324065
This commit is contained in:
Mitsuru Kanabuchi 2014-06-16 21:40:36 +09:00
parent 31604c71a1
commit f6deffa73f
3 changed files with 31 additions and 3 deletions

View File

@ -1,7 +1,7 @@
# heat-api pipeline # heat-api pipeline
[pipeline:heat-api] [pipeline:heat-api]
pipeline = faultwrap ssl versionnegotiation authurl authtoken context apiv1app pipeline = request_id faultwrap ssl versionnegotiation authurl authtoken context apiv1app
# heat-api pipeline for standalone heat # heat-api pipeline for standalone heat
# ie. uses alternative auth backend that authenticates users against keystone # ie. uses alternative auth backend that authenticates users against keystone
@ -12,7 +12,7 @@ pipeline = faultwrap ssl versionnegotiation authurl authtoken context apiv1app
# flavor = standalone # flavor = standalone
# #
[pipeline:heat-api-standalone] [pipeline:heat-api-standalone]
pipeline = faultwrap ssl versionnegotiation authurl authpassword context apiv1app pipeline = request_id faultwrap ssl versionnegotiation authurl authpassword context apiv1app
# heat-api pipeline for custom cloud backends # heat-api pipeline for custom cloud backends
# i.e. in heat.conf: # i.e. in heat.conf:
@ -20,7 +20,7 @@ pipeline = faultwrap ssl versionnegotiation authurl authpassword context apiv1ap
# flavor = custombackend # flavor = custombackend
# #
[pipeline:heat-api-custombackend] [pipeline:heat-api-custombackend]
pipeline = faultwrap versionnegotiation context custombackendauth apiv1app pipeline = request_id faultwrap versionnegotiation context custombackendauth apiv1app
# heat-api-cfn pipeline # heat-api-cfn pipeline
[pipeline:heat-api-cfn] [pipeline:heat-api-cfn]
@ -93,3 +93,7 @@ paste.filter_factory = heat.common.auth_password:filter_factory
# Auth middleware that validates against custom backend # Auth middleware that validates against custom backend
[filter:custombackendauth] [filter:custombackendauth]
paste.filter_factory = heat.common.custom_backend_auth:filter_factory paste.filter_factory = heat.common.custom_backend_auth:filter_factory
# Middleware to set x-openstack-request-id in http response header
[filter:request_id]
paste.filter_factory = heat.openstack.common.middleware.request_id:RequestIdMiddleware.factory

View File

@ -20,6 +20,7 @@ from heat.db import api as db_api
from heat.openstack.common import context from heat.openstack.common import context
from heat.openstack.common import importutils from heat.openstack.common import importutils
from heat.openstack.common import local from heat.openstack.common import local
from heat.openstack.common.middleware import request_id
def generate_request_id(): def generate_request_id():
@ -152,6 +153,7 @@ class ContextMiddleware(wsgi.Middleware):
if roles is not None: if roles is not None:
roles = roles.split(',') roles = roles.split(',')
token_info = environ.get('keystone.token_info') token_info = environ.get('keystone.token_info')
req_id = environ.get(request_id.ENV_REQUEST_ID)
except Exception: except Exception:
raise exception.NotAuthenticated() raise exception.NotAuthenticated()
@ -164,6 +166,7 @@ class ContextMiddleware(wsgi.Middleware):
password=password, password=password,
auth_url=auth_url, auth_url=auth_url,
roles=roles, roles=roles,
request_id=req_id,
auth_token_info=token_info) auth_token_info=token_info)

View File

@ -18,6 +18,7 @@ import webob
from heat.common import context from heat.common import context
from heat.common import exception from heat.common import exception
from heat.openstack.common.middleware import request_id
from heat.openstack.common import policy as base_policy from heat.openstack.common import policy as base_policy
from heat.tests.common import HeatTestCase from heat.tests.common import HeatTestCase
@ -256,3 +257,23 @@ class RequestContextMiddlewareTest(HeatTestCase):
ctx = request.context.to_dict() ctx = request.context.to_dict()
for k, v in self.context_dict.items(): for k, v in self.context_dict.items():
self.assertEqual(v, ctx[k], 'Key %s values do not match' % k) self.assertEqual(v, ctx[k], 'Key %s values do not match' % k)
self.assertIsNotNone(ctx.get('request_id'))
def test_context_middleware_with_requestid(self):
middleware = context.ContextMiddleware(None, None)
request = webob.Request.blank('/stacks', headers=self.headers,
environ=self.environ)
req_id = 'req-5a63f0d7-1b69-447b-b621-4ea87cc7186d'
request.environ[request_id.ENV_REQUEST_ID] = req_id
if self.expected_exception:
self.assertRaises(
self.expected_exception, middleware.process_request, request)
else:
self.assertIsNone(middleware.process_request(request))
ctx = request.context.to_dict()
for k, v in self.context_dict.items():
self.assertEqual(v, ctx[k], 'Key %s values do not match' % k)
self.assertEqual(
ctx.get('request_id'), req_id,
'Key request_id values do not match')