zeromq: how to prevent infinite wait? zeromq: how to prevent infinite wait? python python

zeromq: how to prevent infinite wait?


If you are using zeromq >= 3.0, then you can set the RCVTIMEO socket option:

client_receiver.RCVTIMEO = 1000 # in milliseconds

But in general, you can use pollers:

poller = zmq.Poller()poller.register(client_receiver, zmq.POLLIN) # POLLIN for recv, POLLOUT for send

And poller.poll() takes a timeout:

evts = poller.poll(1000) # wait *up to* one second for a message to arrive.

evts will be an empty list if there is nothing to receive.

You can poll with zmq.POLLOUT, to check if a send will succeed.

Or, to handle the case of a peer that might have failed, a:

worker.send(msg, zmq.NOBLOCK)

might suffice, which will always return immediately - raising a ZMQError(zmq.EAGAIN) if the send could not complete.


This was a quick hack I made after I referred user938949's answer and http://taotetek.wordpress.com/2011/02/02/python-multiprocessing-with-zeromq/ . If you do better, please post your answer, I will recommend your answer.

For those wanting lasting solutions on reliability, refer http://zguide.zeromq.org/page:all#toc64

Version 3.0 of zeromq (beta ATM) supports timeout in ZMQ_RCVTIMEO and ZMQ_SNDTIMEO. http://api.zeromq.org/3-0:zmq-setsockopt

Server

The zmq.NOBLOCK ensures that when a client does not exist, the send() does not block.

import timeimport zmqcontext = zmq.Context()ventilator_send = context.socket(zmq.PUSH)ventilator_send.bind("tcp://127.0.0.1:5557")i=0while True:    i=i+1    time.sleep(0.5)    print ">>sending message ",i    try:        ventilator_send.send(repr(i),zmq.NOBLOCK)        print "  succeed"    except:        print "  failed"

Client

The poller object can listen in on many recieving sockets (see the "Python Multiprocessing with ZeroMQ" linked above. I linked it only on work_receiver. In the infinite loop, the client polls with an interval of 1000ms. The socks object returns empty if no message has been recieved in that time.

import timeimport zmqcontext = zmq.Context()work_receiver = context.socket(zmq.PULL)work_receiver.connect("tcp://127.0.0.1:5557")poller = zmq.Poller()poller.register(work_receiver, zmq.POLLIN)# Loop and accept messages from both channels, acting accordinglywhile True:    socks = dict(poller.poll(1000))    if socks:        if socks.get(work_receiver) == zmq.POLLIN:            print "got message ",work_receiver.recv(zmq.NOBLOCK)    else:        print "error: message timeout"


The send wont block if you use ZMQ_NOBLOCK, but if you try closing the socket and context, this step would block the program from exiting..

The reason is that the socket waits for any peer so that the outgoing messages are ensured to get queued.. To close the socket immediately and flush the outgoing messages from the buffer, use ZMQ_LINGER and set it to 0..