[zmq] Dynamic connections send failure

For dynamic connections it is crucial to close connection
and not to have hanging sockets either we have sent message
successfully or not.

eventlet.green.zmq by default blocks the calling thread on sending message
when connection was not established yet (which is correct DEALER
socket behavior though), but socket cannot be closed when we hang on
sending forever (if we never get the valid host to connect).

eventlet also shields EAGAIN exception in default (blocking) sending mode
so we need to use async zmq.NOBLOCK flag to receive this exception
and hanlde it in our own way to not block forever.

Change-Id: Ib561e061c4b20644213c059a8e8d0efd225edea1
Closes-Bug: #1658913
Closes-Bug: #1663459
This commit is contained in:
ozamiatin 2017-02-06 12:45:26 +02:00
parent 337f499c58
commit 4a1679450d
3 changed files with 25 additions and 11 deletions

View File

@ -14,6 +14,8 @@
import logging import logging
import tenacity
from oslo_messaging._drivers.zmq_driver.client.publishers.dealer \ from oslo_messaging._drivers.zmq_driver.client.publishers.dealer \
import zmq_dealer_publisher_base import zmq_dealer_publisher_base
from oslo_messaging._drivers.zmq_driver.client import zmq_receivers from oslo_messaging._drivers.zmq_driver.client import zmq_receivers
@ -55,7 +57,7 @@ class DealerPublisherDirect(zmq_dealer_publisher_base.DealerPublisherBase):
""" """
def __init__(self, conf, matchmaker): def __init__(self, conf, matchmaker):
sender = zmq_senders.RequestSenderDirect(conf) sender = zmq_senders.RequestSenderDirect(conf, async=True)
receiver = zmq_receivers.ReceiverDirect(conf) receiver = zmq_receivers.ReceiverDirect(conf)
super(DealerPublisherDirect, self).__init__(conf, matchmaker, super(DealerPublisherDirect, self).__init__(conf, matchmaker,
sender, receiver) sender, receiver)
@ -90,11 +92,16 @@ class DealerPublisherDirect(zmq_dealer_publisher_base.DealerPublisherBase):
self.receiver.unregister_socket(socket) self.receiver.unregister_socket(socket)
def send_request(self, socket, request): def send_request(self, socket, request):
if request.msg_type in zmq_names.MULTISEND_TYPES: @tenacity.retry(retry=tenacity.retry_if_exception_type(zmq.Again),
for _ in range(socket.connections_count()): stop=tenacity.stop_after_delay(
self.conf.rpc_response_timeout))
def send_retrying():
if request.msg_type in zmq_names.MULTISEND_TYPES:
for _ in range(socket.connections_count()):
self.sender.send(socket, request)
else:
self.sender.send(socket, request) self.sender.send(socket, request)
else: return send_retrying()
self.sender.send(socket, request)
def cleanup(self): def cleanup(self):
self.routing_table.cleanup() self.routing_table.cleanup()

View File

@ -31,8 +31,9 @@ zmq = zmq_async.import_zmq()
class SenderBase(object): class SenderBase(object):
"""Base request/response sending interface.""" """Base request/response sending interface."""
def __init__(self, conf): def __init__(self, conf, async=False):
self.conf = conf self.conf = conf
self.async = async
self._lock = threading.Lock() self._lock = threading.Lock()
self._send_versions = zmq_version.get_method_versions(self, 'send') self._send_versions = zmq_version.get_method_versions(self, 'send')
@ -155,11 +156,12 @@ class RequestSenderDirect(RequestSenderBase):
"msg_version": request.message_version}) "msg_version": request.message_version})
def _send_v_1_0(self, socket, request): def _send_v_1_0(self, socket, request):
socket.send(b'', zmq.SNDMORE) flags = zmq.NOBLOCK if self.async else 0
socket.send_string('1.0', zmq.SNDMORE) socket.send(b'', zmq.SNDMORE | flags)
socket.send(six.b(str(request.msg_type)), zmq.SNDMORE) socket.send_string('1.0', zmq.SNDMORE | flags)
socket.send_string(request.message_id, zmq.SNDMORE) socket.send(six.b(str(request.msg_type)), zmq.SNDMORE | flags)
socket.send_dumped([request.context, request.message]) socket.send_string(request.message_id, zmq.SNDMORE | flags)
socket.send_dumped([request.context, request.message], flags)
class AckSenderDirect(AckSenderBase): class AckSenderDirect(AckSenderBase):

View File

@ -56,6 +56,11 @@ class ZmqSocket(object):
# Put messages to only connected queues # Put messages to only connected queues
self.handle.setsockopt(zmq.IMMEDIATE, 1 if immediate else 0) self.handle.setsockopt(zmq.IMMEDIATE, 1 if immediate else 0)
# Setup timeout on socket sending
if hasattr(self.conf, 'rpc_response_timeout'):
self.handle.setsockopt(zmq.SNDTIMEO,
self.conf.rpc_response_timeout * 1000)
# Configure TCP keep alive # Configure TCP keep alive
keepalive = self.conf.oslo_messaging_zmq.zmq_tcp_keepalive keepalive = self.conf.oslo_messaging_zmq.zmq_tcp_keepalive
if keepalive < 0: if keepalive < 0: