
This change replaces the hard coded WSGI app creation with a pipeline of WSGI apps declared in a configuration file. Paste Deploy was used to create the pipeline since it is used by many other OpenStack projects and it is an active project with new contributors and supports Python 3. Dependency on Paste is localized so switching to another library would not be hard if OpenStack moves to another package in the future. The changes are small but the changes for the tests were large since many acl tests were assuming a hard coded WSGI app creation. blueprint declarative-filters Change-Id: I5ce05eab980271873269eca2945dc809f2923045
185 lines
6.0 KiB
Python
185 lines
6.0 KiB
Python
# -*- encoding: utf-8 -*-
|
|
#
|
|
# Copyright © 2012 New Dream Network, LLC (DreamHost)
|
|
#
|
|
# Author: Doug Hellmann <doug.hellmann@dreamhost.com>
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
import logging
|
|
import os
|
|
import socket
|
|
from wsgiref import simple_server
|
|
|
|
import netaddr
|
|
from oslo.config import cfg
|
|
from paste import deploy
|
|
import pecan
|
|
|
|
from ceilometer.api import config as api_config
|
|
from ceilometer.api import hooks
|
|
from ceilometer.api import middleware
|
|
from ceilometer.openstack.common import log
|
|
from ceilometer import storage
|
|
|
|
LOG = log.getLogger(__name__)
|
|
|
|
auth_opts = [
|
|
cfg.BoolOpt('enable_v1_api',
|
|
default=True,
|
|
help='Deploy the deprecated v1 API.'),
|
|
cfg.StrOpt('api_paste_config',
|
|
default="api_paste.ini",
|
|
help="Configuration file for WSGI definition of API."
|
|
),
|
|
]
|
|
|
|
CONF = cfg.CONF
|
|
CONF.register_opts(auth_opts)
|
|
|
|
|
|
def get_pecan_config():
|
|
# Set up the pecan configuration
|
|
filename = api_config.__file__.replace('.pyc', '.py')
|
|
return pecan.configuration.conf_from_file(filename)
|
|
|
|
|
|
def setup_app(pecan_config=None, extra_hooks=None):
|
|
# FIXME: Replace DBHook with a hooks.TransactionHook
|
|
app_hooks = [hooks.ConfigHook(),
|
|
hooks.DBHook(
|
|
storage.get_connection(cfg.CONF),
|
|
),
|
|
hooks.PipelineHook(),
|
|
hooks.TranslationHook()]
|
|
if extra_hooks:
|
|
app_hooks.extend(extra_hooks)
|
|
|
|
if not pecan_config:
|
|
pecan_config = get_pecan_config()
|
|
|
|
pecan.configuration.set_config(dict(pecan_config), overwrite=True)
|
|
|
|
app = pecan.make_app(
|
|
pecan_config.app.root,
|
|
static_root=pecan_config.app.static_root,
|
|
template_path=pecan_config.app.template_path,
|
|
debug=CONF.debug,
|
|
force_canonical=getattr(pecan_config.app, 'force_canonical', True),
|
|
hooks=app_hooks,
|
|
wrap_app=middleware.ParsableErrorMiddleware,
|
|
guess_content_type_from_ext=False
|
|
)
|
|
|
|
return app
|
|
|
|
|
|
class VersionSelectorApplication(object):
|
|
def __init__(self):
|
|
pc = get_pecan_config()
|
|
pc.app.debug = CONF.debug
|
|
if cfg.CONF.enable_v1_api:
|
|
from ceilometer.api.v1 import app as v1app
|
|
self.v1 = v1app.make_app(cfg.CONF)
|
|
else:
|
|
def not_found(environ, start_response):
|
|
start_response('404 Not Found', [])
|
|
return []
|
|
self.v1 = not_found
|
|
self.v2 = setup_app(pecan_config=pc)
|
|
|
|
def __call__(self, environ, start_response):
|
|
if environ['PATH_INFO'].startswith('/v1/'):
|
|
return self.v1(environ, start_response)
|
|
return self.v2(environ, start_response)
|
|
|
|
|
|
def get_server_cls(host):
|
|
"""Return an appropriate WSGI server class base on provided host
|
|
|
|
:param host: The listen host for the ceilometer API server.
|
|
"""
|
|
server_cls = simple_server.WSGIServer
|
|
if netaddr.valid_ipv6(host):
|
|
# NOTE(dzyu) make sure use IPv6 sockets if host is in IPv6 pattern
|
|
if getattr(server_cls, 'address_family') == socket.AF_INET:
|
|
class server_cls(server_cls):
|
|
address_family = socket.AF_INET6
|
|
return server_cls
|
|
|
|
|
|
def get_handler_cls():
|
|
cls = simple_server.WSGIRequestHandler
|
|
|
|
# old-style class doesn't support super
|
|
class CeilometerHandler(cls, object):
|
|
def address_string(self):
|
|
if cfg.CONF.api.enable_reverse_dns_lookup:
|
|
return super(CeilometerHandler, self).address_string()
|
|
else:
|
|
# disable reverse dns lookup, directly return ip adress
|
|
return self.client_address[0]
|
|
|
|
return CeilometerHandler
|
|
|
|
|
|
def load_app():
|
|
# Build the WSGI app
|
|
cfg_file = cfg.CONF.api_paste_config
|
|
LOG.info("WSGI config requested: %s" % cfg_file)
|
|
if not os.path.exists(cfg_file):
|
|
# this code is to work around chicken-egg dependency between
|
|
# ceilometer gate jobs use of devstack and this change.
|
|
# The gate job uses devstack to run tempest.
|
|
# devstack does not copy api_paste.ini into /etc/ceilometer because it
|
|
# is introduced in this change. Once this is merged, we will change
|
|
# devstack to copy api_paste.ini and once that is merged will remove
|
|
# this code.
|
|
root = os.path.abspath(os.path.join(os.path.dirname(__file__),
|
|
'..', '..', 'etc', 'ceilometer'
|
|
)
|
|
)
|
|
cfg_file = os.path.join(root, cfg_file)
|
|
if not os.path.exists(cfg_file):
|
|
raise Exception('api_paste_config file not found')
|
|
LOG.info("Full WSGI config used: %s" % cfg_file)
|
|
return deploy.loadapp("config:" + cfg_file)
|
|
|
|
|
|
def build_server():
|
|
app = load_app()
|
|
# Create the WSGI server and start it
|
|
host, port = cfg.CONF.api.host, cfg.CONF.api.port
|
|
server_cls = get_server_cls(host)
|
|
|
|
srv = simple_server.make_server(host, port, app,
|
|
server_cls, get_handler_cls())
|
|
|
|
LOG.info(_('Starting server in PID %s') % os.getpid())
|
|
LOG.info(_("Configuration:"))
|
|
cfg.CONF.log_opt_values(LOG, logging.INFO)
|
|
|
|
if host == '0.0.0.0':
|
|
LOG.info(_(
|
|
'serving on 0.0.0.0:%(sport)s, view at http://127.0.0.1:%(vport)s')
|
|
% ({'sport': port, 'vport': port}))
|
|
else:
|
|
LOG.info(_("serving on http://%(host)s:%(port)s") % (
|
|
{'host': host, 'port': port}))
|
|
|
|
return srv
|
|
|
|
|
|
def app_factory(global_config, **local_conf):
|
|
return VersionSelectorApplication()
|