Remove usage of oslo.config global

Currently application that doesn't use the global configuration object
have to rely on hack to setup the global oslo config object for each middleware
it want to use.

For example, gnocchi have its own middleware loader and add crap to load
keystonemiddleware:

  https://github.com/openstack/gnocchi/blob/master/gnocchi/rest/app.py#L140

And it can't use oslo.middleware that relies on the global conf object.

Also aodh (use 'paste' for middleware) have to hack the global
configuration object for each middlewares it want to use by code...

  https://review.openstack.org/#/c/208632/1/aodh/service.py

But middleware are optional deployer stuffs, we should not write any
code for them...

This change allows application to use paste-deploy (or any middleware
loader) without enforcing the application to use the global oslo.config object.

If the middleware want to use oslo.config it should load the
configuration file himself (and fallback to the global one if any)

The proposed paste configuration to allow this is:

  [filter:cors]
  paste.filter_factory = oslo.middleware:cors
  oslo_config_project = aodh

So the cors middleware can find and load the aodh config and
what is it interested in.

Also, some of them use oslo.config local, some other the global object.
Some can be loaded by an middleware loader like paste, some other not.

This change make consistent the way we bootstrap all middlewares.

Closes-bug: #1482086

Change-Id: Iad197d1f3a386683d818b59718df34e14e15ca5c
This commit is contained in:
Mehdi Abaakouk 2015-08-06 09:15:57 +02:00
parent d65a8f0afe
commit e744501c47
9 changed files with 91 additions and 75 deletions

View File

@ -18,6 +18,8 @@
from inspect import getargspec from inspect import getargspec
import webob.dec import webob.dec
from oslo_config import cfg
class Middleware(object): class Middleware(object):
"""Base WSGI middleware wrapper. """Base WSGI middleware wrapper.
@ -30,10 +32,36 @@ class Middleware(object):
@classmethod @classmethod
def factory(cls, global_conf, **local_conf): def factory(cls, global_conf, **local_conf):
"""Factory method for paste.deploy.""" """Factory method for paste.deploy."""
return cls conf = global_conf.copy() if global_conf else {}
conf.update(local_conf)
def __init__(self, application): def middleware_filter(app):
return cls(app, conf)
return middleware_filter
def __init__(self, application, conf=None):
self.application = application self.application = application
# NOTE(sileht): If the configuration come from oslo.config
# just use it.
if isinstance(conf, cfg.ConfigOpts):
self.conf = []
self.oslo_conf = conf
else:
self.conf = conf or []
if "oslo_config_project" in self.conf:
if 'oslo_config_file' in self.conf:
default_config_files = [self.conf['oslo_config_file']]
else:
default_config_files = None
self.oslo_conf = cfg.ConfigOpts()
self.oslo_conf([], project=self.conf['oslo_config_project'],
default_config_files=default_config_files,
validate_default_values=True)
else:
# Fallback to global object
self.oslo_conf = cfg.CONF
def process_request(self, req): def process_request(self, req):
"""Called on each request. """Called on each request.

View File

@ -54,38 +54,6 @@ CORS_OPTS = [
] ]
def filter_factory(global_conf,
allowed_origin,
allow_credentials=True,
expose_headers=None,
max_age=None,
allow_methods=None,
allow_headers=None):
'''Factory to support paste.deploy
:param global_conf: The paste.ini global configuration object (not used).
:param allowed_origin: Protocol, host, and port for the allowed origin.
:param allow_credentials: Whether to permit credentials.
:param expose_headers: A list of headers to expose.
:param max_age: Maximum cache duration.
:param allow_methods: List of HTTP methods to permit.
:param allow_headers: List of HTTP headers to permit from the client.
:return:
'''
def filter(app):
cors_app = CORS(app)
cors_app.add_origin(allowed_origin=allowed_origin,
allow_credentials=allow_credentials,
expose_headers=expose_headers,
max_age=max_age,
allow_methods=allow_methods,
allow_headers=allow_headers)
return cors_app
return filter
class CORS(base.Middleware): class CORS(base.Middleware):
"""CORS Middleware. """CORS Middleware.
@ -105,15 +73,39 @@ class CORS(base.Middleware):
] ]
def __init__(self, application, conf=None): def __init__(self, application, conf=None):
super(CORS, self).__init__(application) super(CORS, self).__init__(application, conf)
# Begin constructing our configuration hash. # Begin constructing our configuration hash.
self.allowed_origins = {} self.allowed_origins = {}
# Sanity check. Do we have an oslo.config? If so, load it. Else, assume self._init_from_oslo(self.oslo_conf)
# that we'll use add_config. self._init_from_conf()
if conf:
self._init_from_oslo(conf) @classmethod
def factory(cls, global_conf, allowed_origin, **local_conf):
# Ensures allowed_origin config exists
return super(CORS, cls).factory(global_conf,
allowed_origin=allowed_origin,
**local_conf)
def _init_from_conf(self):
"""Load configuration from paste.deploy
allowed_origin: Protocol, host, and port for the allowed origin.
allow_credentials: Whether to permit credentials.
expose_headers: A list of headers to expose.
max_age: Maximum cache duration.
allow_methods: List of HTTP methods to permit.
allow_headers: List of HTTP headers to permit from the client.
"""
if 'allowed_origin' in self.conf:
self.add_origin(
allowed_origin=self.conf['allowed_origin'],
allow_credentials=self.conf.get('allow_credentials', True),
expose_headers=self.conf.get('expose_headers'),
max_age=self.conf.get('max_age'),
allow_methods=self.conf.get('allow_methods'),
allow_headers=self.conf.get('allow_headers'))
def _init_from_oslo(self, conf): def _init_from_oslo(self, conf):
'''Initialize this middleware from an oslo.config instance.''' '''Initialize this middleware from an oslo.config instance.'''
@ -327,3 +319,7 @@ class CORS(base.Middleware):
if cors_config['expose_headers']: if cors_config['expose_headers']:
response.headers['Access-Control-Expose-Headers'] = \ response.headers['Access-Control-Expose-Headers'] = \
','.join(cors_config['expose_headers']) ','.join(cors_config['expose_headers'])
# NOTE(sileht): Shortcut for backwards compatibility
filter_factory = CORS.factory

View File

@ -72,16 +72,6 @@ class Healthcheck(base.Middleware):
NAMESPACE = "oslo.middleware.healthcheck" NAMESPACE = "oslo.middleware.healthcheck"
@classmethod
def factory(cls, global_conf, **local_conf):
"""Factory method for paste.deploy."""
conf = global_conf.copy()
conf.update(local_conf)
def healthcheck_filter(app):
return cls(app, conf)
return healthcheck_filter
def __init__(self, application, conf): def __init__(self, application, conf):
super(Healthcheck, self).__init__(application) super(Healthcheck, self).__init__(application)
self._path = conf.get('path', '/healthcheck') self._path = conf.get('path', '/healthcheck')

View File

@ -18,7 +18,6 @@ Request Body limiting middleware.
""" """
from oslo_config import cfg from oslo_config import cfg
from oslo_config import cfgfilter
import webob.dec import webob.dec
import webob.exc import webob.exc
@ -40,9 +39,6 @@ _opts = [
deprecated_opts=_oldopts) deprecated_opts=_oldopts)
] ]
CONF = cfgfilter.ConfigFilter(cfg.CONF)
CONF.register_opts(_opts, group='oslo_middleware')
class LimitingReader(object): class LimitingReader(object):
"""Reader to limit the size of an incoming request.""" """Reader to limit the size of an incoming request."""
@ -82,9 +78,13 @@ class LimitingReader(object):
class RequestBodySizeLimiter(base.Middleware): class RequestBodySizeLimiter(base.Middleware):
"""Limit the size of incoming requests.""" """Limit the size of incoming requests."""
def __init__(self, application, conf=None):
super(RequestBodySizeLimiter, self).__init__(application, conf)
self.oslo_conf.register_opts(_opts, group='oslo_middleware')
@webob.dec.wsgify @webob.dec.wsgify
def __call__(self, req): def __call__(self, req):
max_size = CONF.oslo_middleware.max_request_body_size max_size = self.oslo_conf.oslo_middleware.max_request_body_size
if (req.content_length is not None and if (req.content_length is not None and
req.content_length > max_size): req.content_length > max_size):
msg = _("Request is too large.") msg = _("Request is too large.")

View File

@ -21,7 +21,6 @@ OPTS = [
"the original request protocol scheme was, even if it was " "the original request protocol scheme was, even if it was "
"hidden by an SSL termination proxy.") "hidden by an SSL termination proxy.")
] ]
cfg.CONF.register_opts(OPTS, group='oslo_middleware')
class SSLMiddleware(base.Middleware): class SSLMiddleware(base.Middleware):
@ -32,12 +31,13 @@ class SSLMiddleware(base.Middleware):
termination proxy. termination proxy.
""" """
def __init__(self, application): def __init__(self, application, conf=None):
super(SSLMiddleware, self).__init__(application) super(SSLMiddleware, self).__init__(application, conf)
self.header_name = 'HTTP_{0}'.format( self.oslo_conf.register_opts(OPTS, group='oslo_middleware')
cfg.CONF.oslo_middleware.secure_proxy_ssl_header.upper()
.replace('-', '_'))
def process_request(self, req): def process_request(self, req):
self.header_name = 'HTTP_{0}'.format(
self.oslo_conf.oslo_middleware.secure_proxy_ssl_header.upper()
.replace('-', '_'))
req.environ['wsgi.url_scheme'] = req.environ.get( req.environ['wsgi.url_scheme'] = req.environ.get(
self.header_name, req.environ['wsgi.url_scheme']) self.header_name, req.environ['wsgi.url_scheme'])

View File

@ -115,6 +115,9 @@ class CORSTestFilterFactory(test_base.BaseTestCase):
"""Test the CORS filter_factory method.""" """Test the CORS filter_factory method."""
def test_filter_factory(self): def test_filter_factory(self):
config = self.useFixture(fixture.Config())
config.conf([])
# Test a valid filter. # Test a valid filter.
filter = cors.filter_factory(None, filter = cors.filter_factory(None,
allowed_origin='http://valid.example.com', allowed_origin='http://valid.example.com',

View File

@ -78,15 +78,15 @@ class TestRequestBodySizeLimiter(test_base.BaseTestCase):
def setUp(self): def setUp(self):
super(TestRequestBodySizeLimiter, self).setUp() super(TestRequestBodySizeLimiter, self).setUp()
fixture = self.useFixture(config.Config(sizelimit.CONF)) self.useFixture(config.Config())
self.MAX_REQUEST_BODY_SIZE = \
fixture.conf.oslo_middleware.max_request_body_size
@webob.dec.wsgify() @webob.dec.wsgify()
def fake_app(req): def fake_app(req):
return webob.Response(req.body) return webob.Response(req.body)
self.middleware = sizelimit.RequestBodySizeLimiter(fake_app) self.middleware = sizelimit.RequestBodySizeLimiter(fake_app)
self.MAX_REQUEST_BODY_SIZE = (
self.middleware.oslo_conf.oslo_middleware.max_request_body_size)
self.request = webob.Request.blank('/', method='POST') self.request = webob.Request.blank('/', method='POST')
def test_content_length_acceptable(self): def test_content_length_acceptable(self):

View File

@ -13,7 +13,6 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
from oslo_config import cfg
from oslo_config import fixture as config from oslo_config import fixture as config
from oslotest import base from oslotest import base
import webob import webob
@ -27,8 +26,12 @@ class SSLMiddlewareTest(base.BaseTestCase):
super(SSLMiddlewareTest, self).setUp() super(SSLMiddlewareTest, self).setUp()
self.useFixture(config.Config()) self.useFixture(config.Config())
def _test_scheme(self, expected, headers): def _test_scheme(self, expected, headers, config=None):
middleware = ssl.SSLMiddleware(None) middleware = ssl.SSLMiddleware(None)
if config:
middleware.oslo_conf.set_override(
'secure_proxy_ssl_header', config,
group='oslo_middleware')
request = webob.Request.blank('http://example.com/', headers=headers) request = webob.Request.blank('http://example.com/', headers=headers)
# Ensure ssl middleware does not stop pipeline execution # Ensure ssl middleware does not stop pipeline execution
@ -44,13 +47,9 @@ class SSLMiddlewareTest(base.BaseTestCase):
self._test_scheme('https', headers) self._test_scheme('https', headers)
def test_with_custom_header(self): def test_with_custom_header(self):
cfg.CONF.set_override('secure_proxy_ssl_header', 'X-My-Header',
group='oslo_middleware')
headers = {'X-Forwarded-Proto': 'https'} headers = {'X-Forwarded-Proto': 'https'}
self._test_scheme('http', headers) self._test_scheme('http', headers, config='X-My-Header')
def test_with_custom_header_and_forwarded_protocol(self): def test_with_custom_header_and_forwarded_protocol(self):
cfg.CONF.set_override('secure_proxy_ssl_header', 'X-My-Header',
group='oslo_middleware')
headers = {'X-My-Header': 'https'} headers = {'X-My-Header': 'https'}
self._test_scheme('https', headers) self._test_scheme('https', headers, config='X-My-Header')

View File

@ -78,15 +78,15 @@ class TestRequestBodySizeLimiter(test_base.BaseTestCase):
def setUp(self): def setUp(self):
super(TestRequestBodySizeLimiter, self).setUp() super(TestRequestBodySizeLimiter, self).setUp()
fixture = self.useFixture(config.Config(sizelimit.CONF)) self.useFixture(config.Config())
self.MAX_REQUEST_BODY_SIZE = \
fixture.conf.oslo_middleware.max_request_body_size
@webob.dec.wsgify() @webob.dec.wsgify()
def fake_app(req): def fake_app(req):
return webob.Response(req.body) return webob.Response(req.body)
self.middleware = sizelimit.RequestBodySizeLimiter(fake_app) self.middleware = sizelimit.RequestBodySizeLimiter(fake_app)
self.MAX_REQUEST_BODY_SIZE = (
self.middleware.oslo_conf.oslo_middleware.max_request_body_size)
self.request = webob.Request.blank('/', method='POST') self.request = webob.Request.blank('/', method='POST')
def test_content_length_acceptable(self): def test_content_length_acceptable(self):