Allow neutron-api load config from WSGI process
neutron-api needs to load config files when deployed under WSGI process. This patch will do this. neutron-rpc-server also need to be aware of plugin extension. uWSGI seems not happy with eventlet.GreenPool. So this patch will switch to futurist.ThreadPoolExcutor for ML2 OVO push notification. Partially-implements: blueprint run-in-wsgi-server Change-Id: Ia2659c9c8f1deeceefc1c75f1149a1e47c91e7a9 Closes-Bug: #1708389
This commit is contained in:
		| @@ -28,6 +28,7 @@ from oslo_middleware import cors | |||||||
| from oslo_service import wsgi | from oslo_service import wsgi | ||||||
|  |  | ||||||
| from neutron._i18n import _ | from neutron._i18n import _ | ||||||
|  | from neutron.common import rpc as n_rpc | ||||||
| from neutron.conf import common as common_config | from neutron.conf import common as common_config | ||||||
| from neutron import policy | from neutron import policy | ||||||
| from neutron import version | from neutron import version | ||||||
| @@ -72,14 +73,12 @@ common_config.register_placement_opts() | |||||||
| logging.register_options(cfg.CONF) | logging.register_options(cfg.CONF) | ||||||
|  |  | ||||||
|  |  | ||||||
| def init(args, **kwargs): | def init(args, default_config_files=None, **kwargs): | ||||||
|     cfg.CONF(args=args, project='neutron', |     cfg.CONF(args=args, project='neutron', | ||||||
|              version='%%(prog)s %s' % version.version_info.release_string(), |              version='%%(prog)s %s' % version.version_info.release_string(), | ||||||
|  |              default_config_files=default_config_files, | ||||||
|              **kwargs) |              **kwargs) | ||||||
|  |  | ||||||
|     # FIXME(ihrachys): if import is put in global, circular import |  | ||||||
|     # failure occurs |  | ||||||
|     from neutron.common import rpc as n_rpc |  | ||||||
|     n_rpc.init(cfg.CONF) |     n_rpc.init(cfg.CONF) | ||||||
|  |  | ||||||
|     # Validate that the base_mac is of the correct format |     # Validate that the base_mac is of the correct format | ||||||
| @@ -119,6 +118,10 @@ def load_paste_app(app_name): | |||||||
|     :param app_name: Name of the application to load |     :param app_name: Name of the application to load | ||||||
|     """ |     """ | ||||||
|     loader = wsgi.Loader(cfg.CONF) |     loader = wsgi.Loader(cfg.CONF) | ||||||
|  |  | ||||||
|  |     # Log the values of registered opts | ||||||
|  |     if cfg.CONF.debug: | ||||||
|  |         cfg.CONF.log_opt_values(LOG, logging.DEBUG) | ||||||
|     app = loader.load_app(app_name) |     app = loader.load_app(app_name) | ||||||
|     return app |     return app | ||||||
|  |  | ||||||
|   | |||||||
| @@ -13,7 +13,9 @@ | |||||||
|  |  | ||||||
| import traceback | import traceback | ||||||
|  |  | ||||||
| import eventlet | import futurist | ||||||
|  | from futurist import waiters | ||||||
|  |  | ||||||
| from neutron_lib.callbacks import events | from neutron_lib.callbacks import events | ||||||
| from neutron_lib.callbacks import registry | from neutron_lib.callbacks import registry | ||||||
| from neutron_lib.callbacks import resources | from neutron_lib.callbacks import resources | ||||||
| @@ -38,7 +40,12 @@ class _ObjectChangeHandler(object): | |||||||
|         self._obj_class = object_class |         self._obj_class = object_class | ||||||
|         self._resource_push_api = resource_push_api |         self._resource_push_api = resource_push_api | ||||||
|         self._resources_to_push = {} |         self._resources_to_push = {} | ||||||
|         self._worker_pool = eventlet.GreenPool() |  | ||||||
|  |         # NOTE(annp): uWSGI seems not happy with eventlet.GreenPool. | ||||||
|  |         # So switching to ThreadPool | ||||||
|  |         self._worker_pool = futurist.ThreadPoolExecutor() | ||||||
|  |         self.fts = [] | ||||||
|  |  | ||||||
|         self._semantic_warned = False |         self._semantic_warned = False | ||||||
|         for event in (events.AFTER_CREATE, events.AFTER_UPDATE, |         for event in (events.AFTER_CREATE, events.AFTER_UPDATE, | ||||||
|                       events.AFTER_DELETE): |                       events.AFTER_DELETE): | ||||||
| @@ -46,7 +53,9 @@ class _ObjectChangeHandler(object): | |||||||
|  |  | ||||||
|     def wait(self): |     def wait(self): | ||||||
|         """Waits for all outstanding events to be dispatched.""" |         """Waits for all outstanding events to be dispatched.""" | ||||||
|         self._worker_pool.waitall() |         done, not_done = waiters.wait_for_all(self.fts) | ||||||
|  |         if not not_done: | ||||||
|  |             del self.fts[:] | ||||||
|  |  | ||||||
|     def _is_session_semantic_violated(self, context, resource, event): |     def _is_session_semantic_violated(self, context, resource, event): | ||||||
|         """Return True and print an ugly error on transaction violation. |         """Return True and print an ugly error on transaction violation. | ||||||
| @@ -82,7 +91,7 @@ class _ObjectChangeHandler(object): | |||||||
|         # to the server-side event that triggered it |         # to the server-side event that triggered it | ||||||
|         self._resources_to_push[resource_id] = context.to_dict() |         self._resources_to_push[resource_id] = context.to_dict() | ||||||
|         # spawn worker so we don't block main AFTER_UPDATE thread |         # spawn worker so we don't block main AFTER_UPDATE thread | ||||||
|         self._worker_pool.spawn(self.dispatch_events) |         self.fts.append(self._worker_pool.submit(self.dispatch_events)) | ||||||
|  |  | ||||||
|     @lockutils.synchronized('event-dispatch') |     @lockutils.synchronized('event-dispatch') | ||||||
|     def dispatch_events(self): |     def dispatch_events(self): | ||||||
|   | |||||||
| @@ -16,6 +16,7 @@ | |||||||
| # If ../neutron/__init__.py exists, add ../ to Python search path, so that | # If ../neutron/__init__.py exists, add ../ to Python search path, so that | ||||||
| # it will override what happens to be installed in /usr/(local/)lib/python... | # it will override what happens to be installed in /usr/(local/)lib/python... | ||||||
|  |  | ||||||
|  | import os | ||||||
| import sys | import sys | ||||||
|  |  | ||||||
| from oslo_config import cfg | from oslo_config import cfg | ||||||
| @@ -24,10 +25,35 @@ from neutron._i18n import _ | |||||||
| from neutron.common import config | from neutron.common import config | ||||||
| from neutron.common import profiler | from neutron.common import profiler | ||||||
|  |  | ||||||
|  | # NOTE(annp): These environment variables are required for deploying | ||||||
|  | # neutron-api under mod_wsgi. Currently, these variables are set as DevStack's | ||||||
|  | # default. If you intend to use other tools for deploying neutron-api under | ||||||
|  | # mod_wsgi, you should export these variables with your values. | ||||||
|  |  | ||||||
|  | NEUTRON_CONF = 'neutron.conf' | ||||||
|  | NEUTRON_CONF_DIR = '/etc/neutron/' | ||||||
|  | NEUTRON_PLUGIN_CONF = 'plugins/ml2/ml2_conf.ini' | ||||||
|  |  | ||||||
|  |  | ||||||
|  | def _get_config_files(env=None): | ||||||
|  |     if env is None: | ||||||
|  |         env = os.environ | ||||||
|  |     dirname = env.get('OS_NEUTRON_CONFIG_DIR', NEUTRON_CONF_DIR).strip() | ||||||
|  |  | ||||||
|  |     files = [s.strip() for s in | ||||||
|  |              env.get('OS_NEUTRON_CONFIG_FILES', '').split(';') if s.strip()] | ||||||
|  |  | ||||||
|  |     if not files: | ||||||
|  |         files = [NEUTRON_CONF, NEUTRON_PLUGIN_CONF] | ||||||
|  |  | ||||||
|  |     return [os.path.join(dirname, fname) for fname in files] | ||||||
|  |  | ||||||
|  |  | ||||||
| def _init_configuration(): | def _init_configuration(): | ||||||
|     # the configuration will be read into the cfg.CONF global data structure |     # the configuration will be read into the cfg.CONF global data structure | ||||||
|     config.init(sys.argv[1:]) |     conf_files = _get_config_files() | ||||||
|  |  | ||||||
|  |     config.init(sys.argv[1:], default_config_files=conf_files) | ||||||
|     config.setup_logging() |     config.setup_logging() | ||||||
|     config.set_config_defaults() |     config.set_config_defaults() | ||||||
|     if not cfg.CONF.config_file: |     if not cfg.CONF.config_file: | ||||||
|   | |||||||
| @@ -18,8 +18,10 @@ | |||||||
| # If ../neutron/__init__.py exists, add ../ to Python search path, so that | # If ../neutron/__init__.py exists, add ../ to Python search path, so that | ||||||
| # it will override what happens to be installed in /usr/(local/)lib/python... | # it will override what happens to be installed in /usr/(local/)lib/python... | ||||||
|  |  | ||||||
|  | from neutron_lib.api import attributes | ||||||
| from oslo_log import log | from oslo_log import log | ||||||
|  |  | ||||||
|  | from neutron.api import extensions | ||||||
| from neutron import manager | from neutron import manager | ||||||
| from neutron import service | from neutron import service | ||||||
|  |  | ||||||
| @@ -31,7 +33,9 @@ def eventlet_rpc_server(): | |||||||
|  |  | ||||||
|     try: |     try: | ||||||
|         manager.init() |         manager.init() | ||||||
|         rpc_workers_launcher = service.start_all_workers() |         ext_mgr = extensions.PluginAwareExtensionManager.get_instance() | ||||||
|  |         ext_mgr.extend_resources("2.0", attributes.RESOURCES) | ||||||
|  |         rpc_workers_launcher = service.start_rpc_workers() | ||||||
|     except NotImplementedError: |     except NotImplementedError: | ||||||
|         LOG.info("RPC was already started in parent process by " |         LOG.info("RPC was already started in parent process by " | ||||||
|                  "plugin.") |                  "plugin.") | ||||||
|   | |||||||
| @@ -269,9 +269,10 @@ def start_all_workers(): | |||||||
|  |  | ||||||
| def start_rpc_workers(): | def start_rpc_workers(): | ||||||
|     rpc_workers = _get_rpc_workers() |     rpc_workers = _get_rpc_workers() | ||||||
|  |  | ||||||
|     LOG.debug('using launcher for rpc, workers=%s', cfg.CONF.rpc_workers) |     LOG.debug('using launcher for rpc, workers=%s', cfg.CONF.rpc_workers) | ||||||
|     return _start_workers(rpc_workers) |     launcher = _start_workers(rpc_workers) | ||||||
|  |     registry.publish(resources.PROCESS, events.AFTER_SPAWN, None) | ||||||
|  |     return launcher | ||||||
|  |  | ||||||
|  |  | ||||||
| def start_plugins_workers(): | def start_plugins_workers(): | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Nguyen Phuong An
					Nguyen Phuong An