[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:
parent
337f499c58
commit
4a1679450d
@ -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()
|
||||||
|
@ -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):
|
||||||
|
@ -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:
|
||||||
|
Loading…
x
Reference in New Issue
Block a user