Handle unexpected failures during call monitor heartbeat

Change-Id: Iec04c18ac3565a3610377d94caf128c6704a89eb
Closes-Bug: #1816816
This commit is contained in:
Kenneth Giusti 2019-02-20 12:08:04 -05:00
parent b2439756a0
commit f948e24294
3 changed files with 57 additions and 3 deletions

View File

@ -32,6 +32,7 @@ from oslo_messaging._i18n import _
from oslo_messaging._i18n import _LE
from oslo_messaging._i18n import _LI
from oslo_messaging._i18n import _LW
from oslo_messaging import MessageDeliveryFailure
LOG = logging.getLogger(__name__)
@ -168,9 +169,15 @@ class AMQPIncomingMessage(base.RpcIncomingMessage):
return
def heartbeat(self):
# generate a keep alive for RPC call monitoring
with self.listener.driver._get_connection(
rpc_common.PURPOSE_SEND) as conn:
try:
self._send_reply(conn, None, None, ending=False)
except rpc_amqp.AMQPDestinationNotFound:
# internal exception that indicates queue/exchange gone -
# broker unreachable.
raise MessageDeliveryFailure("Heartbeat send failed")
# NOTE(sileht): Those have already be ack in RpcListener IO thread
# We keep them as noop until all drivers do the same

View File

@ -218,7 +218,16 @@ class RPCDispatcher(dispatcher.DispatcherBase):
'(interval=%(interval)i)' % (
{'method': incoming.message.get('method'),
'interval': cm_heartbeat_interval}))
try:
incoming.heartbeat()
except Exception as exc:
# The heartbeat message failed to send. Likely the broker or
# client has died. Nothing to do here but exit the watchdog
# thread. If the client is still alive (dead broker) then its
# RPC will timeout as expected.
LOG.debug("Call-monitor heartbeat failed: %(exc)s"
% ({'exc': exc}))
break
def dispatch(self, incoming):
"""Dispatch an RPC message to the appropriate endpoint method.

View File

@ -13,6 +13,7 @@
# under the License.
import testscenarios
import time
import oslo_messaging
from oslo_messaging import rpc
@ -180,7 +181,8 @@ class TestDispatcher(test_utils.BaseTestCase):
dispatcher = oslo_messaging.RPCDispatcher(endpoints, serializer,
self.access_policy)
incoming = mock.Mock(ctxt=self.ctxt, message=self.msg)
incoming = mock.Mock(ctxt=self.ctxt, message=self.msg,
client_timeout=0)
res = None
@ -252,6 +254,7 @@ class TestSerializer(test_utils.BaseTestCase):
incoming = mock.Mock()
incoming.ctxt = self.ctxt
incoming.message = dict(method='foo', args=self.args)
incoming.client_timeout = 0
retval = dispatcher.dispatch(incoming)
if self.retval is not None:
self.assertEqual('s' + self.retval, retval)
@ -265,3 +268,38 @@ class TestSerializer(test_utils.BaseTestCase):
serializer.serialize_entity.assert_called_once_with(self.dctxt,
self.retval)
class TestMonitorFailure(test_utils.BaseTestCase):
"""Test what happens when the call monitor watchdog hits an exception when
sending the heartbeat.
"""
class _SleepyEndpoint(object):
def __init__(self, target=None):
self.target = target
def sleep(self, ctxt, **kwargs):
time.sleep(kwargs['timeout'])
return True
def test_heartbeat_failure(self):
endpoints = [self._SleepyEndpoint()]
dispatcher = oslo_messaging.RPCDispatcher(endpoints,
serializer=None)
# sleep long enough for the client_timeout to expire multiple times
# the timeout is (client_timeout/2) and must be > 1.0
message = {'method': 'sleep',
'args': {'timeout': 3.5}}
ctxt = {'test': 'value'}
incoming = mock.Mock(ctxt=ctxt, message=message, client_timeout=2.0)
incoming.heartbeat = mock.Mock(side_effect=Exception('BOOM!'))
res = dispatcher.dispatch(incoming)
self.assertTrue(res)
# only one call to heartbeat should be made since the watchdog thread
# should exit on the first exception thrown
self.assertEqual(1, incoming.heartbeat.call_count)