Return retries on HTTP CONFLICT to baremetal.attach_vif_to_node

Unfortunately, it is very easy to hit the "node locked" error when retries
are disabled, so add retries to this call as well.

Change-Id: I076cf2537a20687932aca9fd85358bf02882b736
This commit is contained in:
Dmitry Tantsur 2019-01-18 18:15:19 +01:00
parent a977c85070
commit 3b022e8d70
4 changed files with 32 additions and 7 deletions

View File

@ -632,7 +632,7 @@ class Proxy(proxy.Proxy):
return self._delete(_portgroup.PortGroup, port_group, return self._delete(_portgroup.PortGroup, port_group,
ignore_missing=ignore_missing) ignore_missing=ignore_missing)
def attach_vif_to_node(self, node, vif_id): def attach_vif_to_node(self, node, vif_id, retry_on_conflict=True):
"""Attach a VIF to the node. """Attach a VIF to the node.
The exact form of the VIF ID depends on the network interface used by The exact form of the VIF ID depends on the network interface used by
@ -643,12 +643,16 @@ class Proxy(proxy.Proxy):
:param node: The value can be either the name or ID of a node or :param node: The value can be either the name or ID of a node or
a :class:`~openstack.baremetal.v1.node.Node` instance. a :class:`~openstack.baremetal.v1.node.Node` instance.
:param string vif_id: Backend-specific VIF ID. :param string vif_id: Backend-specific VIF ID.
:param retry_on_conflict: Whether to retry HTTP CONFLICT errors.
This can happen when either the VIF is already used on a node or
the node is locked. Since the latter happens more often, the
default value is True.
:return: ``None`` :return: ``None``
:raises: :exc:`~openstack.exceptions.NotSupported` if the server :raises: :exc:`~openstack.exceptions.NotSupported` if the server
does not support the VIF API. does not support the VIF API.
""" """
res = self._get_resource(_node.Node, node) res = self._get_resource(_node.Node, node)
res.attach_vif(self, vif_id) res.attach_vif(self, vif_id, retry_on_conflict=retry_on_conflict)
def detach_vif_from_node(self, node, vif_id, ignore_missing=True): def detach_vif_from_node(self, node, vif_id, ignore_missing=True):
"""Detach a VIF from the node. """Detach a VIF from the node.

View File

@ -447,7 +447,7 @@ class Node(_common.ListMixin, resource.Resource):
"the last error is %(error)s" % "the last error is %(error)s" %
{'node': self.id, 'error': self.last_error}) {'node': self.id, 'error': self.last_error})
def attach_vif(self, session, vif_id): def attach_vif(self, session, vif_id, retry_on_conflict=True):
"""Attach a VIF to the node. """Attach a VIF to the node.
The exact form of the VIF ID depends on the network interface used by The exact form of the VIF ID depends on the network interface used by
@ -458,6 +458,10 @@ class Node(_common.ListMixin, resource.Resource):
:param session: The session to use for making this request. :param session: The session to use for making this request.
:type session: :class:`~keystoneauth1.adapter.Adapter` :type session: :class:`~keystoneauth1.adapter.Adapter`
:param string vif_id: Backend-specific VIF ID. :param string vif_id: Backend-specific VIF ID.
:param retry_on_conflict: Whether to retry HTTP CONFLICT errors.
This can happen when either the VIF is already used on a node or
the node is locked. Since the latter happens more often, the
default value is True.
:return: ``None`` :return: ``None``
:raises: :exc:`~openstack.exceptions.NotSupported` if the server :raises: :exc:`~openstack.exceptions.NotSupported` if the server
does not support the VIF API. does not support the VIF API.
@ -470,12 +474,13 @@ class Node(_common.ListMixin, resource.Resource):
request = self._prepare_request(requires_id=True) request = self._prepare_request(requires_id=True)
request.url = utils.urljoin(request.url, 'vifs') request.url = utils.urljoin(request.url, 'vifs')
body = {'id': vif_id} body = {'id': vif_id}
retriable_status_codes = _common.RETRIABLE_STATUS_CODES
if not retry_on_conflict:
retriable_status_codes = set(retriable_status_codes) - {409}
response = session.post( response = session.post(
request.url, json=body, request.url, json=body,
headers=request.headers, microversion=version, headers=request.headers, microversion=version,
# NOTE(dtantsur): do not retry CONFLICT, it's a valid status code retriable_status_codes=retriable_status_codes)
# in this API when the VIF is already attached to another node.
retriable_status_codes=[503])
msg = ("Failed to attach VIF {vif} to bare metal node {node}" msg = ("Failed to attach VIF {vif} to bare metal node {node}"
.format(node=self.id, vif=vif_id)) .format(node=self.id, vif=vif_id))

View File

@ -342,7 +342,15 @@ class TestNodeVif(base.TestCase):
self.session.post.assert_called_once_with( self.session.post.assert_called_once_with(
'nodes/%s/vifs' % self.node.id, json={'id': self.vif_id}, 'nodes/%s/vifs' % self.node.id, json={'id': self.vif_id},
headers=mock.ANY, microversion='1.28', headers=mock.ANY, microversion='1.28',
retriable_status_codes=[503]) retriable_status_codes=[409, 503])
def test_attach_vif_no_retries(self):
self.assertIsNone(self.node.attach_vif(self.session, self.vif_id,
retry_on_conflict=False))
self.session.post.assert_called_once_with(
'nodes/%s/vifs' % self.node.id, json={'id': self.vif_id},
headers=mock.ANY, microversion='1.28',
retriable_status_codes={503})
def test_detach_vif_existing(self): def test_detach_vif_existing(self):
self.assertTrue(self.node.detach_vif(self.session, self.vif_id)) self.assertTrue(self.node.detach_vif(self.session, self.vif_id))

View File

@ -0,0 +1,8 @@
---
fixes:
- |
Changes the ``baremetal.attach_vif_to_node`` call to retry HTTP CONFLICT
by default. While it's a valid error code when a VIF is already attached
to a node, the same code is also used when the target node is locked.
The latter happens more often, so the retries are now on by default and
can be disabled by setting ``retry_on_conflict`` to ``False``.