Deprecate 'remote_ip_prefix' parameter in metering label rules

As proposed in the RFE and then approved in the spec. The parameter
'remote_ip_prefix' in metering label rules has been deprecated.
Its name expresses the opposite of what does when used, and the
lack of documentation confuses people. Moreover, an alternative
method has been proposed and approved to enable operators to create
metering rules using both source and destination IP addresses.

The parameter will be removed in future releases.

Partially-Implements: https://bugs.launchpad.net/neutron/+bug/1889431
RFE: https://bugs.launchpad.net/neutron/+bug/1889431

Depends-On: https://review.opendev.org/#/c/746203/
Depends-On: https://review.opendev.org/#/c/744702/
Depends-On: https://review.opendev.org/#/c/743828/
Depends-On: https://review.opendev.org/#/c/746142/
Depends-On: https://review.opendev.org/#/c/746347/
Change-Id: Iba2b0d09fdd631f8bd2c3c951fd69b243deed652
This commit is contained in:
Rafael Weingärtner 2020-08-17 14:37:58 -03:00
parent e29794be55
commit d4022f62a0
3 changed files with 64 additions and 6 deletions

View File

@ -44,4 +44,11 @@ class MeteringLabelRule(resource.Resource):
#: The ID of the project this metering label rule is associated with. #: The ID of the project this metering label rule is associated with.
project_id = resource.Body('tenant_id') project_id = resource.Body('tenant_id')
#: The remote IP prefix to be associated with this metering label rule. #: The remote IP prefix to be associated with this metering label rule.
remote_ip_prefix = resource.Body('remote_ip_prefix') remote_ip_prefix = resource.Body(
'remote_ip_prefix', deprecated=True,
deprecation_reason="The use of 'remote_ip_prefix' in metering label "
"rules is deprecated and will be removed in future "
"releases. One should use instead, the "
"'source_ip_prefix' and/or 'destination_ip_prefix' "
"parameters. For more details, you can check the "
"spec: https://review.opendev.org/#/c/744702/.")

View File

@ -50,6 +50,8 @@ from openstack import utils
_SEEN_FORMAT = '{name}_seen' _SEEN_FORMAT = '{name}_seen'
LOG = _log.setup_logging(__name__)
def _convert_type(value, data_type, list_type=None): def _convert_type(value, data_type, list_type=None):
# This should allow handling list of dicts that have their own # This should allow handling list of dicts that have their own
@ -89,8 +91,18 @@ class _BaseComponent:
# The class to be used for mappings # The class to be used for mappings
_map_cls = dict _map_cls = dict
#: Marks the property as deprecated.
deprecated = False
#: Deprecation reason message used to warn users when deprecated == True
deprecation_reason = None
#: Control field used to manage the deprecation warning. We want to warn
# only once when the attribute is retrieved in the code.
already_warned_deprecation = False
def __init__(self, name, type=None, default=None, alias=None, aka=None, def __init__(self, name, type=None, default=None, alias=None, aka=None,
alternate_id=False, list_type=None, coerce_to_default=False, alternate_id=False, list_type=None, coerce_to_default=False,
deprecated=False, deprecation_reason=None,
**kwargs): **kwargs):
"""A typed descriptor for a component that makes up a Resource """A typed descriptor for a component that makes up a Resource
@ -115,6 +127,11 @@ class _BaseComponent:
If the Component is None or not present, force the given default If the Component is None or not present, force the given default
to be used. If a default is not given but a type is given, to be used. If a default is not given but a type is given,
construct an empty version of the type in question. construct an empty version of the type in question.
:param deprecated:
Indicates if the option is deprecated. If it is, we display a
warning message to the user.
:param deprecation_reason:
Custom deprecation message.
""" """
self.name = name self.name = name
self.type = type self.type = type
@ -128,6 +145,9 @@ class _BaseComponent:
self.list_type = list_type self.list_type = list_type
self.coerce_to_default = coerce_to_default self.coerce_to_default = coerce_to_default
self.deprecated = deprecated
self.deprecation_reason = deprecation_reason
def __get__(self, instance, owner): def __get__(self, instance, owner):
if instance is None: if instance is None:
return self return self
@ -137,6 +157,7 @@ class _BaseComponent:
try: try:
value = attributes[self.name] value = attributes[self.name]
except KeyError: except KeyError:
value = self.default
if self.alias: if self.alias:
# Resource attributes can be aliased to each other. If neither # Resource attributes can be aliased to each other. If neither
# of them exist, then simply doing a # of them exist, then simply doing a
@ -156,15 +177,29 @@ class _BaseComponent:
setattr(instance, seen_flag, True) setattr(instance, seen_flag, True)
value = getattr(instance, self.alias) value = getattr(instance, self.alias)
delattr(instance, seen_flag) delattr(instance, seen_flag)
return value self.warn_if_deprecated_property(value)
return self.default return value
# self.type() should not be called on None objects. # self.type() should not be called on None objects.
if value is None: if value is None:
return None return None
self.warn_if_deprecated_property(value)
return _convert_type(value, self.type, self.list_type) return _convert_type(value, self.type, self.list_type)
def warn_if_deprecated_property(self, value):
deprecated = object.__getattribute__(self, 'deprecated')
deprecate_reason = object.__getattribute__(self, 'deprecation_reason')
if value and deprecated and not self.already_warned_deprecation:
self.already_warned_deprecation = True
if not deprecate_reason:
LOG.warning("The option [%s] has been deprecated. "
"Please avoid using it.", self.name)
else:
LOG.warning(deprecate_reason)
return value
def __set__(self, instance, value): def __set__(self, instance, value):
if self.coerce_to_default and value is None: if self.coerce_to_default and value is None:
value = self.default value = self.default
@ -562,6 +597,14 @@ class Resource(dict):
self._computed.attributes == comparand._computed.attributes self._computed.attributes == comparand._computed.attributes
]) ])
def warning_if_attribute_deprecated(self, attr, value):
if value and self.deprecated:
if not self.deprecation_reason:
LOG.warning("The option [%s] has been deprecated. "
"Please avoid using it.", attr)
else:
LOG.warning(self.deprecation_reason)
def __getattribute__(self, name): def __getattribute__(self, name):
"""Return an attribute on this instance """Return an attribute on this instance
@ -575,7 +618,9 @@ class Resource(dict):
else: else:
try: try:
return self._body[self._alternate_id()] return self._body[self._alternate_id()]
except KeyError: except KeyError as e:
LOG.debug("Attribute [%s] not found in [%s]: %s.",
self._alternate_id(), self._body, e)
return None return None
else: else:
try: try:
@ -2014,7 +2059,6 @@ def wait_for_status(session, resource, status, failures, interval=None,
:raises: :class:`~AttributeError` if the resource does not have a status :raises: :class:`~AttributeError` if the resource does not have a status
attribute attribute
""" """
log = _log.setup_logging(__name__)
current_status = getattr(resource, attribute) current_status = getattr(resource, attribute)
if _normalize_status(current_status) == status.lower(): if _normalize_status(current_status) == status.lower():
@ -2048,7 +2092,7 @@ def wait_for_status(session, resource, status, failures, interval=None,
"{name} transitioned to failure state {status}".format( "{name} transitioned to failure state {status}".format(
name=name, status=new_status)) name=name, status=new_status))
log.debug('Still waiting for resource %s to reach state %s, ' LOG.debug('Still waiting for resource %s to reach state %s, '
'current state is %s', name, status, new_status) 'current state is %s', name, status, new_status)

View File

@ -0,0 +1,7 @@
---
deprecations:
- |
Deprecate the use of 'remote_ip_prefix' in metering label rules, and it
will be removed in future releases. One should use instead the
'source_ip_prefix' and/or 'destination_ip_prefix' parameters. For more
details, you can check the spec: https://review.opendev.org/#/c/744702/.