pyzmq/docs/source/howto/devices.md

83 lines
3.9 KiB
Markdown

% PyZMQ devices doc, by Min Ragan-Kelley, 2011
(devices)=
# Devices in PyZMQ
```{seealso}
ØMQ Guide [Device coverage](https://zguide.zeromq.org/docs/chapter2/#ZeroMQ-s-Built-In-Proxy-Function).
```
ØMQ has a notion of Devices - simple programs that manage a send-recv pattern for
connecting two or more sockets. Being full programs, devices include a `while(True)`
loop and thus block execution permanently once invoked. We have provided in the
{mod}`devices` subpackage some facilities for running these devices in the background, as
well as a custom three-socket [MonitoredQueue](monitored-queue) device.
## BackgroundDevices
It seems fairly rare that in a Python program one would actually want to create a zmq
device via {func}`.device` in the main thread, since such a call would block execution
forever. The most likely model for launching devices is in background threads or
processes. We have provided classes for launching devices in a background thread with
{class}`.ThreadDevice` and via multiprocessing with {class}`.ProcessDevice`. For
threadsafety and running across processes, these methods do not take Socket objects as
arguments, but rather socket types, and then the socket creation and configuration happens
via the BackgroundDevice's {meth}`foo_in` proxy methods. For each configuration method
(bind/connect/setsockopt), there are proxy methods for calling those methods on the Socket
objects created in the background thread or process, prefixed with 'in\_' or 'out\_',
corresponding to the `in_socket` and `out_socket`:
```
from zmq.devices import ProcessDevice
pd = ProcessDevice(zmq.QUEUE, zmq.ROUTER, zmq.DEALER)
pd.bind_in('tcp://*:12345')
pd.connect_out('tcp://127.0.0.1:12543')
pd.setsockopt_in(zmq.IDENTITY, 'ROUTER')
pd.setsockopt_out(zmq.IDENTITY, 'DEALER')
pd.start()
# it will now be running in a background process
```
(monitored-queue)=
## MonitoredQueue
One of ØMQ's builtin devices is the `QUEUE`. This is a symmetric two-socket device that
fully supports passing messages in either direction via any pattern. We saw a logical
extension of the `QUEUE` as one that behaves in the same way with respect to the in/out
sockets, but also sends every message in either direction *also* on a third `monitor`
socket. For performance reasons, this {func}`.monitored_queue` function is written in
Cython, so the loop does not involve Python, and should have the same performance as the
basic `QUEUE` device.
One shortcoming of the `QUEUE` device is that it does not support having `ROUTER`
sockets as both input and output. This is because `ROUTER` sockets, when they receive a
message, prepend the `IDENTITY` of the socket that sent the message (for use in routing
the reply). The result is that the output socket will always try to route the incoming
message back to the original sender, which is presumably not the intended pattern. In
order for the queue to support a ROUTER-ROUTER connection, it must swap the first two parts
of the message in order to get the right message out the other side.
To invoke a monitored queue is similar to invoking a regular ØMQ device:
```
from zmq.devices import monitored_queue
ins = ctx.socket(zmq.ROUTER)
outs = ctx.socket(zmq.DEALER)
mons = ctx.socket(zmq.PUB)
configure_sockets(ins,outs,mons)
monitored_queue(ins, outs, mons, in_prefix='in', out_prefix='out')
```
The `in_prefix` and `out_prefix` default to 'in' and 'out' respectively, and a PUB socket
is most logical for the monitor socket, since it will never receive messages, and the
in/out prefix is well suited to the PUB/SUB topic subscription model. All messages sent on
`mons` will be multipart, the first part being the prefix corresponding to the socket that
received the message.
Or for launching an MQ in the background, there are {class}`.ThreadMonitoredQueue` and
{class}`.ProcessMonitoredQueue`, which function just like the base
BackgroundDevice objects, but add {meth}`foo_mon` methods for configuring the monitor socket.