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:
parent
d65a8f0afe
commit
e744501c47
@ -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.
|
||||||
|
@ -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
|
||||||
|
@ -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')
|
||||||
|
@ -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.")
|
||||||
|
@ -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'])
|
||||||
|
@ -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',
|
||||||
|
@ -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):
|
||||||
|
@ -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')
|
||||||
|
@ -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):
|
||||||
|
Loading…
Reference in New Issue
Block a user