Retry connections to Nova
Sometimes Neutron is failing to send notification to Nova due to timeout, refused connection or another HTTP error. Retry send in those cases. Closes-Bug: #1987780 Change-Id: Iaaccec770484234b704f70f3c144efac4d8ffba0
This commit is contained in:
parent
4b83bf462d
commit
cd475f9af8
@ -16,6 +16,7 @@
|
|||||||
import contextlib
|
import contextlib
|
||||||
|
|
||||||
from eventlet.green import threading
|
from eventlet.green import threading
|
||||||
|
from keystoneauth1 import exceptions as ks_exceptions
|
||||||
from keystoneauth1 import loading as ks_loading
|
from keystoneauth1 import loading as ks_loading
|
||||||
from neutron_lib.callbacks import events
|
from neutron_lib.callbacks import events
|
||||||
from neutron_lib.callbacks import registry
|
from neutron_lib.callbacks import registry
|
||||||
@ -32,6 +33,7 @@ from oslo_context import context as common_context
|
|||||||
from oslo_log import log as logging
|
from oslo_log import log as logging
|
||||||
from oslo_utils import uuidutils
|
from oslo_utils import uuidutils
|
||||||
from sqlalchemy.orm import attributes as sql_attr
|
from sqlalchemy.orm import attributes as sql_attr
|
||||||
|
import tenacity
|
||||||
|
|
||||||
from neutron.notifiers import batch_notifier
|
from neutron.notifiers import batch_notifier
|
||||||
|
|
||||||
@ -267,6 +269,12 @@ class Notifier(object):
|
|||||||
'tag': port.id})
|
'tag': port.id})
|
||||||
self.send_port_status(None, None, port)
|
self.send_port_status(None, None, port)
|
||||||
|
|
||||||
|
@tenacity.retry(
|
||||||
|
retry=tenacity.retry_if_exception_type(
|
||||||
|
ks_exceptions.RetriableConnectionFailure),
|
||||||
|
wait=tenacity.wait_exponential(multiplier=0.01, max=1),
|
||||||
|
stop=tenacity.stop_after_delay(1),
|
||||||
|
after=tenacity.after_log(LOG, logging.DEBUG))
|
||||||
def send_events(self, batched_events):
|
def send_events(self, batched_events):
|
||||||
LOG.debug("Sending events: %s", batched_events)
|
LOG.debug("Sending events: %s", batched_events)
|
||||||
novaclient = self._get_nova_client()
|
novaclient = self._get_nova_client()
|
||||||
@ -276,6 +284,10 @@ class Notifier(object):
|
|||||||
except nova_exceptions.NotFound:
|
except nova_exceptions.NotFound:
|
||||||
LOG.debug("Nova returned NotFound for event: %s",
|
LOG.debug("Nova returned NotFound for event: %s",
|
||||||
batched_events)
|
batched_events)
|
||||||
|
except ks_exceptions.RetriableConnectionFailure:
|
||||||
|
raise
|
||||||
|
# next clause handles all exceptions
|
||||||
|
# so reraise for retry decorator
|
||||||
except Exception:
|
except Exception:
|
||||||
LOG.exception("Failed to notify nova on events: %s",
|
LOG.exception("Failed to notify nova on events: %s",
|
||||||
batched_events)
|
batched_events)
|
||||||
|
@ -17,6 +17,7 @@ import queue
|
|||||||
from unittest import mock
|
from unittest import mock
|
||||||
|
|
||||||
import eventlet
|
import eventlet
|
||||||
|
from keystoneauth1 import exceptions as ks_exc
|
||||||
from neutron_lib import constants as n_const
|
from neutron_lib import constants as n_const
|
||||||
from neutron_lib import context as n_ctx
|
from neutron_lib import context as n_ctx
|
||||||
from neutron_lib import exceptions as n_exc
|
from neutron_lib import exceptions as n_exc
|
||||||
@ -248,6 +249,14 @@ class TestNovaNotify(base.BaseTestCase):
|
|||||||
nova_exceptions.NotFound)
|
nova_exceptions.NotFound)
|
||||||
self.nova_notifier.send_events([])
|
self.nova_notifier.send_events([])
|
||||||
|
|
||||||
|
@mock.patch('novaclient.client.Client')
|
||||||
|
def test_nova_send_events_raises_connect_exc(self, mock_client):
|
||||||
|
create = mock_client().server_external_events.create
|
||||||
|
create.side_effect = (
|
||||||
|
ks_exc.ConnectFailure, ks_exc.ConnectTimeout, [])
|
||||||
|
self.nova_notifier.send_events([])
|
||||||
|
self.assertEqual(3, create.call_count)
|
||||||
|
|
||||||
@mock.patch('novaclient.client.Client')
|
@mock.patch('novaclient.client.Client')
|
||||||
def test_nova_send_events_raises(self, mock_client):
|
def test_nova_send_events_raises(self, mock_client):
|
||||||
mock_client.server_external_events.create.return_value = Exception
|
mock_client.server_external_events.create.return_value = Exception
|
||||||
|
Loading…
x
Reference in New Issue
Block a user