neutron.api.rpc.callbacks interface rework

Split rpc.callbacks interface into consumer and producer parts.

Better terms are chosen for two RPC APIs we have:
- pull when a component actively requests a new object state;
- push when a component updates anyone interested about an object
  change.

Also, for callback registration, the following terms are used:
- subscribe when a component is registered in consumer registry;
- provide when a component is registered in provider registry.

Covered the registries with some unit tests.

Lots of existing tests utilize the registries now, and need to be
isolated from other tests that mess with the managers (that are
singletons), so introduced a common qos base test class to mock the
manager with per-test instance of it).

Co-Authored-By: Ihar Hrachyshka <ihrachys@redhat.com>
Partially-Implements: blueprint quantum-qos-api
Change-Id: I130cfbc8b78da6df4405b90ea1ab47899491ba41
This commit is contained in:
Miguel Angel Ajo 2015-07-24 02:45:35 +02:00 committed by Ihar Hrachyshka
parent aa8a292d93
commit 11e22a435a
26 changed files with 691 additions and 443 deletions

View File

@ -4,7 +4,7 @@ Neutron Messaging Callback System
Neutron already has a callback system [link-to: callbacks.rst] for Neutron already has a callback system [link-to: callbacks.rst] for
in-process resource callbacks where publishers and subscribers are able in-process resource callbacks where publishers and subscribers are able
to publish, subscribe and extend resources. to publish and subscribe for resource events.
This system is different, and is intended to be used for inter-process This system is different, and is intended to be used for inter-process
callbacks, via the messaging fanout mechanisms. callbacks, via the messaging fanout mechanisms.
@ -16,12 +16,11 @@ modify existing RPC calls, or creating new RPC messages.
A few resource which can benefit of this system: A few resource which can benefit of this system:
* security groups members * QoS policies;
* security group rules, * Security Groups.
* QoS policies.
Using a remote publisher/subscriber pattern, the information about such Using a remote publisher/subscriber pattern, the information about such
resources could be published using fanout queues to all interested nodes, resources could be published using fanout messages to all interested nodes,
minimizing messaging requests from agents to server since the agents minimizing messaging requests from agents to server since the agents
get subscribed for their whole lifecycle (unless they unsubscribe). get subscribed for their whole lifecycle (unless they unsubscribe).
@ -38,8 +37,6 @@ allow object version down/up conversion. #[vo_mkcompat]_ #[vo_mkcptests]_
For the VO's versioning schema look here: #[vo_versioning]_ For the VO's versioning schema look here: #[vo_versioning]_
versioned_objects serialization/deserialization with the versioned_objects serialization/deserialization with the
obj_to_primitive(target_version=..) and primitive_to_obj() #[ov_serdes]_ obj_to_primitive(target_version=..) and primitive_to_obj() #[ov_serdes]_
methods is used internally to convert/retrieve objects before/after messaging. methods is used internally to convert/retrieve objects before/after messaging.
@ -58,42 +55,21 @@ Considering rolling upgrades, there are several scenarios to look at:
to deserialize the object, in this case (PLEASE DISCUSS), we can think of two to deserialize the object, in this case (PLEASE DISCUSS), we can think of two
strategies: strategies:
a) During upgrades, we pin neutron-server to a compatible version for resource
fanout updates, and server sends both the old, and the newer version to
different topic, queues. Old agents receive the updates on the old version
topic, new agents receive updates on the new version topic.
When the whole system upgraded, we un-pin the compatible version fanout.
A variant of this could be using a single fanout queue, and sending the The strategy for upgrades will be:
pinned version of the object to all. Newer agents can deserialize to the During upgrades, we pin neutron-server to a compatible version for resource
latest version and upgrade any fields internally. Again at the end, we fanout updates, and the server sends both the old, and the newer version.
unpin the version and restart the service. The new agents process updates, taking the newer version of the resource
fanout updates. When the whole system upgraded, we un-pin the compatible
b) The subscriber will rpc call the publisher to start publishing also a downgraded version fanout.
version of the object on every update on a separate queue. The complication
of this version, is the need to ignore new version objects as long as we keep
receiving the downgraded ones, and otherwise resend the request to send the
downgraded objects after a certain timeout (thinking of the case where the
request for downgraded queue is done, but the publisher restarted).
This approach is more complicated to implement, but more automated from the
administrator point of view. We may want to look into it as a second step
from a
c) The subscriber will send a registry.get_info for the latest specific version
he knows off. This can have scalability issues during upgrade as any outdated
agent will require a flow of two messages (request, and response). This is
indeed very bad at scale if you have hundreds or thousands of agents.
Option a seems like a reasonable strategy, similar to what nova does now with
versioned objects.
Serialized versioned objects look like:: Serialized versioned objects look like::
{'versioned_object.version': '1.0', {'versioned_object.version': '1.0',
'versioned_object.name': 'QoSProfile', 'versioned_object.name': 'QoSPolicy',
'versioned_object.data': {'rules': [ 'versioned_object.data': {'rules': [
{'versioned_object.version': '1.0', {'versioned_object.version': '1.0',
'versioned_object.name': 'QoSRule', 'versioned_object.name': 'QoSBandwidthLimitRule',
'versioned_object.data': {'name': u'a'}, 'versioned_object.data': {'name': u'a'},
'versioned_object.namespace': 'versionedobjects'} 'versioned_object.namespace': 'versionedobjects'}
], ],
@ -101,19 +77,18 @@ Serialized versioned objects look like::
'name': u'aaa'}, 'name': u'aaa'},
'versioned_object.namespace': 'versionedobjects'} 'versioned_object.namespace': 'versionedobjects'}
Topic names for the fanout queues Topic names for every resource type RPC endpoint
================================= ================================================
if we adopted option a: neutron-vo-<resource_class_name>-<version>
neutron-<resouce_type>_<resource_id>-<vo_version>
[neutron-<resouce_type>_<resource_id>-<vo_version_compat>]
if we adopted option b for rolling upgrades: In the future, we may want to get oslo messaging to support subscribing
neutron-<resource_type>-<resource_id> topics dynamically, then we may want to use:
neutron-<resource_type>-<resource_id>-<vo_version>
for option c, just: neutron-vo-<resource_class_name>-<resource_id>-<version> instead,
neutron-<resource_type>-<resource_id>
or something equivalent which would allow fine granularity for the receivers
to only get interesting information to them.
Subscribing to resources Subscribing to resources
======================== ========================
@ -123,103 +98,86 @@ has an associated security group, and QoS policy.
The agent code processing port updates may look like:: The agent code processing port updates may look like::
from neutron.rpc_resources import events from neutron.api.rpc.callbacks.consumer import registry
from neutron.rpc_resources import resources from neutron.api.rpc.callbacks import events
from neutron.rpc_resources import registry from neutron.api.rpc.callbacks import resources
def process_resource_updates(resource_type, resource_id, resource_list, action_type): def process_resource_updates(resource_type, resource, event_type):
# send to the right handler which will update any control plane # send to the right handler which will update any control plane
# details related to the updated resource... # details related to the updated resource...
def port_update(...): def subscribe_resources():
registry.subscribe(process_resource_updates, resources.SEC_GROUP)
registry.subscribe(process_resource_updates, resources.QOS_POLICY)
def port_update(port):
# here we extract sg_id and qos_policy_id from port.. # here we extract sg_id and qos_policy_id from port..
registry.subscribe(resources.SG_RULES, sg_id, sec_group = registry.pull(resources.SEC_GROUP, sg_id)
callback=process_resource_updates) qos_policy = registry.pull(resources.QOS_POLICY, qos_policy_id)
sg_rules = registry.get_info(resources.SG_RULES, sg_id)
registry.subscribe(resources.SG_MEMBERS, sg_id,
callback=process_resource_updates)
sg_members = registry.get_info(resources.SG_MEMBERS, sg_id)
registry.subscribe(resources.QOS_RULES, qos_policy_id,
callback=process_resource_updates)
qos_rules = registry.get_info(resources.QOS_RULES, qos_policy_id,
callback=process_resource_updates)
cleanup_subscriptions()
def cleanup_subscriptions() The relevant function is:
sg_ids = determine_unreferenced_sg_ids()
qos_policy_id = determine_unreferenced_qos_policy_ids()
registry.unsubscribe_info(resource.SG_RULES, sg_ids)
registry.unsubscribe_info(resource.SG_MEMBERS, sg_ids)
registry.unsubscribe_info(resource.QOS_RULES, qos_policy_id)
Another unsubscription strategy could be to lazily unsubscribe resources when * subscribe(callback, resource_type): subscribes callback to a resource type.
we receive updates for them, and we discover that they are not needed anymore.
Deleted resources are automatically unsubscribed as we receive the delete event.
NOTE(irenab): this could be extended to core resources like ports, making use
of the standard neutron in-process callbacks at server side and propagating
AFTER_UPDATE events, for example, but we may need to wait until those callbacks
are used with proper versioned objects.
Unsubscribing to resources The callback function will receive the following arguments:
==========================
There are a few options to unsubscribe registered callbacks: * resource_type: the type of resource which is receiving the update.
* resource: resource of supported object
* event_type: will be one of CREATED, UPDATED, or DELETED, see
neutron.api.rpc.callbacks.events for details.
* unsubscribe_resource_id(): it selectively unsubscribes an specific With the underlaying oslo_messaging support for dynamic topics on the receiver
resource type + id. we cannot implement a per "resource type + resource id" topic, rabbitmq seems
* unsubscribe_resource_type(): it unsubscribes from an specific resource type, to handle 10000's of topics without suffering, but creating 100's of
any ID. oslo_messaging receivers on different topics seems to crash.
* unsubscribe_all(): it unsubscribes all subscribed resources and ids.
We may want to look into that later, to avoid agents receiving resource updates
which are uninteresting to them.
Unsubscribing from resources
============================
To unsubscribe registered callbacks:
* unsubscribe(callback, resource_type): unsubscribe from specific resource type.
* unsubscribe_all(): unsubscribe from all resources.
Sending resource updates Sending resource events
======================== =======================
On the server side, resource updates could come from anywhere, a service plugin, On the server side, resource updates could come from anywhere, a service plugin,
an extension, anything that updates the resource and that it's of any interest an extension, anything that updates, creates, or destroys the resource and that
to the agents. is of any interest to subscribed agents.
The server/publisher side may look like:: The server/publisher side may look like::
from neutron.rpc_resources import events from neutron.api.rpc.callbacks.producer import registry
from neutron.rpc_resources import resources from neutron.api.rpc.callbacks import events
from neutron.rpc_resources import registry as rpc_registry
def add_qos_x_rule(...): def create_qos_policy(...):
policy = fetch_policy(...)
update_the_db(...) update_the_db(...)
send_rpc_updates_on_qos_policy(qos_policy_id) registry.push(policy, events.CREATED)
def del_qos_x_rule(...): def update_qos_policy(...):
policy = fetch_policy(...)
update_the_db(...) update_the_db(...)
send_rpc_deletion_of_qos_policy(qos_policy_id) registry.push(policy, events.UPDATED)
def send_rpc_updates_on_qos_policy(qos_policy_id): def delete_qos_policy(...):
rules = get_qos_policy_rules_versioned_object(qos_policy_id) policy = fetch_policy(...)
rpc_registry.notify(resources.QOS_RULES, qos_policy_id, rules, events.UPDATE) update_the_db(...)
registry.push(policy, events.DELETED)
def send_rpc_deletion_of_qos_policy(qos_policy_id):
rpc_registry.notify(resources.QOS_RULES, qos_policy_id, None, events.DELETE)
# This part is added for the registry mechanism, to be able to request
# older versions of the notified objects if any oudated agent requires
# them.
def retrieve_older_version_callback(qos_policy_id, version):
return get_qos_policy_rules_versioned_object(qos_policy_id, version)
rpc_registry.register_retrieve_callback(resource.QOS_RULES,
retrieve_older_version_callback)
References References
========== ==========

View File

@ -76,7 +76,7 @@ class QosAgentExtension(agent_extension.AgentCoreResourceExtension):
""" """
super(QosAgentExtension, self).initialize() super(QosAgentExtension, self).initialize()
self.resource_rpc = resources_rpc.ResourcesServerRpcApi() self.resource_rpc = resources_rpc.ResourcesPullRpcApi()
self.qos_driver = manager.NeutronManager.load_class_for_provider( self.qos_driver = manager.NeutronManager.load_class_for_provider(
'neutron.qos.agent_drivers', cfg.CONF.qos.agent_driver)() 'neutron.qos.agent_drivers', cfg.CONF.qos.agent_driver)()
self.qos_driver.initialize() self.qos_driver.initialize()
@ -111,8 +111,8 @@ class QosAgentExtension(agent_extension.AgentCoreResourceExtension):
# 1. to add new api for subscribe # 1. to add new api for subscribe
# registry.subscribe(self._process_policy_updates, # registry.subscribe(self._process_policy_updates,
# resources.QOS_POLICY, qos_policy_id) # resources.QOS_POLICY, qos_policy_id)
# 2. combine get_info rpc to also subscribe to the resource # 2. combine pull rpc to also subscribe to the resource
qos_policy = self.resource_rpc.get_info( qos_policy = self.resource_rpc.pull(
context, context,
resources.QOS_POLICY, resources.QOS_POLICY,
qos_policy_id) qos_policy_id)

View File

@ -0,0 +1,44 @@
# 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.
from oslo_log import log as logging
from neutron.api.rpc.callbacks import resource_manager
LOG = logging.getLogger(__name__)
#TODO(ajo): consider adding locking to _get_manager, it's
# safe for eventlet, but not for normal threading.
def _get_manager():
return resource_manager.ConsumerResourceCallbacksManager()
def subscribe(callback, resource_type):
_get_manager().register(callback, resource_type)
def unsubscribe(callback, resource_type):
_get_manager().unregister(callback, resource_type)
def push(resource_type, resource, event_type):
"""Push resource events into all registered callbacks for the type."""
callbacks = _get_manager().get_callbacks(resource_type)
for callback in callbacks:
callback(resource_type, resource, event_type)
def clear():
_get_manager().clear()

View File

@ -10,10 +10,12 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
CREATED = 'created'
UPDATED = 'updated' UPDATED = 'updated'
DELETED = 'deleted' DELETED = 'deleted'
VALID = ( VALID = (
CREATED,
UPDATED, UPDATED,
DELETED DELETED
) )

View File

@ -0,0 +1,25 @@
# 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.
from neutron.common import exceptions
class CallbackWrongResourceType(exceptions.NeutronException):
message = _('Callback for %(resource_type)s returned wrong resource type')
class CallbackNotFound(exceptions.NeutronException):
message = _('Callback for %(resource_type)s not found')
class CallbacksMaxLimitReached(exceptions.NeutronException):
message = _("Cannot add multiple callbacks for %(resource_type)s")

View File

@ -0,0 +1,62 @@
# 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.
from oslo_log import log as logging
from neutron.api.rpc.callbacks import exceptions
from neutron.api.rpc.callbacks import resource_manager
from neutron.objects import base
LOG = logging.getLogger(__name__)
# TODO(ajo): consider adding locking: it's safe for eventlet but not
# for other types of threading.
def _get_manager():
return resource_manager.ProducerResourceCallbacksManager()
def provide(callback, resource_type):
"""Register a callback as a producer for the resource type.
This callback will be used to produce resources of corresponding type for
interested parties.
"""
_get_manager().register(callback, resource_type)
def unprovide(callback, resource_type):
"""Unregister a callback for corresponding resource type."""
_get_manager().unregister(callback, resource_type)
def clear():
"""Clear all callbacks."""
_get_manager().clear()
def pull(resource_type, resource_id, **kwargs):
"""Get resource object that corresponds to resource id.
The function will return an object that is provided by resource producer.
:returns: NeutronObject
"""
callback = _get_manager().get_callback(resource_type)
obj = callback(resource_type, resource_id, **kwargs)
if obj:
if (not isinstance(obj, base.NeutronObject) or
resource_type != obj.obj_name()):
raise exceptions.CallbackWrongResourceType(
resource_type=resource_type)
return obj

View File

@ -1,87 +0,0 @@
# 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.
from neutron.api.rpc.callbacks import resource_manager
from neutron.api.rpc.callbacks import resources
from neutron.common import exceptions
# TODO(ajo): consider adding locking
CALLBACK_MANAGER = None
def _get_resources_callback_manager():
global CALLBACK_MANAGER
if CALLBACK_MANAGER is None:
CALLBACK_MANAGER = resource_manager.ResourcesCallbacksManager()
return CALLBACK_MANAGER
class CallbackReturnedWrongObjectType(exceptions.NeutronException):
message = _('Callback for %(resource_type)s returned wrong object type')
class CallbackNotFound(exceptions.NeutronException):
message = _('Callback for %(resource_type)s not found')
#resource implementation callback registration functions
def get_info(resource_type, resource_id, **kwargs):
"""Get information about resource type with resource id.
The function will check the providers for a specific remotable
resource and get the resource.
:returns: NeutronObject
"""
callback = _get_resources_callback_manager().get_callback(resource_type)
if not callback:
raise CallbackNotFound(resource_type=resource_type)
obj = callback(resource_type, resource_id, **kwargs)
if obj:
expected_cls = resources.get_resource_cls(resource_type)
if not isinstance(obj, expected_cls):
raise CallbackReturnedWrongObjectType(
resource_type=resource_type)
return obj
def register_provider(callback, resource_type):
_get_resources_callback_manager().register(callback, resource_type)
# resource RPC callback for pub/sub
#Agent side
def subscribe(callback, resource_type, resource_id):
#TODO(QoS): we have to finish the real update notifications
raise NotImplementedError("we should finish update notifications")
def unsubscribe(callback, resource_type, resource_id):
#TODO(QoS): we have to finish the real update notifications
raise NotImplementedError("we should finish update notifications")
def unsubscribe_all():
#TODO(QoS): we have to finish the real update notifications
raise NotImplementedError("we should finish update notifications")
#Server side
def notify(resource_type, event, obj):
#TODO(QoS): we have to finish the real update notifications
raise NotImplementedError("we should finish update notifications")
def clear():
_get_resources_callback_manager().clear()

View File

@ -10,58 +10,130 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import abc
import collections import collections
from oslo_log import log as logging from oslo_log import log as logging
import six
from neutron.api.rpc.callbacks import exceptions as rpc_exc
from neutron.api.rpc.callbacks import resources from neutron.api.rpc.callbacks import resources
from neutron.callbacks import exceptions from neutron.callbacks import exceptions
LOG = logging.getLogger(__name__) LOG = logging.getLogger(__name__)
# TODO(QoS): split the registry/resources_rpc modules into two separate things:
# one for pull and one for push APIs
class ResourcesCallbacksManager(object):
def _validate_resource_type(resource_type):
if not resources.is_valid_resource_type(resource_type):
raise exceptions.Invalid(element='resource', value=resource_type)
@six.add_metaclass(abc.ABCMeta)
class ResourceCallbacksManager(object):
"""A callback system that allows information providers in a loose manner. """A callback system that allows information providers in a loose manner.
""" """
def __init__(self): # This hook is to allow tests to get new objects for the class
self.clear() _singleton = True
def __new__(cls, *args, **kwargs):
if not cls._singleton:
return super(ResourceCallbacksManager, cls).__new__(cls)
if not hasattr(cls, '_instance'):
cls._instance = super(ResourceCallbacksManager, cls).__new__(cls)
return cls._instance
@abc.abstractmethod
def _add_callback(self, callback, resource_type):
pass
@abc.abstractmethod
def _delete_callback(self, callback, resource_type):
pass
def register(self, callback, resource_type): def register(self, callback, resource_type):
"""Register a callback for a resource type. """Register a callback for a resource type.
Only one callback can be registered for a resource type.
:param callback: the callback. It must raise or return NeutronObject. :param callback: the callback. It must raise or return NeutronObject.
:param resource_type: must be a valid resource type. :param resource_type: must be a valid resource type.
""" """
LOG.debug("register: %(callback)s %(resource_type)s", LOG.debug("Registering callback for %s", resource_type)
{'callback': callback, 'resource_type': resource_type}) _validate_resource_type(resource_type)
if not resources.is_valid_resource_type(resource_type): self._add_callback(callback, resource_type)
raise exceptions.Invalid(element='resource', value=resource_type)
self._callbacks[resource_type] = callback def unregister(self, callback, resource_type):
def unregister(self, resource_type):
"""Unregister callback from the registry. """Unregister callback from the registry.
:param resource: must be a valid resource type. :param callback: the callback.
:param resource_type: must be a valid resource type.
""" """
LOG.debug("Unregister: %s", resource_type) LOG.debug("Unregistering callback for %s", resource_type)
if not resources.is_valid_resource_type(resource_type): _validate_resource_type(resource_type)
raise exceptions.Invalid(element='resource', value=resource_type) self._delete_callback(callback, resource_type)
self._callbacks[resource_type] = None
@abc.abstractmethod
def clear(self): def clear(self):
"""Brings the manager to a clean state.""" """Brings the manager to a clean state."""
self._callbacks = collections.defaultdict(dict)
def get_subscribed_types(self):
return list(self._callbacks.keys())
class ProducerResourceCallbacksManager(ResourceCallbacksManager):
_callbacks = dict()
def _add_callback(self, callback, resource_type):
if resource_type in self._callbacks:
raise rpc_exc.CallbacksMaxLimitReached(resource_type=resource_type)
self._callbacks[resource_type] = callback
def _delete_callback(self, callback, resource_type):
try:
del self._callbacks[resource_type]
except KeyError:
raise rpc_exc.CallbackNotFound(resource_type=resource_type)
def clear(self):
self._callbacks = dict()
def get_callback(self, resource_type): def get_callback(self, resource_type):
_validate_resource_type(resource_type)
try:
return self._callbacks[resource_type]
except KeyError:
raise rpc_exc.CallbackNotFound(resource_type=resource_type)
class ConsumerResourceCallbacksManager(ResourceCallbacksManager):
_callbacks = collections.defaultdict(set)
def _add_callback(self, callback, resource_type):
self._callbacks[resource_type].add(callback)
def _delete_callback(self, callback, resource_type):
try:
self._callbacks[resource_type].remove(callback)
if not self._callbacks[resource_type]:
del self._callbacks[resource_type]
except KeyError:
raise rpc_exc.CallbackNotFound(resource_type=resource_type)
def clear(self):
self._callbacks = collections.defaultdict(set)
def get_callbacks(self, resource_type):
"""Return the callback if found, None otherwise. """Return the callback if found, None otherwise.
:param resource_type: must be a valid resource type. :param resource_type: must be a valid resource type.
""" """
if not resources.is_valid_resource_type(resource_type): _validate_resource_type(resource_type)
raise exceptions.Invalid(element='resource', value=resource_type) callbacks = self._callbacks[resource_type]
if not callbacks:
return self._callbacks[resource_type] raise rpc_exc.CallbackNotFound(resource_type=resource_type)
return callbacks

View File

@ -17,7 +17,7 @@ from oslo_log import helpers as log_helpers
from oslo_log import log as logging from oslo_log import log as logging
import oslo_messaging import oslo_messaging
from neutron.api.rpc.callbacks import registry from neutron.api.rpc.callbacks.producer import registry
from neutron.api.rpc.callbacks import resources from neutron.api.rpc.callbacks import resources
from neutron.common import constants from neutron.common import constants
from neutron.common import exceptions from neutron.common import exceptions
@ -46,14 +46,20 @@ def _validate_resource_type(resource_type):
raise InvalidResourceTypeClass(resource_type=resource_type) raise InvalidResourceTypeClass(resource_type=resource_type)
class ResourcesServerRpcApi(object): class ResourcesPullRpcApi(object):
"""Agent-side RPC (stub) for agent-to-plugin interaction. """Agent-side RPC (stub) for agent-to-plugin interaction.
This class implements the client side of an rpc interface. The server side This class implements the client side of an rpc interface. The server side
can be found below: ResourcesServerRpcCallback. For more information on can be found below: ResourcesPullRpcCallback. For more information on
this RPC interface, see doc/source/devref/rpc_callbacks.rst. this RPC interface, see doc/source/devref/rpc_callbacks.rst.
""" """
def __new__(cls):
# make it a singleton
if not hasattr(cls, '_instance'):
cls._instance = super(ResourcesPullRpcApi, cls).__new__(cls)
return cls._instance
def __init__(self): def __init__(self):
target = oslo_messaging.Target( target = oslo_messaging.Target(
topic=topics.PLUGIN, version='1.0', topic=topics.PLUGIN, version='1.0',
@ -61,7 +67,7 @@ class ResourcesServerRpcApi(object):
self.client = n_rpc.get_client(target) self.client = n_rpc.get_client(target)
@log_helpers.log_method_call @log_helpers.log_method_call
def get_info(self, context, resource_type, resource_id): def pull(self, context, resource_type, resource_id):
_validate_resource_type(resource_type) _validate_resource_type(resource_type)
# we've already validated the resource type, so we are pretty sure the # we've already validated the resource type, so we are pretty sure the
@ -69,7 +75,7 @@ class ResourcesServerRpcApi(object):
resource_type_cls = resources.get_resource_cls(resource_type) resource_type_cls = resources.get_resource_cls(resource_type)
cctxt = self.client.prepare() cctxt = self.client.prepare()
primitive = cctxt.call(context, 'get_info', primitive = cctxt.call(context, 'pull',
resource_type=resource_type, resource_type=resource_type,
version=resource_type_cls.VERSION, resource_id=resource_id) version=resource_type_cls.VERSION, resource_id=resource_id)
@ -82,11 +88,11 @@ class ResourcesServerRpcApi(object):
return obj return obj
class ResourcesServerRpcCallback(object): class ResourcesPullRpcCallback(object):
"""Plugin-side RPC (implementation) for agent-to-plugin interaction. """Plugin-side RPC (implementation) for agent-to-plugin interaction.
This class implements the server side of an rpc interface. The client side This class implements the server side of an rpc interface. The client side
can be found above: ResourcesServerRpcApi. For more information on can be found above: ResourcesPullRpcApi. For more information on
this RPC interface, see doc/source/devref/rpc_callbacks.rst. this RPC interface, see doc/source/devref/rpc_callbacks.rst.
""" """
@ -96,14 +102,10 @@ class ResourcesServerRpcCallback(object):
target = oslo_messaging.Target( target = oslo_messaging.Target(
version='1.0', namespace=constants.RPC_NAMESPACE_RESOURCES) version='1.0', namespace=constants.RPC_NAMESPACE_RESOURCES)
def get_info(self, context, resource_type, version, resource_id): def pull(self, context, resource_type, version, resource_id):
_validate_resource_type(resource_type) _validate_resource_type(resource_type)
obj = registry.get_info( obj = registry.pull(resource_type, resource_id, context=context)
resource_type,
resource_id,
context=context)
if obj: if obj:
# don't request a backport for the latest known version # don't request a backport for the latest known version
if version == obj.VERSION: if version == obj.VERSION:

View File

@ -164,7 +164,7 @@ class Ml2Plugin(db_base_plugin_v2.NeutronDbPluginV2,
dhcp_rpc.DhcpRpcCallback(), dhcp_rpc.DhcpRpcCallback(),
agents_db.AgentExtRpcCallback(), agents_db.AgentExtRpcCallback(),
metadata_rpc.MetadataRpcCallback(), metadata_rpc.MetadataRpcCallback(),
resources_rpc.ResourcesServerRpcCallback() resources_rpc.ResourcesPullRpcCallback()
] ]
def _setup_dhcp(self): def _setup_dhcp(self):

View File

@ -12,8 +12,7 @@
from oslo_log import log as logging from oslo_log import log as logging
from neutron.api.rpc.callbacks import events from neutron.api.rpc.callbacks.producer import registry
from neutron.api.rpc.callbacks import registry as rpc_registry
from neutron.api.rpc.callbacks import resources from neutron.api.rpc.callbacks import resources
from neutron.i18n import _LW from neutron.i18n import _LW
from neutron.objects.qos import policy as policy_object from neutron.objects.qos import policy as policy_object
@ -41,9 +40,7 @@ class RpcQosServiceNotificationDriver(
"""RPC message queue service notification driver for QoS.""" """RPC message queue service notification driver for QoS."""
def __init__(self): def __init__(self):
rpc_registry.register_provider( registry.provide(_get_qos_policy_cb, resources.QOS_POLICY)
_get_qos_policy_cb,
resources.QOS_POLICY)
def get_description(self): def get_description(self):
return "Message queue updates" return "Message queue updates"
@ -53,19 +50,9 @@ class RpcQosServiceNotificationDriver(
pass pass
def update_policy(self, policy): def update_policy(self, policy):
# TODO(QoS): this is temporary until we get notify() implemented # TODO(QoS): implement notification
try:
rpc_registry.notify(resources.QOS_POLICY,
events.UPDATED,
policy)
except NotImplementedError:
pass pass
def delete_policy(self, policy): def delete_policy(self, policy):
# TODO(QoS): this is temporary until we get notify() implemented # TODO(QoS): implement notification
try:
rpc_registry.notify(resources.QOS_POLICY,
events.DELETED,
policy)
except NotImplementedError:
pass pass

View File

@ -60,8 +60,8 @@ class QoSPlugin(qos.QoSPluginBase):
def delete_policy(self, context, policy_id): def delete_policy(self, context, policy_id):
policy = policy_object.QosPolicy(context) policy = policy_object.QosPolicy(context)
policy.id = policy_id policy.id = policy_id
self.notification_driver_manager.delete_policy(policy)
policy.delete() policy.delete()
self.notification_driver_manager.delete_policy(policy)
def _get_policy_obj(self, context, policy_id): def _get_policy_obj(self, context, policy_id):
obj = policy_object.QosPolicy.get_by_id(context, policy_id) obj = policy_object.QosPolicy.get_by_id(context, policy_id)

View File

@ -23,7 +23,7 @@ from neutron.tests import base
# This is a minimalistic mock of rules to be passed/checked around # This is a minimalistic mock of rules to be passed/checked around
# which should be exteneded as needed to make real rules # which should be exteneded as needed to make real rules
TEST_GET_INFO_RULES = ['rule1', 'rule2'] TEST_GET_RESOURCE_RULES = ['rule1', 'rule2']
class QosAgentExtensionTestCase(base.BaseTestCase): class QosAgentExtensionTestCase(base.BaseTestCase):
@ -40,11 +40,10 @@ class QosAgentExtensionTestCase(base.BaseTestCase):
).start() ).start()
self.qos_ext.initialize() self.qos_ext.initialize()
self._create_fake_resource_rpc()
def _create_fake_resource_rpc(self): self.pull_mock = mock.patch.object(
self.get_info_mock = mock.Mock(return_value=TEST_GET_INFO_RULES) self.qos_ext.resource_rpc, 'pull',
self.qos_ext.resource_rpc.get_info = self.get_info_mock return_value=TEST_GET_RESOURCE_RULES).start()
def _create_test_port_dict(self): def _create_test_port_dict(self):
return {'port_id': uuidutils.generate_uuid(), return {'port_id': uuidutils.generate_uuid(),
@ -65,7 +64,7 @@ class QosAgentExtensionTestCase(base.BaseTestCase):
# we make sure the underlaying qos driver is called with the # we make sure the underlaying qos driver is called with the
# right parameters # right parameters
self.qos_ext.qos_driver.create.assert_called_once_with( self.qos_ext.qos_driver.create.assert_called_once_with(
port, TEST_GET_INFO_RULES) port, TEST_GET_RESOURCE_RULES)
self.assertEqual(port, self.assertEqual(port,
self.qos_ext.qos_policy_ports[qos_policy_id][port_id]) self.qos_ext.qos_policy_ports[qos_policy_id][port_id])
self.assertTrue(port_id in self.qos_ext.known_ports) self.assertTrue(port_id in self.qos_ext.known_ports)
@ -81,10 +80,10 @@ class QosAgentExtensionTestCase(base.BaseTestCase):
def test_handle_known_port_change_policy_id(self): def test_handle_known_port_change_policy_id(self):
port = self._create_test_port_dict() port = self._create_test_port_dict()
self.qos_ext.handle_port(self.context, port) self.qos_ext.handle_port(self.context, port)
self.qos_ext.resource_rpc.get_info.reset_mock() self.qos_ext.resource_rpc.pull.reset_mock()
port['qos_policy_id'] = uuidutils.generate_uuid() port['qos_policy_id'] = uuidutils.generate_uuid()
self.qos_ext.handle_port(self.context, port) self.qos_ext.handle_port(self.context, port)
self.get_info_mock.assert_called_once_with( self.pull_mock.assert_called_once_with(
self.context, resources.QOS_POLICY, self.context, resources.QOS_POLICY,
port['qos_policy_id']) port['qos_policy_id'])
#TODO(QoS): handle qos_driver.update call check when #TODO(QoS): handle qos_driver.update call check when

View File

@ -0,0 +1,56 @@
# 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 mock
from neutron.api.rpc.callbacks.consumer import registry
from neutron.tests import base
class ConsumerRegistryTestCase(base.BaseTestCase):
def setUp(self):
super(ConsumerRegistryTestCase, self).setUp()
def test__get_manager_is_singleton(self):
self.assertIs(registry._get_manager(), registry._get_manager())
@mock.patch.object(registry, '_get_manager')
def test_subscribe(self, manager_mock):
callback = lambda: None
registry.subscribe(callback, 'TYPE')
manager_mock().register.assert_called_with(callback, 'TYPE')
@mock.patch.object(registry, '_get_manager')
def test_unsubscribe(self, manager_mock):
callback = lambda: None
registry.unsubscribe(callback, 'TYPE')
manager_mock().unregister.assert_called_with(callback, 'TYPE')
@mock.patch.object(registry, '_get_manager')
def test_clear(self, manager_mock):
registry.clear()
manager_mock().clear.assert_called_with()
@mock.patch.object(registry, '_get_manager')
def test_push(self, manager_mock):
resource_type_ = object()
resource_ = object()
event_type_ = object()
callback1 = mock.Mock()
callback2 = mock.Mock()
callbacks = {callback1, callback2}
manager_mock().get_callbacks.return_value = callbacks
registry.push(resource_type_, resource_, event_type_)
for callback in callbacks:
callback.assert_called_with(resource_type_, resource_, event_type_)

View File

@ -0,0 +1,81 @@
# 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.
from neutron.api.rpc.callbacks import exceptions
from neutron.api.rpc.callbacks.producer import registry
from neutron.api.rpc.callbacks import resources
from neutron.objects.qos import policy
from neutron.tests.unit.services.qos import base
class ProducerRegistryTestCase(base.BaseQosTestCase):
def test_pull_returns_callback_result(self):
policy_obj = policy.QosPolicy(context=None)
def _fake_policy_cb(*args, **kwargs):
return policy_obj
registry.provide(_fake_policy_cb, resources.QOS_POLICY)
self.assertEqual(
policy_obj,
registry.pull(resources.QOS_POLICY, 'fake_id'))
def test_pull_does_not_raise_on_none(self):
def _none_cb(*args, **kwargs):
pass
registry.provide(_none_cb, resources.QOS_POLICY)
obj = registry.pull(resources.QOS_POLICY, 'fake_id')
self.assertIsNone(obj)
def test_pull_raises_on_wrong_object_type(self):
def _wrong_type_cb(*args, **kwargs):
return object()
registry.provide(_wrong_type_cb, resources.QOS_POLICY)
self.assertRaises(
exceptions.CallbackWrongResourceType,
registry.pull, resources.QOS_POLICY, 'fake_id')
def test_pull_raises_on_callback_not_found(self):
self.assertRaises(
exceptions.CallbackNotFound,
registry.pull, resources.QOS_POLICY, 'fake_id')
def test__get_manager_is_singleton(self):
self.assertIs(registry._get_manager(), registry._get_manager())
def test_unprovide(self):
def _fake_policy_cb(*args, **kwargs):
pass
registry.provide(_fake_policy_cb, resources.QOS_POLICY)
registry.unprovide(_fake_policy_cb, resources.QOS_POLICY)
self.assertRaises(
exceptions.CallbackNotFound,
registry.pull, resources.QOS_POLICY, 'fake_id')
def test_clear_unprovides_all_producers(self):
def _fake_policy_cb(*args, **kwargs):
pass
registry.provide(_fake_policy_cb, resources.QOS_POLICY)
registry.clear()
self.assertRaises(
exceptions.CallbackNotFound,
registry.pull, resources.QOS_POLICY, 'fake_id')

View File

@ -1,63 +0,0 @@
# 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 mock
from neutron.api.rpc.callbacks import registry
from neutron.api.rpc.callbacks import resource_manager
from neutron.api.rpc.callbacks import resources
from neutron.objects.qos import policy
from neutron.tests import base
class GetInfoTestCase(base.BaseTestCase):
def setUp(self):
super(GetInfoTestCase, self).setUp()
mgr = resource_manager.ResourcesCallbacksManager()
mgr_p = mock.patch.object(
registry, '_get_resources_callback_manager', return_value=mgr)
mgr_p.start()
def test_returns_callback_result(self):
policy_obj = policy.QosPolicy(context=None)
def _fake_policy_cb(*args, **kwargs):
return policy_obj
registry.register_provider(_fake_policy_cb, resources.QOS_POLICY)
self.assertEqual(policy_obj,
registry.get_info(resources.QOS_POLICY, 'fake_id'))
def test_does_not_raise_on_none(self):
def _wrong_type_cb(*args, **kwargs):
pass
registry.register_provider(_wrong_type_cb, resources.QOS_POLICY)
obj = registry.get_info(resources.QOS_POLICY, 'fake_id')
self.assertIsNone(obj)
def test_raises_on_wrong_object_type(self):
def _wrong_type_cb(*args, **kwargs):
return object()
registry.register_provider(_wrong_type_cb, resources.QOS_POLICY)
self.assertRaises(
registry.CallbackReturnedWrongObjectType,
registry.get_info, resources.QOS_POLICY, 'fake_id')
def test_raises_on_callback_not_found(self):
self.assertRaises(
registry.CallbackNotFound,
registry.get_info, resources.QOS_POLICY, 'fake_id')

View File

@ -10,52 +10,131 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import mock
from neutron.api.rpc.callbacks import registry as rpc_registry from neutron.api.rpc.callbacks import exceptions as rpc_exc
from neutron.api.rpc.callbacks import resources from neutron.api.rpc.callbacks import resource_manager
from neutron.objects.qos import policy from neutron.callbacks import exceptions as exceptions
from neutron.objects.qos import rule from neutron.tests.unit.services.qos import base
IS_VALID_RESOURCE_TYPE = (
'neutron.api.rpc.callbacks.resources.is_valid_resource_type')
from neutron.tests import base class ResourceCallbacksManagerTestCaseMixin(object):
def test_register_fails_on_invalid_type(self):
self.assertRaises(
exceptions.Invalid,
self.mgr.register, lambda: None, 'TYPE')
@mock.patch(IS_VALID_RESOURCE_TYPE, return_value=True)
def test_clear_unregisters_all_callbacks(self, *mocks):
self.mgr.register(lambda: None, 'TYPE1')
self.mgr.register(lambda: None, 'TYPE2')
self.mgr.clear()
self.assertEqual([], self.mgr.get_subscribed_types())
def test_unregister_fails_on_invalid_type(self):
self.assertRaises(
exceptions.Invalid,
self.mgr.unregister, lambda: None, 'TYPE')
@mock.patch(IS_VALID_RESOURCE_TYPE, return_value=True)
def test_unregister_fails_on_unregistered_callback(self, *mocks):
self.assertRaises(
rpc_exc.CallbackNotFound,
self.mgr.unregister, lambda: None, 'TYPE')
@mock.patch(IS_VALID_RESOURCE_TYPE, return_value=True)
def test_unregister_unregisters_callback(self, *mocks):
callback = lambda: None
self.mgr.register(callback, 'TYPE')
self.mgr.unregister(callback, 'TYPE')
self.assertEqual([], self.mgr.get_subscribed_types())
@mock.patch(IS_VALID_RESOURCE_TYPE, return_value=True)
def test___init___does_not_reset_callbacks(self, *mocks):
callback = lambda: None
self.mgr.register(callback, 'TYPE')
resource_manager.ProducerResourceCallbacksManager()
self.assertEqual(['TYPE'], self.mgr.get_subscribed_types())
class ResourcesCallbackRequestTestCase(base.BaseTestCase): class ProducerResourceCallbacksManagerTestCase(
base.BaseQosTestCase, ResourceCallbacksManagerTestCaseMixin):
def setUp(self): def setUp(self):
super(ResourcesCallbackRequestTestCase, self).setUp() super(ProducerResourceCallbacksManagerTestCase, self).setUp()
self.resource_id = '46ebaec0-0570-43ac-82f6-60d2b03168c4' self.mgr = self.prod_mgr
self.qos_rule_id = '5f126d84-551a-4dcf-bb01-0e9c0df0c793'
def test_resource_callback_request(self): @mock.patch(IS_VALID_RESOURCE_TYPE, return_value=True)
def test_register_registers_callback(self, *mocks):
callback = lambda: None
self.mgr.register(callback, 'TYPE')
self.assertEqual(callback, self.mgr.get_callback('TYPE'))
def _get_qos_policy_cb(resource, policy_id, **kwargs): @mock.patch(IS_VALID_RESOURCE_TYPE, return_value=True)
context = kwargs.get('context') def test_register_fails_on_multiple_calls(self, *mocks):
qos_policy = policy.QosPolicy(context, self.mgr.register(lambda: None, 'TYPE')
tenant_id="8d4c70a21fed4aeba121a1a429ba0d04", self.assertRaises(
id="46ebaec0-0570-43ac-82f6-60d2b03168c4", rpc_exc.CallbacksMaxLimitReached,
name="10Mbit", self.mgr.register, lambda: None, 'TYPE')
description="This policy limits the ports to 10Mbit max.",
shared=False,
rules=[
rule.QosBandwidthLimitRule(context,
id="5f126d84-551a-4dcf-bb01-0e9c0df0c793",
max_kbps=10000,
max_burst_kbps=0)
]
)
qos_policy.obj_reset_changes()
return qos_policy
rpc_registry.register_provider( def test_get_callback_fails_on_invalid_type(self):
_get_qos_policy_cb, self.assertRaises(
resources.QOS_POLICY) exceptions.Invalid,
self.mgr.get_callback, 'TYPE')
self.ctx = None @mock.patch(IS_VALID_RESOURCE_TYPE, return_value=True)
kwargs = {'context': self.ctx} def test_get_callback_fails_on_unregistered_callback(
self, *mocks):
self.assertRaises(
rpc_exc.CallbackNotFound,
self.mgr.get_callback, 'TYPE')
qos_policy = rpc_registry.get_info( @mock.patch(IS_VALID_RESOURCE_TYPE, return_value=True)
resources.QOS_POLICY, def test_get_callback_returns_proper_callback(self, *mocks):
self.resource_id, callback1 = lambda: None
**kwargs) callback2 = lambda: None
self.assertEqual(self.resource_id, qos_policy['id']) self.mgr.register(callback1, 'TYPE1')
self.mgr.register(callback2, 'TYPE2')
self.assertEqual(callback1, self.mgr.get_callback('TYPE1'))
self.assertEqual(callback2, self.mgr.get_callback('TYPE2'))
class ConsumerResourceCallbacksManagerTestCase(
base.BaseQosTestCase, ResourceCallbacksManagerTestCaseMixin):
def setUp(self):
super(ConsumerResourceCallbacksManagerTestCase, self).setUp()
self.mgr = self.cons_mgr
@mock.patch(IS_VALID_RESOURCE_TYPE, return_value=True)
def test_register_registers_callback(self, *mocks):
callback = lambda: None
self.mgr.register(callback, 'TYPE')
self.assertEqual({callback}, self.mgr.get_callbacks('TYPE'))
@mock.patch(IS_VALID_RESOURCE_TYPE, return_value=True)
def test_register_succeeds_on_multiple_calls(self, *mocks):
callback1 = lambda: None
callback2 = lambda: None
self.mgr.register(callback1, 'TYPE')
self.mgr.register(callback2, 'TYPE')
@mock.patch(IS_VALID_RESOURCE_TYPE, return_value=True)
def test_get_callbacks_fails_on_unregistered_callback(
self, *mocks):
self.assertRaises(
rpc_exc.CallbackNotFound,
self.mgr.get_callbacks, 'TYPE')
@mock.patch(IS_VALID_RESOURCE_TYPE, return_value=True)
def test_get_callbacks_returns_proper_callbacks(self, *mocks):
callback1 = lambda: None
callback2 = lambda: None
self.mgr.register(callback1, 'TYPE1')
self.mgr.register(callback2, 'TYPE2')
self.assertEqual(set([callback1]), self.mgr.get_callbacks('TYPE1'))
self.assertEqual(set([callback2]), self.mgr.get_callbacks('TYPE2'))

View File

@ -42,55 +42,59 @@ class ResourcesRpcBaseTestCase(base.BaseTestCase):
return policy_obj return policy_obj
class ResourcesServerRpcApiTestCase(ResourcesRpcBaseTestCase): class ResourcesPullRpcApiTestCase(ResourcesRpcBaseTestCase):
def setUp(self): def setUp(self):
super(ResourcesServerRpcApiTestCase, self).setUp() super(ResourcesPullRpcApiTestCase, self).setUp()
self.client_p = mock.patch.object(resources_rpc.n_rpc, 'get_client') self.client_p = mock.patch.object(resources_rpc.n_rpc, 'get_client')
self.client = self.client_p.start() self.client = self.client_p.start()
self.rpc = resources_rpc.ResourcesServerRpcApi() self.rpc = resources_rpc.ResourcesPullRpcApi()
self.mock_cctxt = self.rpc.client.prepare.return_value self.mock_cctxt = self.rpc.client.prepare.return_value
def test_get_info(self): def test_is_singleton(self):
self.assertEqual(id(self.rpc),
id(resources_rpc.ResourcesPullRpcApi()))
def test_pull(self):
policy_dict = self._create_test_policy_dict() policy_dict = self._create_test_policy_dict()
expected_policy_obj = self._create_test_policy(policy_dict) expected_policy_obj = self._create_test_policy(policy_dict)
qos_policy_id = policy_dict['id'] qos_policy_id = policy_dict['id']
self.mock_cctxt.call.return_value = ( self.mock_cctxt.call.return_value = (
expected_policy_obj.obj_to_primitive()) expected_policy_obj.obj_to_primitive())
get_info_result = self.rpc.get_info( pull_result = self.rpc.pull(
self.context, resources.QOS_POLICY, qos_policy_id) self.context, resources.QOS_POLICY, qos_policy_id)
self.mock_cctxt.call.assert_called_once_with( self.mock_cctxt.call.assert_called_once_with(
self.context, 'get_info', resource_type=resources.QOS_POLICY, self.context, 'pull', resource_type=resources.QOS_POLICY,
version=policy.QosPolicy.VERSION, resource_id=qos_policy_id) version=policy.QosPolicy.VERSION, resource_id=qos_policy_id)
self.assertEqual(expected_policy_obj, get_info_result) self.assertEqual(expected_policy_obj, pull_result)
def test_get_info_invalid_resource_type_cls(self): def test_pull_invalid_resource_type_cls(self):
self.assertRaises( self.assertRaises(
resources_rpc.InvalidResourceTypeClass, self.rpc.get_info, resources_rpc.InvalidResourceTypeClass, self.rpc.pull,
self.context, 'foo_type', 'foo_id') self.context, 'foo_type', 'foo_id')
def test_get_info_resource_not_found(self): def test_pull_resource_not_found(self):
policy_dict = self._create_test_policy_dict() policy_dict = self._create_test_policy_dict()
qos_policy_id = policy_dict['id'] qos_policy_id = policy_dict['id']
self.mock_cctxt.call.return_value = None self.mock_cctxt.call.return_value = None
self.assertRaises( self.assertRaises(
resources_rpc.ResourceNotFound, self.rpc.get_info, self.context, resources_rpc.ResourceNotFound, self.rpc.pull,
resources.QOS_POLICY, qos_policy_id) self.context, resources.QOS_POLICY, qos_policy_id)
class ResourcesServerRpcCallbackTestCase(ResourcesRpcBaseTestCase): class ResourcesPullRpcCallbackTestCase(ResourcesRpcBaseTestCase):
def setUp(self): def setUp(self):
super(ResourcesServerRpcCallbackTestCase, self).setUp() super(ResourcesPullRpcCallbackTestCase, self).setUp()
self.callbacks = resources_rpc.ResourcesServerRpcCallback() self.callbacks = resources_rpc.ResourcesPullRpcCallback()
def test_get_info(self): def test_pull(self):
policy_dict = self._create_test_policy_dict() policy_dict = self._create_test_policy_dict()
policy_obj = self._create_test_policy(policy_dict) policy_obj = self._create_test_policy(policy_dict)
qos_policy_id = policy_dict['id'] qos_policy_id = policy_dict['id']
with mock.patch.object(resources_rpc.registry, 'get_info', with mock.patch.object(resources_rpc.registry, 'pull',
return_value=policy_obj) as registry_mock: return_value=policy_obj) as registry_mock:
primitive = self.callbacks.get_info( primitive = self.callbacks.pull(
self.context, resource_type=resources.QOS_POLICY, self.context, resource_type=resources.QOS_POLICY,
version=policy.QosPolicy.VERSION, version=policy.QosPolicy.VERSION,
resource_id=qos_policy_id) resource_id=qos_policy_id)
@ -101,26 +105,26 @@ class ResourcesServerRpcCallbackTestCase(ResourcesRpcBaseTestCase):
self.assertEqual(policy_obj.obj_to_primitive(), primitive) self.assertEqual(policy_obj.obj_to_primitive(), primitive)
@mock.patch.object(policy.QosPolicy, 'obj_to_primitive') @mock.patch.object(policy.QosPolicy, 'obj_to_primitive')
def test_get_info_no_backport_for_latest_version(self, to_prim_mock): def test_pull_no_backport_for_latest_version(self, to_prim_mock):
policy_dict = self._create_test_policy_dict() policy_dict = self._create_test_policy_dict()
policy_obj = self._create_test_policy(policy_dict) policy_obj = self._create_test_policy(policy_dict)
qos_policy_id = policy_dict['id'] qos_policy_id = policy_dict['id']
with mock.patch.object(resources_rpc.registry, 'get_info', with mock.patch.object(resources_rpc.registry, 'pull',
return_value=policy_obj): return_value=policy_obj):
self.callbacks.get_info( self.callbacks.pull(
self.context, resource_type=resources.QOS_POLICY, self.context, resource_type=resources.QOS_POLICY,
version=policy.QosPolicy.VERSION, version=policy.QosPolicy.VERSION,
resource_id=qos_policy_id) resource_id=qos_policy_id)
to_prim_mock.assert_called_with(target_version=None) to_prim_mock.assert_called_with(target_version=None)
@mock.patch.object(policy.QosPolicy, 'obj_to_primitive') @mock.patch.object(policy.QosPolicy, 'obj_to_primitive')
def test_get_info_backports_to_older_version(self, to_prim_mock): def test_pull_backports_to_older_version(self, to_prim_mock):
policy_dict = self._create_test_policy_dict() policy_dict = self._create_test_policy_dict()
policy_obj = self._create_test_policy(policy_dict) policy_obj = self._create_test_policy(policy_dict)
qos_policy_id = policy_dict['id'] qos_policy_id = policy_dict['id']
with mock.patch.object(resources_rpc.registry, 'get_info', with mock.patch.object(resources_rpc.registry, 'pull',
return_value=policy_obj): return_value=policy_obj):
self.callbacks.get_info( self.callbacks.pull(
self.context, resource_type=resources.QOS_POLICY, self.context, resource_type=resources.QOS_POLICY,
version='0.9', # less than initial version 1.0 version='0.9', # less than initial version 1.0
resource_id=qos_policy_id) resource_id=qos_policy_id)

View File

@ -0,0 +1,38 @@
# 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 mock
from neutron.api.rpc.callbacks.consumer import registry as cons_registry
from neutron.api.rpc.callbacks.producer import registry as prod_registry
from neutron.api.rpc.callbacks import resource_manager
from neutron.tests import base
class BaseQosTestCase(base.BaseTestCase):
def setUp(self):
super(BaseQosTestCase, self).setUp()
with mock.patch.object(
resource_manager.ResourceCallbacksManager, '_singleton',
new_callable=mock.PropertyMock(return_value=False)):
self.cons_mgr = resource_manager.ConsumerResourceCallbacksManager()
self.prod_mgr = resource_manager.ProducerResourceCallbacksManager()
for mgr in (self.cons_mgr, self.prod_mgr):
mgr.clear()
mock.patch.object(
cons_registry, '_get_manager', return_value=self.cons_mgr).start()
mock.patch.object(
prod_registry, '_get_manager', return_value=self.prod_mgr).start()

View File

@ -14,12 +14,11 @@ import mock
from oslo_config import cfg from oslo_config import cfg
from neutron.api.rpc.callbacks import events from neutron.api.rpc.callbacks import events
from neutron.api.rpc.callbacks import resources
from neutron import context from neutron import context
from neutron.objects.qos import policy as policy_object from neutron.objects.qos import policy as policy_object
from neutron.services.qos.notification_drivers import manager as driver_mgr from neutron.services.qos.notification_drivers import manager as driver_mgr
from neutron.services.qos.notification_drivers import message_queue from neutron.services.qos.notification_drivers import message_queue
from neutron.tests import base from neutron.tests.unit.services.qos import base
DUMMY_DRIVER = ("neutron.tests.unit.services.qos.notification_drivers." DUMMY_DRIVER = ("neutron.tests.unit.services.qos.notification_drivers."
"dummy.DummyQosServiceNotificationDriver") "dummy.DummyQosServiceNotificationDriver")
@ -32,16 +31,12 @@ def _load_multiple_drivers():
"qos") "qos")
class TestQosDriversManager(base.BaseTestCase): class TestQosDriversManagerBase(base.BaseQosTestCase):
def setUp(self): def setUp(self):
super(TestQosDriversManager, self).setUp() super(TestQosDriversManagerBase, self).setUp()
self.config_parse() self.config_parse()
self.setup_coreplugin() self.setup_coreplugin()
self.registry_p = mock.patch(
'neutron.api.rpc.callbacks.registry.notify')
self.registry_m = self.registry_p.start()
self.driver_manager = driver_mgr.QosServiceNotificationDriverManager()
config = cfg.ConfigOpts() config = cfg.ConfigOpts()
config.register_opts(driver_mgr.QOS_PLUGIN_OPTS, "qos") config.register_opts(driver_mgr.QOS_PLUGIN_OPTS, "qos")
self.policy_data = {'policy': { self.policy_data = {'policy': {
@ -56,17 +51,20 @@ class TestQosDriversManager(base.BaseTestCase):
ctxt = None ctxt = None
self.kwargs = {'context': ctxt} self.kwargs = {'context': ctxt}
class TestQosDriversManager(TestQosDriversManagerBase):
def setUp(self):
super(TestQosDriversManager, self).setUp()
self.driver_manager = driver_mgr.QosServiceNotificationDriverManager()
def _validate_registry_params(self, event_type, policy): def _validate_registry_params(self, event_type, policy):
self.assertTrue(self.registry_m.called, policy) #TODO(QoS): actually validate the notification once implemented
self.registry_m.assert_called_with( pass
resources.QOS_POLICY,
event_type,
policy)
def test_create_policy_default_configuration(self): def test_create_policy_default_configuration(self):
#RPC driver should be loaded by default #RPC driver should be loaded by default
self.driver_manager.create_policy(self.policy) self.driver_manager.create_policy(self.policy)
self.assertFalse(self.registry_m.called)
def test_update_policy_default_configuration(self): def test_update_policy_default_configuration(self):
#RPC driver should be loaded by default #RPC driver should be loaded by default
@ -78,9 +76,11 @@ class TestQosDriversManager(base.BaseTestCase):
self.driver_manager.delete_policy(self.policy) self.driver_manager.delete_policy(self.policy)
self._validate_registry_params(events.DELETED, self.policy) self._validate_registry_params(events.DELETED, self.policy)
class TestQosDriversManagerMulti(TestQosDriversManagerBase):
def _test_multi_drivers_configuration_op(self, op): def _test_multi_drivers_configuration_op(self, op):
_load_multiple_drivers() _load_multiple_drivers()
# create a new manager with new configuration
driver_manager = driver_mgr.QosServiceNotificationDriverManager() driver_manager = driver_mgr.QosServiceNotificationDriverManager()
handler = '%s_policy' % op handler = '%s_policy' % op
with mock.patch('.'.join([DUMMY_DRIVER, handler])) as dummy_mock: with mock.patch('.'.join([DUMMY_DRIVER, handler])) as dummy_mock:

View File

@ -10,27 +10,20 @@
# License for the specific language governing permissions and limitations # License for the specific language governing permissions and limitations
# under the License. # under the License.
import mock
from neutron.api.rpc.callbacks import events from neutron.api.rpc.callbacks import events
from neutron.api.rpc.callbacks import resources
from neutron import context from neutron import context
from neutron.objects.qos import policy as policy_object from neutron.objects.qos import policy as policy_object
from neutron.objects.qos import rule as rule_object from neutron.objects.qos import rule as rule_object
from neutron.services.qos.notification_drivers import message_queue from neutron.services.qos.notification_drivers import message_queue
from neutron.tests import base from neutron.tests.unit.services.qos import base
DB_PLUGIN_KLASS = 'neutron.db.db_base_plugin_v2.NeutronDbPluginV2' DB_PLUGIN_KLASS = 'neutron.db.db_base_plugin_v2.NeutronDbPluginV2'
class TestQosRpcNotificationDriver(base.BaseTestCase): class TestQosRpcNotificationDriver(base.BaseQosTestCase):
def setUp(self): def setUp(self):
super(TestQosRpcNotificationDriver, self).setUp() super(TestQosRpcNotificationDriver, self).setUp()
registry_p = mock.patch(
'neutron.api.rpc.callbacks.registry.notify')
self.registry_m = registry_p.start()
self.driver = message_queue.RpcQosServiceNotificationDriver() self.driver = message_queue.RpcQosServiceNotificationDriver()
self.policy_data = {'policy': { self.policy_data = {'policy': {
@ -52,21 +45,18 @@ class TestQosRpcNotificationDriver(base.BaseTestCase):
context, context,
**self.rule_data['bandwidth_limit_rule']) **self.rule_data['bandwidth_limit_rule'])
def _validate_registry_params(self, event_type, policy): def _validate_push_params(self, event_type, policy):
self.assertTrue(self.registry_m.called, policy) # TODO(QoS): actually validate push works once implemented
self.registry_m.assert_called_once_with( pass
resources.QOS_POLICY,
event_type,
policy)
def test_create_policy(self): def test_create_policy(self):
self.driver.create_policy(self.policy) self.driver.create_policy(self.policy)
self.assertFalse(self.registry_m.called) self._validate_push_params(events.CREATED, self.policy)
def test_update_policy(self): def test_update_policy(self):
self.driver.update_policy(self.policy) self.driver.update_policy(self.policy)
self._validate_registry_params(events.UPDATED, self.policy) self._validate_push_params(events.UPDATED, self.policy)
def test_delete_policy(self): def test_delete_policy(self):
self.driver.delete_policy(self.policy) self.driver.delete_policy(self.policy)
self._validate_registry_params(events.DELETED, self.policy) self._validate_push_params(events.DELETED, self.policy)

View File

@ -13,8 +13,6 @@
import mock import mock
from oslo_config import cfg from oslo_config import cfg
from neutron.api.rpc.callbacks import events
from neutron.api.rpc.callbacks import resources
from neutron.common import exceptions as n_exc from neutron.common import exceptions as n_exc
from neutron import context from neutron import context
from neutron import manager from neutron import manager
@ -22,13 +20,13 @@ from neutron.objects import base as base_object
from neutron.objects.qos import policy as policy_object from neutron.objects.qos import policy as policy_object
from neutron.objects.qos import rule as rule_object from neutron.objects.qos import rule as rule_object
from neutron.plugins.common import constants from neutron.plugins.common import constants
from neutron.tests import base from neutron.tests.unit.services.qos import base
DB_PLUGIN_KLASS = 'neutron.db.db_base_plugin_v2.NeutronDbPluginV2' DB_PLUGIN_KLASS = 'neutron.db.db_base_plugin_v2.NeutronDbPluginV2'
class TestQosPlugin(base.BaseTestCase): class TestQosPlugin(base.BaseQosTestCase):
def setUp(self): def setUp(self):
super(TestQosPlugin, self).setUp() super(TestQosPlugin, self).setUp()
@ -40,15 +38,18 @@ class TestQosPlugin(base.BaseTestCase):
mock.patch('neutron.db.api.get_object').start() mock.patch('neutron.db.api.get_object').start()
mock.patch( mock.patch(
'neutron.objects.qos.policy.QosPolicy.obj_load_attr').start() 'neutron.objects.qos.policy.QosPolicy.obj_load_attr').start()
self.registry_p = mock.patch(
'neutron.api.rpc.callbacks.registry.notify')
self.registry_m = self.registry_p.start()
cfg.CONF.set_override("core_plugin", DB_PLUGIN_KLASS) cfg.CONF.set_override("core_plugin", DB_PLUGIN_KLASS)
cfg.CONF.set_override("service_plugins", ["qos"]) cfg.CONF.set_override("service_plugins", ["qos"])
mgr = manager.NeutronManager.get_instance() mgr = manager.NeutronManager.get_instance()
self.qos_plugin = mgr.get_service_plugins().get( self.qos_plugin = mgr.get_service_plugins().get(
constants.QOS) constants.QOS)
self.notif_driver_p = mock.patch.object(
self.qos_plugin, 'notification_driver_manager')
self.notif_driver_m = self.notif_driver_p.start()
self.ctxt = context.Context('fake_user', 'fake_tenant') self.ctxt = context.Context('fake_user', 'fake_tenant')
self.policy_data = { self.policy_data = {
'policy': {'id': 7777777, 'policy': {'id': 7777777,
@ -68,50 +69,48 @@ class TestQosPlugin(base.BaseTestCase):
self.rule = rule_object.QosBandwidthLimitRule( self.rule = rule_object.QosBandwidthLimitRule(
context, **self.rule_data['bandwidth_limit_rule']) context, **self.rule_data['bandwidth_limit_rule'])
def _validate_registry_params(self, event_type): def _validate_notif_driver_params(self, method_name):
self.registry_m.assert_called_once_with( method = getattr(self.notif_driver_m, method_name)
resources.QOS_POLICY, self.assertTrue(method.called)
event_type,
mock.ANY)
self.assertIsInstance( self.assertIsInstance(
self.registry_m.call_args[0][2], policy_object.QosPolicy) method.call_args[0][0], policy_object.QosPolicy)
def test_add_policy(self): def test_add_policy(self):
self.qos_plugin.create_policy(self.ctxt, self.policy_data) self.qos_plugin.create_policy(self.ctxt, self.policy_data)
self.assertFalse(self.registry_m.called) self._validate_notif_driver_params('create_policy')
def test_update_policy(self): def test_update_policy(self):
fields = base_object.get_updatable_fields( fields = base_object.get_updatable_fields(
policy_object.QosPolicy, self.policy_data['policy']) policy_object.QosPolicy, self.policy_data['policy'])
self.qos_plugin.update_policy( self.qos_plugin.update_policy(
self.ctxt, self.policy.id, {'policy': fields}) self.ctxt, self.policy.id, {'policy': fields})
self._validate_registry_params(events.UPDATED) self._validate_notif_driver_params('update_policy')
@mock.patch('neutron.db.api.get_object', return_value=None) @mock.patch('neutron.db.api.get_object', return_value=None)
def test_delete_policy(self, *mocks): def test_delete_policy(self, *mocks):
self.qos_plugin.delete_policy(self.ctxt, self.policy.id) self.qos_plugin.delete_policy(self.ctxt, self.policy.id)
self._validate_registry_params(events.DELETED) self._validate_notif_driver_params('delete_policy')
def test_create_policy_rule(self): def test_create_policy_rule(self):
with mock.patch('neutron.objects.qos.policy.QosPolicy.get_by_id', with mock.patch('neutron.objects.qos.policy.QosPolicy.get_by_id',
return_value=self.policy): return_value=self.policy):
self.qos_plugin.create_policy_bandwidth_limit_rule( self.qos_plugin.create_policy_bandwidth_limit_rule(
self.ctxt, self.policy.id, self.rule_data) self.ctxt, self.policy.id, self.rule_data)
self._validate_registry_params(events.UPDATED) self._validate_notif_driver_params('update_policy')
def test_update_policy_rule(self): def test_update_policy_rule(self):
with mock.patch('neutron.objects.qos.policy.QosPolicy.get_by_id', with mock.patch('neutron.objects.qos.policy.QosPolicy.get_by_id',
return_value=self.policy): return_value=self.policy):
self.qos_plugin.update_policy_bandwidth_limit_rule( self.qos_plugin.update_policy_bandwidth_limit_rule(
self.ctxt, self.rule.id, self.policy.id, self.rule_data) self.ctxt, self.rule.id, self.policy.id, self.rule_data)
self._validate_registry_params(events.UPDATED) self._validate_notif_driver_params('update_policy')
def test_delete_policy_rule(self): def test_delete_policy_rule(self):
with mock.patch('neutron.objects.qos.policy.QosPolicy.get_by_id', with mock.patch('neutron.objects.qos.policy.QosPolicy.get_by_id',
return_value=self.policy): return_value=self.policy):
self.qos_plugin.delete_policy_bandwidth_limit_rule( self.qos_plugin.delete_policy_bandwidth_limit_rule(
self.ctxt, self.rule.id, self.policy.id) self.ctxt, self.rule.id, self.policy.id)
self._validate_registry_params(events.UPDATED) self._validate_notif_driver_params('update_policy')
def test_get_policy_for_nonexistent_policy(self): def test_get_policy_for_nonexistent_policy(self):
with mock.patch('neutron.objects.qos.policy.QosPolicy.get_by_id', with mock.patch('neutron.objects.qos.policy.QosPolicy.get_by_id',