Refactor QoS plugin to avoid code duplication
This change refactors the rule-specific methods in qos_plugin.py in order to facilitate implementation of additional QoS rule types. Further, with the recent merge of https://review.openstack.org/#/c/292488/, which prevents primary keys in Neutron's oslo.versioned objects from being updated, the QoS plugin needed additional refactoring. This change implements a generic method in Neutron's base objects to handle object updates without updating the primary key. This method is then used in the QoS plugin. Co-Authored-By: Slawek Kaplonski <slawek@kaplonski.pl> Change-Id: I863f063a0cfbb464cedd00bddc15dd853cbb6389 Partial-Bug: #1468353
This commit is contained in:
parent
0eed94a777
commit
b3ca00f7a6
@ -52,6 +52,13 @@ repositories under the neutron tent. Below you can find a list of known
|
|||||||
incompatible changes that could or are known to trigger those breakages.
|
incompatible changes that could or are known to trigger those breakages.
|
||||||
The changes are listed in reverse chronological order (newer at the top).
|
The changes are listed in reverse chronological order (newer at the top).
|
||||||
|
|
||||||
|
* change: QoS plugin refactor
|
||||||
|
|
||||||
|
- commit: I863f063a0cfbb464cedd00bddc15dd853cbb6389
|
||||||
|
- solution: implement the new abstract methods in
|
||||||
|
neutron.extensions.qos.QoSPluginBase.
|
||||||
|
- severity: Low (some out-of-tree plugins might be affected).
|
||||||
|
|
||||||
* change: Consume ConfigurableMiddleware from oslo_middleware.
|
* change: Consume ConfigurableMiddleware from oslo_middleware.
|
||||||
|
|
||||||
- commit: If7360608f94625b7d0972267b763f3e7d7624fee
|
- commit: If7360608f94625b7d0972267b763f3e7d7624fee
|
||||||
|
@ -76,6 +76,23 @@ Service side design
|
|||||||
integrated into other plugins with ease.
|
integrated into other plugins with ease.
|
||||||
|
|
||||||
|
|
||||||
|
QoS plugin implementation guide
|
||||||
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
|
The neutron.extensions.qos.QoSPluginBase class uses method proxies for methods
|
||||||
|
relating to QoS policy rules. Each of these such methods is generic in the sense
|
||||||
|
that it is intended to handle any rule type. For example, QoSPluginBase has a
|
||||||
|
create_policy_rule method instead of both create_policy_dscp_marking_rule and
|
||||||
|
create_policy_bandwidth_limit_rule methods. The logic behind the proxies allows
|
||||||
|
a call to a plugin's create_policy_dscp_marking_rule to be handled by the
|
||||||
|
create_policy_rule method, which will receive a QosDscpMarkingRule object as an
|
||||||
|
argument in order to execute behavior specific to the DSCP marking rule type.
|
||||||
|
This approach allows new rule types to be introduced without requiring a plugin
|
||||||
|
to modify code as a result. As would be expected, any subclass of QoSPluginBase
|
||||||
|
must override the base class's abc.abstractmethod methods, even if to raise
|
||||||
|
NotImplemented.
|
||||||
|
|
||||||
|
|
||||||
Supported QoS rule types
|
Supported QoS rule types
|
||||||
~~~~~~~~~~~~~~~~~~~~~~~~
|
~~~~~~~~~~~~~~~~~~~~~~~~
|
||||||
|
|
||||||
|
@ -15,6 +15,7 @@
|
|||||||
|
|
||||||
import abc
|
import abc
|
||||||
import itertools
|
import itertools
|
||||||
|
import re
|
||||||
|
|
||||||
from neutron_lib.api import converters
|
from neutron_lib.api import converters
|
||||||
import six
|
import six
|
||||||
@ -25,6 +26,7 @@ from neutron.api.v2 import base
|
|||||||
from neutron.api.v2 import resource_helper
|
from neutron.api.v2 import resource_helper
|
||||||
from neutron.common import constants as common_constants
|
from neutron.common import constants as common_constants
|
||||||
from neutron import manager
|
from neutron import manager
|
||||||
|
from neutron.objects.qos import rule as rule_object
|
||||||
from neutron.plugins.common import constants
|
from neutron.plugins.common import constants
|
||||||
from neutron.services.qos import qos_consts
|
from neutron.services.qos import qos_consts
|
||||||
from neutron.services import service_base
|
from neutron.services import service_base
|
||||||
@ -188,6 +190,93 @@ class QoSPluginBase(service_base.ServicePluginBase):
|
|||||||
|
|
||||||
path_prefix = QOS_PREFIX
|
path_prefix = QOS_PREFIX
|
||||||
|
|
||||||
|
# The rule object type to use for each incoming rule-related request.
|
||||||
|
rule_objects = {'bandwidth_limit': rule_object.QosBandwidthLimitRule,
|
||||||
|
'dscp_marking': rule_object.QosDscpMarkingRule}
|
||||||
|
|
||||||
|
# Patterns used to call method proxies for all policy-rule-specific
|
||||||
|
# method calls (see __getattr__ docstring, below).
|
||||||
|
qos_rule_method_patterns = [
|
||||||
|
re.compile(
|
||||||
|
r"^((create|update|delete)_policy_(?P<rule_type>.*)_rule)$"),
|
||||||
|
re.compile(
|
||||||
|
r"^(get_policy_(?P<rule_type>.*)_(rules|rule))$"),
|
||||||
|
]
|
||||||
|
|
||||||
|
def __getattr__(self, attrib):
|
||||||
|
"""Implement method proxies for all policy-rule-specific requests. For
|
||||||
|
a given request type (such as to update a rule), a single method will
|
||||||
|
handle requests for all rule types. For example, the
|
||||||
|
update_policy_rule method will handle requests for both
|
||||||
|
update_policy_dscp_marking_rule and update_policy_bandwidth_limit_rule.
|
||||||
|
|
||||||
|
:param attrib: the requested method; in the normal case, this will be,
|
||||||
|
for example, "update_policy_dscp_marking_rule"
|
||||||
|
:type attrib: str
|
||||||
|
"""
|
||||||
|
# Find and call the proxy method that implements the requested one.
|
||||||
|
for pattern in self.qos_rule_method_patterns:
|
||||||
|
res = re.match(pattern, attrib)
|
||||||
|
if res:
|
||||||
|
rule_type = res.group('rule_type')
|
||||||
|
if rule_type in self.rule_objects:
|
||||||
|
# Remove the rule_type value (plus underscore) from attrib
|
||||||
|
# in order to get the proxy method name. So, for instance,
|
||||||
|
# from "delete_policy_dscp_marking_rule" we'll get
|
||||||
|
# "delete_policy_rule".
|
||||||
|
proxy_method = attrib.replace(rule_type + '_', '')
|
||||||
|
|
||||||
|
rule_obj = self.rule_objects[rule_type]
|
||||||
|
return self._call_proxy_method(proxy_method, rule_obj)
|
||||||
|
|
||||||
|
# If we got here, then either attrib matched no pattern or the
|
||||||
|
# rule_type embedded in attrib wasn't in self.rule_objects.
|
||||||
|
raise AttributeError(attrib)
|
||||||
|
|
||||||
|
def _call_proxy_method(self, method_name, rule_obj):
|
||||||
|
"""Call proxy method. We need to add the rule_obj, obtained from the
|
||||||
|
self.rule_objects dictionary, to the incoming args. The context is
|
||||||
|
passed to proxy method as first argument; the remaining args will
|
||||||
|
follow rule_obj.
|
||||||
|
|
||||||
|
Some of the incoming method calls have the policy rule name as one of
|
||||||
|
the keys in the kwargs. For instance, the incoming kwargs for the
|
||||||
|
create_policy_bandwidth_limit_rule take this form:
|
||||||
|
|
||||||
|
{ 'bandwidth_limit_rule': {
|
||||||
|
u'bandwidth_limit_rule':
|
||||||
|
{ 'max_burst_kbps': 0,
|
||||||
|
u'max_kbps': u'100',
|
||||||
|
'tenant_id': u'a8a31c9434ff431cb789c809777505ec'}
|
||||||
|
},
|
||||||
|
'policy_id': u'46985da5-9684-402e-b0d7-b7adac909c3a'
|
||||||
|
}
|
||||||
|
|
||||||
|
We need to generalize this structure for all rule types so will
|
||||||
|
(effectively) rename the rule-specific keyword (e.g., in the above, the
|
||||||
|
first occurrence of 'bandwidth_limit_rule') to be 'rule_data'.
|
||||||
|
|
||||||
|
:param method_name: the name of the method to call
|
||||||
|
:type method_name: str
|
||||||
|
:param rule_obj: the rule object, which is sent as an argument to the
|
||||||
|
proxy method
|
||||||
|
:type rule_obj: a class from the rule_object (qos.objects.rule) module
|
||||||
|
"""
|
||||||
|
def _make_call(method_name, rule_obj, *args, **kwargs):
|
||||||
|
context = args[0]
|
||||||
|
args_list = list(args[1:])
|
||||||
|
params = kwargs
|
||||||
|
rule_data_name = rule_obj.rule_type + "_rule"
|
||||||
|
if rule_data_name in params:
|
||||||
|
params['rule_data'] = params.pop(rule_data_name)
|
||||||
|
|
||||||
|
return getattr(self, method_name)(
|
||||||
|
context, rule_obj, *args_list, **params
|
||||||
|
)
|
||||||
|
|
||||||
|
return lambda *args, **kwargs: _make_call(
|
||||||
|
method_name, rule_obj, *args, **kwargs)
|
||||||
|
|
||||||
def get_plugin_description(self):
|
def get_plugin_description(self):
|
||||||
return "QoS Service Plugin for ports and networks"
|
return "QoS Service Plugin for ports and networks"
|
||||||
|
|
||||||
@ -195,13 +284,8 @@ class QoSPluginBase(service_base.ServicePluginBase):
|
|||||||
return constants.QOS
|
return constants.QOS
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def get_policy(self, context, policy_id, fields=None):
|
def get_rule_types(self, context, filters=None, fields=None, sorts=None,
|
||||||
pass
|
limit=None, marker=None, page_reverse=False):
|
||||||
|
|
||||||
@abc.abstractmethod
|
|
||||||
def get_policies(self, context, filters=None, fields=None,
|
|
||||||
sorts=None, limit=None, marker=None,
|
|
||||||
page_reverse=False):
|
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
@ -217,59 +301,34 @@ class QoSPluginBase(service_base.ServicePluginBase):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def get_policy_bandwidth_limit_rule(self, context, rule_id,
|
def get_policy(self, context, policy_id, fields=None):
|
||||||
policy_id, fields=None):
|
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def get_policy_bandwidth_limit_rules(self, context, policy_id,
|
def get_policies(self, context, filters=None, fields=None, sorts=None,
|
||||||
filters=None, fields=None,
|
limit=None, marker=None, page_reverse=False):
|
||||||
sorts=None, limit=None,
|
|
||||||
marker=None, page_reverse=False):
|
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def create_policy_bandwidth_limit_rule(self, context, policy_id,
|
def create_policy_rule(self, context, policy_id, rule_data, rule_obj):
|
||||||
bandwidth_limit_rule):
|
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def update_policy_bandwidth_limit_rule(self, context, rule_id, policy_id,
|
def update_policy_rule(self, context, rule_id, policy_id, rule_data,
|
||||||
bandwidth_limit_rule):
|
rule_obj):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def delete_policy_bandwidth_limit_rule(self, context, rule_id, policy_id):
|
def delete_policy_rule(self, context, rule_id, policy_id):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def get_policy_dscp_marking_rule(self, context, rule_id,
|
def get_policy_rule(self, context, rule_id, policy_id, rule_obj,
|
||||||
policy_id, fields=None):
|
fields=None):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
@abc.abstractmethod
|
@abc.abstractmethod
|
||||||
def get_policy_dscp_marking_rules(self, context, policy_id,
|
def get_policy_rules(self, context, policy_id, rule_obj,
|
||||||
filters=None, fields=None,
|
filters=None, fields=None, sorts=None, limit=None,
|
||||||
sorts=None, limit=None,
|
marker=None, page_reverse=False):
|
||||||
marker=None, page_reverse=False):
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abc.abstractmethod
|
|
||||||
def create_policy_dscp_marking_rule(self, context, policy_id,
|
|
||||||
dscp_marking_rule):
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abc.abstractmethod
|
|
||||||
def update_policy_dscp_marking_rule(self, context, rule_id, policy_id,
|
|
||||||
dscp_marking_rule):
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abc.abstractmethod
|
|
||||||
def delete_policy_dscp_marking_rule(self, context, rule_id, policy_id):
|
|
||||||
pass
|
|
||||||
|
|
||||||
@abc.abstractmethod
|
|
||||||
def get_rule_types(self, context, filters=None, fields=None,
|
|
||||||
sorts=None, limit=None,
|
|
||||||
marker=None, page_reverse=False):
|
|
||||||
pass
|
pass
|
||||||
|
@ -341,6 +341,24 @@ class NeutronDbObject(NeutronObject):
|
|||||||
keys[key] = getattr(self, key)
|
keys[key] = getattr(self, key)
|
||||||
return self.modify_fields_to_db(keys)
|
return self.modify_fields_to_db(keys)
|
||||||
|
|
||||||
|
def update_nonidentifying_fields(self, obj_data, reset_changes=False):
|
||||||
|
"""Updates non-identifying fields of an object.
|
||||||
|
|
||||||
|
:param obj_data: the full set of object data
|
||||||
|
:type obj_data: dict
|
||||||
|
:param reset_changes: indicates whether the object's current set of
|
||||||
|
changed fields should be cleared
|
||||||
|
:type reset_changes: boolean
|
||||||
|
|
||||||
|
:returns: None
|
||||||
|
"""
|
||||||
|
|
||||||
|
if reset_changes:
|
||||||
|
self.obj_reset_changes()
|
||||||
|
for k, v in obj_data.items():
|
||||||
|
if k not in self.primary_keys:
|
||||||
|
setattr(self, k, v)
|
||||||
|
|
||||||
def update(self):
|
def update(self):
|
||||||
updates = self._get_changed_persistent_fields()
|
updates = self._get_changed_persistent_fields()
|
||||||
updates = self._validate_changed_fields(updates)
|
updates = self._validate_changed_fields(updates)
|
||||||
|
@ -18,7 +18,6 @@ from neutron.db import api as db_api
|
|||||||
from neutron.db import db_base_plugin_common
|
from neutron.db import db_base_plugin_common
|
||||||
from neutron.extensions import qos
|
from neutron.extensions import qos
|
||||||
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_type as rule_type_object
|
from neutron.objects.qos import rule_type as rule_type_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 import qos_consts
|
from neutron.services.qos import qos_consts
|
||||||
@ -27,9 +26,8 @@ from neutron.services.qos import qos_consts
|
|||||||
class QoSPlugin(qos.QoSPluginBase):
|
class QoSPlugin(qos.QoSPluginBase):
|
||||||
"""Implementation of the Neutron QoS Service Plugin.
|
"""Implementation of the Neutron QoS Service Plugin.
|
||||||
|
|
||||||
This class implements a Quality of Service plugin that
|
This class implements a Quality of Service plugin that provides quality of
|
||||||
provides quality of service parameters over ports and
|
service parameters over ports and networks.
|
||||||
networks.
|
|
||||||
|
|
||||||
"""
|
"""
|
||||||
supported_extension_aliases = ['qos']
|
supported_extension_aliases = ['qos']
|
||||||
@ -41,29 +39,67 @@ class QoSPlugin(qos.QoSPluginBase):
|
|||||||
|
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
@db_base_plugin_common.convert_result_to_dict
|
||||||
def create_policy(self, context, policy):
|
def create_policy(self, context, policy):
|
||||||
policy = policy_object.QosPolicy(context, **policy['policy'])
|
"""Create a QoS policy.
|
||||||
policy.create()
|
|
||||||
self.notification_driver_manager.create_policy(context, policy)
|
:param context: neutron api request context
|
||||||
return policy
|
:type context: neutron.context.Context
|
||||||
|
:param policy: policy data to be applied
|
||||||
|
:type policy: dict
|
||||||
|
|
||||||
|
:returns: a QosPolicy object
|
||||||
|
"""
|
||||||
|
policy_obj = policy_object.QosPolicy(context, **policy['policy'])
|
||||||
|
policy_obj.create()
|
||||||
|
self.notification_driver_manager.create_policy(context, policy_obj)
|
||||||
|
return policy_obj
|
||||||
|
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
@db_base_plugin_common.convert_result_to_dict
|
||||||
def update_policy(self, context, policy_id, policy):
|
def update_policy(self, context, policy_id, policy):
|
||||||
obj = policy_object.QosPolicy(context, id=policy_id)
|
"""Update a QoS policy.
|
||||||
obj.obj_reset_changes()
|
|
||||||
for k, v in policy['policy'].items():
|
:param context: neutron api request context
|
||||||
if k != 'id':
|
:type context: neutron.context.Context
|
||||||
setattr(obj, k, v)
|
:param policy_id: the id of the QosPolicy to update
|
||||||
obj.update()
|
:param policy_id: str uuid
|
||||||
self.notification_driver_manager.update_policy(context, obj)
|
:param policy: new policy data to be applied
|
||||||
return obj
|
:type policy: dict
|
||||||
|
|
||||||
|
:returns: a QosPolicy object
|
||||||
|
"""
|
||||||
|
policy_data = policy['policy']
|
||||||
|
policy_obj = policy_object.QosPolicy(context, id=policy_id)
|
||||||
|
policy_obj.update_nonidentifying_fields(policy_data,
|
||||||
|
reset_changes=True)
|
||||||
|
policy_obj.update()
|
||||||
|
self.notification_driver_manager.update_policy(context, policy_obj)
|
||||||
|
return policy_obj
|
||||||
|
|
||||||
def delete_policy(self, context, policy_id):
|
def delete_policy(self, context, policy_id):
|
||||||
|
"""Delete a QoS policy.
|
||||||
|
|
||||||
|
:param context: neutron api request context
|
||||||
|
:type context: neutron.context.Context
|
||||||
|
:param policy_id: the id of the QosPolicy to delete
|
||||||
|
:type policy_id: str uuid
|
||||||
|
|
||||||
|
:returns: None
|
||||||
|
"""
|
||||||
policy = policy_object.QosPolicy(context)
|
policy = policy_object.QosPolicy(context)
|
||||||
policy.id = policy_id
|
policy.id = policy_id
|
||||||
self.notification_driver_manager.delete_policy(context, policy)
|
self.notification_driver_manager.delete_policy(context, policy)
|
||||||
policy.delete()
|
policy.delete()
|
||||||
|
|
||||||
def _get_policy_obj(self, context, policy_id):
|
def _get_policy_obj(self, context, policy_id):
|
||||||
|
"""Fetch a QoS policy.
|
||||||
|
|
||||||
|
:param context: neutron api request context
|
||||||
|
:type context: neutron.context.Context
|
||||||
|
:param policy_id: the id of the QosPolicy to fetch
|
||||||
|
:type policy_id: str uuid
|
||||||
|
|
||||||
|
:returns: a QosPolicy object
|
||||||
|
:raises: n_exc.QosPolicyNotFound
|
||||||
|
"""
|
||||||
obj = policy_object.QosPolicy.get_object(context, id=policy_id)
|
obj = policy_object.QosPolicy.get_object(context, id=policy_id)
|
||||||
if obj is None:
|
if obj is None:
|
||||||
raise n_exc.QosPolicyNotFound(policy_id=policy_id)
|
raise n_exc.QosPolicyNotFound(policy_id=policy_id)
|
||||||
@ -72,169 +108,169 @@ class QoSPlugin(qos.QoSPluginBase):
|
|||||||
@db_base_plugin_common.filter_fields
|
@db_base_plugin_common.filter_fields
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
@db_base_plugin_common.convert_result_to_dict
|
||||||
def get_policy(self, context, policy_id, fields=None):
|
def get_policy(self, context, policy_id, fields=None):
|
||||||
|
"""Get a QoS policy.
|
||||||
|
|
||||||
|
:param context: neutron api request context
|
||||||
|
:type context: neutron.context.Context
|
||||||
|
:param policy_id: the id of the QosPolicy to update
|
||||||
|
:type policy_id: str uuid
|
||||||
|
|
||||||
|
:returns: a QosPolicy object
|
||||||
|
"""
|
||||||
return self._get_policy_obj(context, policy_id)
|
return self._get_policy_obj(context, policy_id)
|
||||||
|
|
||||||
@db_base_plugin_common.filter_fields
|
@db_base_plugin_common.filter_fields
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
@db_base_plugin_common.convert_result_to_dict
|
||||||
def get_policies(self, context, filters=None, fields=None,
|
def get_policies(self, context, filters=None, fields=None, sorts=None,
|
||||||
sorts=None, limit=None, marker=None,
|
limit=None, marker=None, page_reverse=False):
|
||||||
page_reverse=False):
|
"""Get QoS policies.
|
||||||
|
|
||||||
|
:param context: neutron api request context
|
||||||
|
:type context: neutron.context.Context
|
||||||
|
:param filters: search criteria
|
||||||
|
:type filters: dict
|
||||||
|
|
||||||
|
:returns: QosPolicy objects meeting the search criteria
|
||||||
|
"""
|
||||||
return policy_object.QosPolicy.get_objects(context, **filters)
|
return policy_object.QosPolicy.get_objects(context, **filters)
|
||||||
|
|
||||||
#TODO(mangelajo): need to add a proxy catch-all for rules, so
|
|
||||||
# we capture the API function call, and just pass
|
|
||||||
# the rule type as a parameter removing lots of
|
|
||||||
# future code duplication when we have more rules.
|
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
|
||||||
def create_policy_bandwidth_limit_rule(self, context, policy_id,
|
|
||||||
bandwidth_limit_rule):
|
|
||||||
# make sure we will have a policy object to push resource update
|
|
||||||
with db_api.autonested_transaction(context.session):
|
|
||||||
# first, validate that we have access to the policy
|
|
||||||
policy = self._get_policy_obj(context, policy_id)
|
|
||||||
rule = rule_object.QosBandwidthLimitRule(
|
|
||||||
context, qos_policy_id=policy_id,
|
|
||||||
**bandwidth_limit_rule['bandwidth_limit_rule'])
|
|
||||||
rule.create()
|
|
||||||
policy.reload_rules()
|
|
||||||
self.notification_driver_manager.update_policy(context, policy)
|
|
||||||
return rule
|
|
||||||
|
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
|
||||||
def update_policy_bandwidth_limit_rule(self, context, rule_id, policy_id,
|
|
||||||
bandwidth_limit_rule):
|
|
||||||
# make sure we will have a policy object to push resource update
|
|
||||||
with db_api.autonested_transaction(context.session):
|
|
||||||
# first, validate that we have access to the policy
|
|
||||||
policy = self._get_policy_obj(context, policy_id)
|
|
||||||
# check if the rule belong to the policy
|
|
||||||
policy.get_rule_by_id(rule_id)
|
|
||||||
rule = rule_object.QosBandwidthLimitRule(
|
|
||||||
context, id=rule_id)
|
|
||||||
rule.obj_reset_changes()
|
|
||||||
for k, v in bandwidth_limit_rule['bandwidth_limit_rule'].items():
|
|
||||||
if k != 'id':
|
|
||||||
setattr(rule, k, v)
|
|
||||||
rule.update()
|
|
||||||
policy.reload_rules()
|
|
||||||
self.notification_driver_manager.update_policy(context, policy)
|
|
||||||
return rule
|
|
||||||
|
|
||||||
def delete_policy_bandwidth_limit_rule(self, context, rule_id, policy_id):
|
|
||||||
# make sure we will have a policy object to push resource update
|
|
||||||
with db_api.autonested_transaction(context.session):
|
|
||||||
# first, validate that we have access to the policy
|
|
||||||
policy = self._get_policy_obj(context, policy_id)
|
|
||||||
rule = policy.get_rule_by_id(rule_id)
|
|
||||||
rule.delete()
|
|
||||||
policy.reload_rules()
|
|
||||||
self.notification_driver_manager.update_policy(context, policy)
|
|
||||||
|
|
||||||
@db_base_plugin_common.filter_fields
|
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
|
||||||
def get_policy_bandwidth_limit_rule(self, context, rule_id,
|
|
||||||
policy_id, fields=None):
|
|
||||||
# make sure we have access to the policy when fetching the rule
|
|
||||||
with db_api.autonested_transaction(context.session):
|
|
||||||
# first, validate that we have access to the policy
|
|
||||||
self._get_policy_obj(context, policy_id)
|
|
||||||
rule = rule_object.QosBandwidthLimitRule.get_object(
|
|
||||||
context, id=rule_id)
|
|
||||||
if not rule:
|
|
||||||
raise n_exc.QosRuleNotFound(policy_id=policy_id, rule_id=rule_id)
|
|
||||||
return rule
|
|
||||||
|
|
||||||
@db_base_plugin_common.filter_fields
|
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
|
||||||
def get_policy_bandwidth_limit_rules(self, context, policy_id,
|
|
||||||
filters=None, fields=None,
|
|
||||||
sorts=None, limit=None,
|
|
||||||
marker=None, page_reverse=False):
|
|
||||||
# make sure we have access to the policy when fetching rules
|
|
||||||
with db_api.autonested_transaction(context.session):
|
|
||||||
# first, validate that we have access to the policy
|
|
||||||
self._get_policy_obj(context, policy_id)
|
|
||||||
filters = filters or dict()
|
|
||||||
filters[qos_consts.QOS_POLICY_ID] = policy_id
|
|
||||||
return rule_object.QosBandwidthLimitRule.get_objects(context,
|
|
||||||
**filters)
|
|
||||||
|
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
|
||||||
def create_policy_dscp_marking_rule(self, context, policy_id,
|
|
||||||
dscp_marking_rule):
|
|
||||||
with db_api.autonested_transaction(context.session):
|
|
||||||
# first, validate that we have access to the policy
|
|
||||||
policy = self._get_policy_obj(context, policy_id)
|
|
||||||
rule = rule_object.QosDscpMarkingRule(
|
|
||||||
context, qos_policy_id=policy_id,
|
|
||||||
**dscp_marking_rule['dscp_marking_rule'])
|
|
||||||
rule.create()
|
|
||||||
policy.reload_rules()
|
|
||||||
self.notification_driver_manager.update_policy(context, policy)
|
|
||||||
return rule
|
|
||||||
|
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
|
||||||
def update_policy_dscp_marking_rule(self, context, rule_id, policy_id,
|
|
||||||
dscp_marking_rule):
|
|
||||||
with db_api.autonested_transaction(context.session):
|
|
||||||
# first, validate that we have access to the policy
|
|
||||||
policy = self._get_policy_obj(context, policy_id)
|
|
||||||
# check if the rule belong to the policy
|
|
||||||
policy.get_rule_by_id(rule_id)
|
|
||||||
rule = rule_object.QosDscpMarkingRule(
|
|
||||||
context, id=rule_id)
|
|
||||||
rule.obj_reset_changes()
|
|
||||||
for k, v in dscp_marking_rule['dscp_marking_rule'].items():
|
|
||||||
if k != 'id':
|
|
||||||
setattr(rule, k, v)
|
|
||||||
rule.update()
|
|
||||||
policy.reload_rules()
|
|
||||||
self.notification_driver_manager.update_policy(context, policy)
|
|
||||||
return rule
|
|
||||||
|
|
||||||
def delete_policy_dscp_marking_rule(self, context, rule_id, policy_id):
|
|
||||||
# make sure we will have a policy object to push resource update
|
|
||||||
with db_api.autonested_transaction(context.session):
|
|
||||||
# first, validate that we have access to the policy
|
|
||||||
policy = self._get_policy_obj(context, policy_id)
|
|
||||||
rule = policy.get_rule_by_id(rule_id)
|
|
||||||
rule.delete()
|
|
||||||
policy.reload_rules()
|
|
||||||
self.notification_driver_manager.update_policy(context, policy)
|
|
||||||
|
|
||||||
@db_base_plugin_common.filter_fields
|
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
|
||||||
def get_policy_dscp_marking_rule(self, context, rule_id,
|
|
||||||
policy_id, fields=None):
|
|
||||||
# make sure we have access to the policy when fetching the rule
|
|
||||||
with db_api.autonested_transaction(context.session):
|
|
||||||
# first, validate that we have access to the policy
|
|
||||||
self._get_policy_obj(context, policy_id)
|
|
||||||
rule = rule_object.QosDscpMarkingRule.get_object(
|
|
||||||
context, id=rule_id)
|
|
||||||
if not rule:
|
|
||||||
raise n_exc.QosRuleNotFound(policy_id=policy_id, rule_id=rule_id)
|
|
||||||
return rule
|
|
||||||
|
|
||||||
@db_base_plugin_common.filter_fields
|
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
|
||||||
def get_policy_dscp_marking_rules(self, context, policy_id,
|
|
||||||
filters=None, fields=None,
|
|
||||||
sorts=None, limit=None,
|
|
||||||
marker=None, page_reverse=False):
|
|
||||||
# make sure we have access to the policy when fetching rules
|
|
||||||
with db_api.autonested_transaction(context.session):
|
|
||||||
# first, validate that we have access to the policy
|
|
||||||
self._get_policy_obj(context, policy_id)
|
|
||||||
filters = filters or dict()
|
|
||||||
filters[qos_consts.QOS_POLICY_ID] = policy_id
|
|
||||||
return rule_object.QosDscpMarkingRule.get_objects(context,
|
|
||||||
**filters)
|
|
||||||
|
|
||||||
# TODO(QoS): enforce rule types when accessing rule objects
|
|
||||||
@db_base_plugin_common.filter_fields
|
@db_base_plugin_common.filter_fields
|
||||||
@db_base_plugin_common.convert_result_to_dict
|
@db_base_plugin_common.convert_result_to_dict
|
||||||
def get_rule_types(self, context, filters=None, fields=None,
|
def get_rule_types(self, context, filters=None, fields=None,
|
||||||
sorts=None, limit=None,
|
sorts=None, limit=None,
|
||||||
marker=None, page_reverse=False):
|
marker=None, page_reverse=False):
|
||||||
|
if not filters:
|
||||||
|
filters = {}
|
||||||
return rule_type_object.QosRuleType.get_objects(**filters)
|
return rule_type_object.QosRuleType.get_objects(**filters)
|
||||||
|
|
||||||
|
@db_base_plugin_common.convert_result_to_dict
|
||||||
|
def create_policy_rule(self, context, rule_obj, policy_id, rule_data):
|
||||||
|
"""Create a QoS policy rule.
|
||||||
|
|
||||||
|
:param context: neutron api request context
|
||||||
|
:type context: neutron.context.Context
|
||||||
|
:param rule_obj: the rule object
|
||||||
|
:type rule_obj: a class from the rule_object (qos.objects.rule) module
|
||||||
|
:param policy_id: the id of the QosPolicy for which to create the rule
|
||||||
|
:type policy_id: str uuid
|
||||||
|
:param rule_data: the rule data to be applied
|
||||||
|
:type rule_data: dict
|
||||||
|
|
||||||
|
:returns: a QoS policy rule object
|
||||||
|
"""
|
||||||
|
rule_type = rule_obj.rule_type
|
||||||
|
rule_data = rule_data[rule_type + '_rule']
|
||||||
|
|
||||||
|
with db_api.autonested_transaction(context.session):
|
||||||
|
# Ensure that we have access to the policy.
|
||||||
|
policy = self._get_policy_obj(context, policy_id)
|
||||||
|
rule = rule_obj(context, qos_policy_id=policy_id, **rule_data)
|
||||||
|
rule.create()
|
||||||
|
policy.reload_rules()
|
||||||
|
self.notification_driver_manager.update_policy(context, policy)
|
||||||
|
return rule
|
||||||
|
|
||||||
|
@db_base_plugin_common.convert_result_to_dict
|
||||||
|
def update_policy_rule(self, context, rule_obj, rule_id, policy_id,
|
||||||
|
rule_data):
|
||||||
|
"""Update a QoS policy rule.
|
||||||
|
|
||||||
|
:param context: neutron api request context
|
||||||
|
:type context: neutron.context.Context
|
||||||
|
:param rule_obj: the rule object
|
||||||
|
:type rule_obj: a class from the rule_object (qos.objects.rule) module
|
||||||
|
:param rule_id: the id of the QoS policy rule to update
|
||||||
|
:type rule_id: str uuid
|
||||||
|
:param policy_id: the id of the rule's policy
|
||||||
|
:type policy_id: str uuid
|
||||||
|
:param rule_data: the new rule data to update
|
||||||
|
:type rule_data: dict
|
||||||
|
|
||||||
|
:returns: a QoS policy rule object
|
||||||
|
"""
|
||||||
|
rule_type = rule_obj.rule_type
|
||||||
|
rule_data = rule_data[rule_type + '_rule']
|
||||||
|
|
||||||
|
with db_api.autonested_transaction(context.session):
|
||||||
|
# Ensure we have access to the policy.
|
||||||
|
policy = self._get_policy_obj(context, policy_id)
|
||||||
|
# Ensure the rule belongs to the policy.
|
||||||
|
policy.get_rule_by_id(rule_id)
|
||||||
|
rule = rule_obj(context, id=rule_id)
|
||||||
|
rule.update_nonidentifying_fields(rule_data, reset_changes=True)
|
||||||
|
rule.update()
|
||||||
|
policy.reload_rules()
|
||||||
|
self.notification_driver_manager.update_policy(context, policy)
|
||||||
|
return rule
|
||||||
|
|
||||||
|
def delete_policy_rule(self, context, rule_obj, rule_id, policy_id):
|
||||||
|
"""Delete a QoS policy rule.
|
||||||
|
|
||||||
|
:param context: neutron api request context
|
||||||
|
:type context: neutron.context.Context
|
||||||
|
:param rule_obj: the rule object
|
||||||
|
:type rule_obj: a class from the rule_object (qos.objects.rule) module
|
||||||
|
:param rule_id: the id of the QosPolicy Rule to delete
|
||||||
|
:type rule_id: str uuid
|
||||||
|
:param policy_id: the id of the rule's policy
|
||||||
|
:type policy_id: str uuid
|
||||||
|
|
||||||
|
:returns: None
|
||||||
|
"""
|
||||||
|
with db_api.autonested_transaction(context.session):
|
||||||
|
# Ensure we have access to the policy.
|
||||||
|
policy = self._get_policy_obj(context, policy_id)
|
||||||
|
rule = policy.get_rule_by_id(rule_id)
|
||||||
|
rule.delete()
|
||||||
|
policy.reload_rules()
|
||||||
|
self.notification_driver_manager.update_policy(context, policy)
|
||||||
|
|
||||||
|
@db_base_plugin_common.filter_fields
|
||||||
|
@db_base_plugin_common.convert_result_to_dict
|
||||||
|
def get_policy_rule(self, context, rule_obj, rule_id, policy_id,
|
||||||
|
fields=None):
|
||||||
|
"""Get a QoS policy rule.
|
||||||
|
|
||||||
|
:param context: neutron api request context
|
||||||
|
:type context: neutron.context.Context
|
||||||
|
:param rule_obj: the rule object
|
||||||
|
:type rule_obj: a class from the rule_object (qos.objects.rule) module
|
||||||
|
:param rule_id: the id of the QoS policy rule to get
|
||||||
|
:type rule_id: str uuid
|
||||||
|
:param policy_id: the id of the rule's policy
|
||||||
|
:type policy_id: str uuid
|
||||||
|
|
||||||
|
:returns: a QoS policy rule object
|
||||||
|
:raises: n_exc.QosRuleNotFound
|
||||||
|
"""
|
||||||
|
with db_api.autonested_transaction(context.session):
|
||||||
|
# Ensure we have access to the policy.
|
||||||
|
self._get_policy_obj(context, policy_id)
|
||||||
|
rule = rule_obj.get_object(context, id=rule_id)
|
||||||
|
if not rule:
|
||||||
|
raise n_exc.QosRuleNotFound(policy_id=policy_id, rule_id=rule_id)
|
||||||
|
return rule
|
||||||
|
|
||||||
|
# TODO(QoS): enforce rule types when accessing rule objects
|
||||||
|
@db_base_plugin_common.filter_fields
|
||||||
|
@db_base_plugin_common.convert_result_to_dict
|
||||||
|
def get_policy_rules(self, context, rule_obj, policy_id, filters=None,
|
||||||
|
fields=None, sorts=None, limit=None, marker=None,
|
||||||
|
page_reverse=False):
|
||||||
|
"""Get QoS policy rules.
|
||||||
|
|
||||||
|
:param context: neutron api request context
|
||||||
|
:type context: neutron.context.Context
|
||||||
|
:param rule_obj: the rule object
|
||||||
|
:type rule_obj: a class from the rule_object (qos.objects.rule) module
|
||||||
|
:param policy_id: the id of the QosPolicy for which to get rules
|
||||||
|
:type policy_id: str uuid
|
||||||
|
|
||||||
|
:returns: QoS policy rule objects meeting the search criteria
|
||||||
|
"""
|
||||||
|
with db_api.autonested_transaction(context.session):
|
||||||
|
# Ensure we have access to the policy.
|
||||||
|
self._get_policy_obj(context, policy_id)
|
||||||
|
filters = filters or dict()
|
||||||
|
filters[qos_consts.QOS_POLICY_ID] = policy_id
|
||||||
|
return rule_obj.get_objects(context, **filters)
|
||||||
|
@ -469,6 +469,27 @@ class BaseObjectIfaceTestCase(_BaseObjectTestCase, test_base.BaseTestCase):
|
|||||||
obj = self._test_class(self.context, **self.obj_fields[0])
|
obj = self._test_class(self.context, **self.obj_fields[0])
|
||||||
self.assertRaises(base.NeutronDbObjectDuplicateEntry, obj.create)
|
self.assertRaises(base.NeutronDbObjectDuplicateEntry, obj.create)
|
||||||
|
|
||||||
|
def test_update_nonidentifying_fields(self):
|
||||||
|
if not self._test_class.primary_keys:
|
||||||
|
self.skipTest(
|
||||||
|
'Test class %r has no primary keys' % self._test_class)
|
||||||
|
|
||||||
|
with mock.patch.object(obj_base.VersionedObject, 'obj_reset_changes'):
|
||||||
|
expected = self._test_class(self.context, **self.obj_fields[0])
|
||||||
|
for key, val in self.obj_fields[1].items():
|
||||||
|
if key not in expected.primary_keys:
|
||||||
|
setattr(expected, key, val)
|
||||||
|
observed = self._test_class(self.context, **self.obj_fields[0])
|
||||||
|
observed.update_nonidentifying_fields(self.obj_fields[1],
|
||||||
|
reset_changes=True)
|
||||||
|
self.assertEqual(expected, observed)
|
||||||
|
self.assertTrue(observed.obj_reset_changes.called)
|
||||||
|
|
||||||
|
with mock.patch.object(obj_base.VersionedObject, 'obj_reset_changes'):
|
||||||
|
obj = self._test_class(self.context, **self.obj_fields[0])
|
||||||
|
obj.update_nonidentifying_fields(self.obj_fields[1])
|
||||||
|
self.assertFalse(obj.obj_reset_changes.called)
|
||||||
|
|
||||||
@mock.patch.object(obj_db_api, 'update_object')
|
@mock.patch.object(obj_db_api, 'update_object')
|
||||||
def test_update_no_changes(self, update_mock):
|
def test_update_no_changes(self, update_mock):
|
||||||
with mock.patch.object(base.NeutronDbObject,
|
with mock.patch.object(base.NeutronDbObject,
|
||||||
|
@ -21,6 +21,7 @@ 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.services.qos import qos_consts
|
||||||
from neutron.tests.unit.services.qos import base
|
from neutron.tests.unit.services.qos import base
|
||||||
|
|
||||||
|
|
||||||
@ -157,6 +158,17 @@ class TestQosPlugin(base.BaseQosTestCase):
|
|||||||
self.qos_plugin.delete_policy_bandwidth_limit_rule,
|
self.qos_plugin.delete_policy_bandwidth_limit_rule,
|
||||||
self.ctxt, self.rule.id, _policy.id)
|
self.ctxt, self.rule.id, _policy.id)
|
||||||
|
|
||||||
|
def test_get_policy_bandwidth_limit_rule(self):
|
||||||
|
with mock.patch('neutron.objects.qos.policy.QosPolicy.get_object',
|
||||||
|
return_value=self.policy):
|
||||||
|
with mock.patch('neutron.objects.qos.rule.'
|
||||||
|
'QosBandwidthLimitRule.'
|
||||||
|
'get_object') as get_object_mock:
|
||||||
|
self.qos_plugin.get_policy_bandwidth_limit_rule(
|
||||||
|
self.ctxt, self.rule.id, self.policy.id)
|
||||||
|
get_object_mock.assert_called_once_with(self.ctxt,
|
||||||
|
id=self.rule.id)
|
||||||
|
|
||||||
def test_get_policy_bandwidth_limit_rules_for_policy(self):
|
def test_get_policy_bandwidth_limit_rules_for_policy(self):
|
||||||
with mock.patch('neutron.objects.qos.policy.QosPolicy.get_object',
|
with mock.patch('neutron.objects.qos.policy.QosPolicy.get_object',
|
||||||
return_value=self.policy):
|
return_value=self.policy):
|
||||||
@ -300,3 +312,19 @@ class TestQosPlugin(base.BaseQosTestCase):
|
|||||||
n_exc.QosPolicyNotFound,
|
n_exc.QosPolicyNotFound,
|
||||||
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)
|
||||||
|
|
||||||
|
def test_verify_bad_method_call(self):
|
||||||
|
self.assertRaises(AttributeError, getattr, self.qos_plugin,
|
||||||
|
'create_policy_bandwidth_limit_rules')
|
||||||
|
|
||||||
|
def test_get_rule_types(self):
|
||||||
|
core_plugin = manager.NeutronManager.get_plugin()
|
||||||
|
rule_types_mock = mock.PropertyMock(
|
||||||
|
return_value=qos_consts.VALID_RULE_TYPES)
|
||||||
|
filters = {'type': 'type_id'}
|
||||||
|
with mock.patch.object(core_plugin, 'supported_qos_rule_types',
|
||||||
|
new_callable=rule_types_mock,
|
||||||
|
create=True):
|
||||||
|
types = self.qos_plugin.get_rule_types(self.ctxt, filters=filters)
|
||||||
|
self.assertEqual(sorted(qos_consts.VALID_RULE_TYPES),
|
||||||
|
sorted(type_['type'] for type_ in types))
|
||||||
|
Loading…
x
Reference in New Issue
Block a user