Skip to content

Latest commit

 

History

History
255 lines (209 loc) · 12.6 KB

more-details.rst

File metadata and controls

255 lines (209 loc) · 12.6 KB

More details about the multiple RPi video streaming example

The beginning of the README.rst for this GitHub repository showed a screenshot of a Mac computer displaying simultaneous video streams from 8 Raspberry Pi cameras:

images/screenshottest.png

This display was generated by running test_2_send_images.py simultaneously on 8 Raspberry Pi computers with PiCamera modules and then running test_2_receive_images.py on a to receive all of those image streams and display them single Mac. Here is a more detailed description of the test programs and how they work. The example uses the ZMQ REQ/REP messaging protocol which is the imageZMQ default. There is also an example below describing a matching pair of programs that can generate the same display by using the PUB/SUB messaging pattern.

The Raspberry Pi computers send images and the Mac computer displays them. The imageZMQ python classes transport the images between the computers. There can be a single Raspberry Pi sending images, or there can be 8 or more. There is always exactly one computer (a Mac in this example) receiving and displaying the images. In ZMQ parlance, the Raspberry Pi computers are acting as clients sending requests (REQ) and the Mac is acting as the server sending back replies (REP). Each "request" from the Raspberry Pi is a 2 part message consisting of an OpenCV image and the Raspberry Pi hostname. Each "reply" sent by the Mac message is an "OK" that tells the Raspberry Pi that it can send another image. It is a very important imageZMQ design element that each message is a (text_message, image) tuple. If the imageZMQ messages were images only, there would be no way to identify and distinguish multiple image streams.

ZMQ is a powerful messaging library that allows many patterns for sending and receiving messages. imageZMQ provides access to both REQ/REP and PUB/SUB ZMQ messaging patterns. The 2 patterns differ in how they are used, as described below. Pay particular attention to the way the addresses of the senders and hubs are handled in the 2 patterns. The "8 RPi's streaming to and displaying on 1 Mac" example above can be done with either the REQ/REP messaging patern or the PUB/SUB messaging pattern. But the tcp addresses are specified differently in each.

When using the REQ/REP (request/reply) pattern, every time a Raspberry Pi sends an image, it waits for an "OK" reply from the Mac before sending another image. It also means that there can be multiple Raspberry Pi computers sending messages to the Mac at the same time, since the ZMQ REQ/REP pattern allows many clients to send REQ messages to a single REP server. With each image sent, the Raspberry Pi sends an identifier (the Raspberry Pi hostname, in these test programs), so that the Mac can display the images from each Raspberry Pi in a different window using cv2.imshow(). This works because cv2.imshow() opens a new display window for each image based on a unique name. That way all the images for "RPi10" show up in 1 window, and all the images for "RPi06" show up in another windwow, etc. In larger imagenode / imageZMQ / imagehub projects, this same approach uses the text portion of each imageZMQ message pair to specify image source and other image properties.

Let's look at the Python code in the Raspberry Pi sending program:

 1   # run this program on each RPi to send a labelled image stream
 2   # you can run it on multiple RPi's; 8 RPi's running in above example
 3   import socket
 4   import time
 5   from picamera2 import Picamera2
 6   import imagezmq
 7 
 8   sender = imagezmq.ImageSender(connect_to='tcp://jeff-macbook:5555')
 9 
10   rpi_name = socket.gethostname() # send RPi hostname with each image
11   picam = Picamera2()
12   picam.start()
13   time.sleep(2)  # allow camera sensor to warm up
14   while True:  # send images as stream until Ctrl-C
15       image = picam.capture_array()
16       sender.send_image(rpi_name, image)

Lines 2 to 5 import the Python packages we will be using. Line 7 instantiates an ImageSender class from imageZMQ. Line 9 sets rpi_name to the hostname of the Raspberry Pi. This will keep each Raspberry Pi's image stream in a separate window on the Mac (provided that each Raspberry Pi has a unique hostname). Line 10 starts a VideoSteam from the PiCamera. Line 11 allows the PiCamera sensor to warm up (if we grab the first frame from the PiCamera without this warmup time, it will fail with an error). Line 12 begins a forever loop of 2 lines: Line 13 reads a frame from the PiCamera into the image variable. Line 14 uses imageZMQ's send_image method to send the Raspberry Pi hostname and the image to the Mac. These "read and send" lines repeat until Ctrl-C is pressed to stop the program. This effectively sends a continuous stream of images (up to 32 images per second) to the Mac, with each image labeled with the hostname of the Raspberry Pi that is sending it. If there are multiple Raspberry Pi computers sending images at the same time, the Mac receiving the images is able to sort them to labelled windows because of the unique Raspberry Pi hostname sent with each image.

Notice that the "connect_to" address is the address of the Mac hub, which is the same for every RPi, since they are all sending to the same Mac hub. That means we can copy this same program to all 8 RPi's in our demonstration and we only need to know one "connect_to" address -- the address of the Mac Hub.

Now, lets look at the Python code on the Mac (or other display computer):

 1   # run this program on the Mac to display image streams from multiple RPis
 2   import cv2
 3   import imagezmq
 4 
 5   image_hub = imagezmq.ImageHub()
 6   while True:  # show streamed images until Ctrl-C
 7       rpi_name, image = image_hub.recv_image()
 8       cv2.imshow(rpi_name, image) # 1 window for each RPi
 9       cv2.waitKey(1)
10       image_hub.send_reply(b'OK')

Lines 2 and 3 import the Python packages we will be using: cv2 (OpenCV) and imageZMQ. Line 5 instantiates an ImageHub class from imageZMQ. Line 6 begins a forever loop: line 7 receives an rpi_name and an image from imageZMQ's recv_image method. Line 8 shows the image in a display window with a window title of rpi_name. Line 9 waits for a millisecond, then line 10 sends the required "reply" back to the Raspberry Pi per the ZMQ REQ/REP pattern. Lines 9 and 10 repeatedly receive and display images as they come in. The cv2.imshow() method displays each image received in a window corresponding to the window name. If all the images come from a single rpi_name, then all the image streams will appear in a single window. But if the income stream has images from multiple rpi_name's, then cv2.imshow() automatically sorts the images by rpi_name into unique windows. Thus, if 3 Raspberry Pi computers are sending images, the images will be displayed in 3 separate windows with each one labelled by its rpi_name. The ZMQ library is fast enough to make these 3 streams of images appear as 3 continuous video streams in separate windows. To create the picture at the top of this page, 8 Raspberry Pi computers were sending images to a single Mac. The picture is a screenshot of the Mac's display with the 8 cv2.imshow() windows arranged in 2 rows.

Notice that we do not have to specify any "connect_to" address for the Mac hub. The default localhost address is fine and is the same for every RPi that will be connecting to this Mac in the REQ/REP messaging pattern. The way addresses are specified is an important difference between the REQ/REP messaging pattern and the PUB/SUB messaging pattern.

The above example that uses REQ/REP pattern has one important feature that can be a huge disadvantage in certain scenarios: sending images in this pattern is a blocking operation.

This means that if a Hub stops responding or simply disconnects the sender will stop at the send_image() method until it receives a REP response from the Hub. This is useful if the sender wants explicit acknowledgement of every single frame that is sent. But it can cause the sender to freeze up if there is any problem with the Hub or the network. The application code for any image sender using REQ/REP must include specific code to deal with any lack of a timely response from the Hub.

If this is not desirable in your application, you can use PUB/SUB (publish/subscribe) pattern. Subscribers can connect and disconnect to publisher (sender) at any time. No REP reply is sent or expected in the PUB/SUB messaging pattern.

When using PUB/SUB mode, the image sender creates a ZMQ PUB socket, but images are pushed to the socket only if at least one subscriber is connected to this socket. If there are no subscribers, then the images are discarded immediately and execution continues.

Here is a PUB/SUB example. The code of the sender is pretty similar to the previous REQ/REP example:

 1   import socket
 2   import time
 3   from imutils.video import VideoStream
 4   import imagezmq
 5 
 6   # Accept connections on all tcp addresses, port 5555
 7   sender = imagezmq.ImageSender(connect_to='tcp://*:5555', REQ_REP=False)
 8 
 9   rpi_name = socket.gethostname() # send RPi hostname with each image
10   picam = VideoStream(usePiCamera=True).start()
11   time.sleep(2.0)  # allow camera sensor to warm up
12   while True:  # send images until Ctrl-C
13       image = picam.read()
14       sender.send_image(rpi_name, image)
15       # The execution loop will continue even if no subscriber is connected

Notice that there is different pattern for the connect_to argument. It does not need to specify a specific address for the Hub, because the hub will the side doing the job of connecting to this sender. Which, of course, means that the hub will need the address of this sender and also the address of every other sender.

Notice we also have a new REQ_REP=False argument in line 8. Since REQ/REP is the default argument in imageZMQ, this is the way to specify PUB/SUB as the desired messaging protocol.

Receiver Hub example code:

 1   import cv2
 2   import imagezmq
 3 
 4   # Instantiate and provide the first sender / publisher address
 5   image_hub = imagezmq.ImageHub(open_port='tcp://192.168.1.100:5555', REQ_REP=False)
 6   image_hub.connect('tcp://192.168.0.101:5555')
 7   # image_hub.connect('tcp://192.168.0.102:5555')  # must specify address for every sender
 8   # image_hub.connect('tcp://192.168.0.103:5555')  # repeat as needed
 9 
10   while True:  # show received images
11       rpi_name, image = image_hub.recv_image()
12       cv2.imshow(rpi_name, image) # 1 window for each unique RPi name
13       cv2.waitKey(1)

The receiver code is very similar to REQ/REP example, however there are several important differences.

Note that in Line 7, we have to know IP address of the sender in order to connect to it. In REQ/REP case the direction of connection was opposite - the sender had to know address of the recipient. Also, we must use REQ_REP=False parameter to specify that we are using the PUB/SUB messaging pattern.

Also note that we have no send_reply line like the image_hub.send_reply(b'OK') line in the REQ/REP example. The PUB/SUB messaging pattern does not send or expect REP replies.

Also note that we need to specify EVERY IP address for EVERY sender we wish to subscribe to. To duplicate the original example of having 8 RPi's sending images to a single Mac Hub, we will need the 8 RPi address. So in the case of the REQ/REP pattern we only need to know 1 IP address: the address of the Mac Hub, which is the same for every RPi sender. But in the PUB/SUB messaging pattern, the ImageHub must know the address of every PUB sender. The first PUB sender address is specified in the ImageHub instantiation on line 7. The remainder of the PUB sender addresses are specified using a ImageHub connect method, as illustrated in line 8. It would take 7 additional lines of code that specifiy the addresses of all the RPi's to replicate our example displayed above.

The REQ/REP and PUB/SUB messaging patterns both have advantages and disadvantages. You can learn more about them here: REQ/REP versus PUB/SUB Messaging Patterns

Return to main documentation page OR Return to examples documentation page.