Skip to content

use a edge device to gather data managed by edge services in containers

License

Notifications You must be signed in to change notification settings

arduino/arduino-edge-container-demo

Repository files navigation

In this tutorial we'll show you how you can use a Raspberry Pi in conjunction with the Arduino Create platform to create some cool projects involving face detection, Node-RED flows and Mqtt messages! The tutorial will be divided into seven iterations. In each iteration we will add a new ingredient to the project in order to achieve our final goals at Iterations 6 and 7.

1st Iteration: Raspberry Pi setup

In the first iteration we'll setup our Raspberry Pi and test that the GPIOs are working. We'll be using Raspbian Stretch Lite, which is the newest version of Raspbian at the current time, installed on a Raspberry Pi 3b+. We'll use the Lite version because we don't need the GUI. It's easier to work on it if you have SSH enabled. To achieve that goal you have to create an empty file named ssh (with no extension) and put it in the boot partition of the microSD card.

After powering up the Raspberry, we need to get its IP address in order to ssh into it. We have two option for this:

  • We can connect the Raspberry Pi to our pc using an Ethernet cable (here are some guides for Windows 10 and Ubuntu 16.04);
  • Or we can connect via WiFi (but we will need keyboard and display). Once logged in, just run sudo ifconfig and read the IP address next to the wlan0 entry. Now that we now the Raspberry IP, we can control it (in a bash window) using SSH with the command:
$ ssh pi@<ip_address_raspi>

The default log-in informations are: user: pi, password raspberry.

GPIO Libraries

At this point we need to check if the GPIO libraries for Python are already installed/updated.

$ sudo apt-get update
$ sudo apt-get install python-rpi.gpio python3-rpi.gpio

Then, we can write two simple Python scripts to check the status of a pushbutton (connected to Pin 8 - i.e. GPIO 14, see the Raspberry pinout below) and to write a message on the terminal (pushbutton.py and pushbutton_event.py in the repo linked at the end of the tutorial). The second script uses the events related to GPIO pins to call a function which prints the button status on the terminal. This is a more complex way, but the most efficient one.

Let's create a new folder to save the scripts:

$ cd ~
mkdir python-data

Now transfer the two script files to the Raspberry Pi:

  • On Linux, we can use the scp command from our local machine (if you copy bash command from here, remember to first clone the github repository in ~/Documents/):
$ scp ~/Documents/first_iteration/pushbutton.py ~/Documents/first_iteration/pushbutton_event.py pi@<ip_address_raspi>:~/python-data/
  • On Windows, we can use an ssh client like PuTTY or WinScp. For a more complete guide on file transfer have a look at this tutorial.

If scp is not working (e.g. we get Permission Denied), we need to change ownership of the destination folder in the Raspberry using the command chown pi python-data.

For all the connections to the Pi, we used the following scheme: pin raspberry If you want further information on the GPIO pins you can read something useful here

To run the scripts simply run this command:

$ python3 ~/Documents/first_iteration/pushbutton.py

or

$ python3 ~/Documents/first_iteration/pushbutton_event.py

2nd Iteration: adding Docker

After installing Raspbian and testing the GPIO pins, let's add Docker!

First of all, we need to bind the Raspberry Pi to our Arduino account by following the Getting Started flow. Create a new account, if you don't have one yet.

Select RaspberryPi

Select Raspberry Pi in the device lists, then follow the instructions...

Click Next Click My device is ready

Select option A and enter the IP addres of the Raspberry (which we know from the 1st Iteration).

Choose option A

Now we will be prompted with a modal asking for the Raspberry Pi credentials (if you did not changed them, default are pi and raspberry).

Insert credentials

Doing this will install some software on the Raspberry Pi, including the Arduino Connector and Docker.

If we don't want to use sudo each time we use docker, we need to create a group named docker and add our user to it:

$ sudo groupadd docker
$ sudo usermod -aG docker $USER

For our project we need Node-RED, which is a flow-based development tool for wiring together hardware devices, APIs and online services as part of the Internet of Things.

We can use this Node-RED Docker container which is already configured for GPIO use. To install and run the container we can execute the following command:

$ docker run -d -p 1880:1880 -v ~/node-red-data:/data --privileged --name mynodered nieleyde/rpi-nodered-gpio:latest

Here's a brief explanation of it:

  • run -d nieleyde/rpi-nodered-gpio:latest This command will download the container from DockerHub (if it's not already been downloaded) and run it in background;
  • -p 1880:1880 This option exposes 1880 port outside the container;
  • -v ~/node-red-data:/data This option mounts the host’s ~/node-red-data directory as the user configuration directory inside the container. It's useful to backup flows.json file. This file contains the configuration of all flows created in Node-RED browser editor;
  • --privileged This option allows the container to access to all devices, in particular to Raspberry's GPIO pins;
  • -- name mynodered This option gives a human readable name to the container;

Now we can check the status of the container with docker ps: it should be up. Furthermore we can stop the container with docker stop mynodered and start it again with docker start mynodered.

It's also possible to modify the flow and nodes through Node-RED's browser web interface, connecting to this URL: http://<ip_address_raspi>:1880/.

Finally it's also possible to restore the backup of Node-RED's nodes and flows using scp to copy flows.json in the folder node-red-data created before.

$ scp ~/Documents/second_iteration/flows.json pi@<ip_address_raspi>:~/node-red-data/

The example flows.json in this iteration simply reads the value of a GPIO input pin (Pin 8) and prints its values in the debug tab.

Second Iteration alternative

For the 2nd Iteration we could have used the official container of Node-RED. To make it work, we need to use this command:

docker run -d -p 1880:1880 -v ~/node-red-data:/data --user=root --privileged --name nodered nodered/node-red-docker:0.18.7-rpi-v8

It is better to use this container because it is the official one, so it's better mantained and more updated in comparison to the one we used previously. We need to run this container with --user=root beacuse in its Dockerfile the developers created the user node-red which cannot access /dev/mem (to control the GPIO pins). Further information about users here.


3rd Iteration: using Arduino Web Editor

Our goal in this iteration is to write an Arduino sketch which reads the value of a pushbutton and notifies Node-RED inside the Docker container. The flow in Node-RED has to rework the value received from the sketch and send an email to a speicifed email address.

For this step there's no need of configuring anything. We simply have to import and flash the arduino sketch on the Raspberry Pi using the Arduino Web Editor and import the flow in Node-RED.

We can upload the sketch in two different ways:

  • By downloading the sketch pushbutton_http_post.ino and importing it into the editor;

Import sketch

Add to my sketchbook

The sketch reads the status of the pushbutton (connected to Pin 8) and, if it's pressed, performs an HTTP POST request using the bash command curl. The request is directed to Node-RED's ip and port.

On the Node-RED side we have to import the flow located in the third_iteration folder (in the github repo):

  • We can do that as described in the 2nd Iteration, but copying this new flow:
$ scp ~/Documents/third_iteration/flows.json pi@<ip_address_raspi>:~/node-red-data/
  • Or you can use a simpler solution: open the flows.json file and copy all its content to the clipboard, then use Node-RED import function (toast menu > import > clipboard).

The flow for this iteration should look like this:

To make it work we need to add the properties inside the e-mail node (recipient, e-mail address and password).

If using a gmail account it may be necessary to disable less secure apps here.

Now it's time to deploy the flow and test it!


4th Iteration: the OpenCV library

In this iteration we are going to test OpenCV, which is a computer vision library written in C/C++ supporting Python. Our goal for now is to install OpenCV, take a photo and save it using this library. Since it's not an easy task to install it, we can use a docker container already built! We can use this one which comes already compiled for Raspberry Pi and Raspbian Stretch.

To install it use this command:

$ docker run -it --privileged -v ~/opencv-data:/data --name opencv sgtwilko/rpi-raspbian-opencv:stretch-latest

The options used in this command were already described in the 2nd Iteration. By the way, the option --priviliged let the container access all the peripherals connected to our Raspberry, even the webcam (or Raspberry Pi Cam).

As usual we can stop the container with docker stop opencv and start it again with docker start -i opencv (option -i attaches container's STDIN).

Now we should be in the container's bash! First of all let's check if OpenCV is installed by running Python interpeter:

$ python
Python 2.7.13 (default, Nov 24 2017, 17:33:09) 
[GCC 6.3.0 20170516] on linux2
Type "help", "copyright", "credits" or "license" for more information.
>>>

And then try to import OpenCV library:

>>> import cv2
>>> 

If there isn't any error, we can move on to the next part. To exit the interpreter use exit().

>>> exit()

The python script in the github repo (test.py) simply takes a photo using the webcam and saves it in /opencv-data/opencv.png.

As previously described, we can copy the script into the Raspberry using scp:

scp ~/Documents/fourth_iteration/test.py pi@<ip_address_raspi>:~/opencv-data/

To run the script, let's start the docker container (docker start -i opencv), move to the correct directory (cd opencv-data/) and finally run the script with python test.py.

We can open and see the photo if we copy it back to our pc:

scp pi@<ip_address_raspi>:~/opencv-data/opencv.png ~/Documents/

5th Iteration: doing some face detection

In this iteration we are going to do some face detection! This step clearly requires to have OpenCV installed on the Raspberry Pi, as shown in the 4th Iteration.

Basically, face detection will detect if a photo contains a face and will find the position of the face, highlighting it with a green rectangle (see figure below).

Before OpenCV After OpenCV
Before After

OpenCV uses a cascade file (an .xml file) which contains data to detect objects, in our case faces. We only have to initialize it in our code. OpenCV comes with different built-in cascades for detecting several things such as eyes, hands, legs, animals and so on... In this tutorial we are going to use Haar Cascade Classifier. It is a machine learning based approach where a cascade function is trained from a lot of positive (images with faces) and negative (images without faces) images.

So, we need to copy the cascade file (haarcascade_frontalface_alt.xml) to the Raspberry:

scp ~/Documents/fifth_iteration/haarcascade_frontalface_alt.xml pi@<ip_address_raspi>:~/opencv-data/

The we can copy some test images to the Raspberry:

scp ~/Documents/fifth_iteration/test.jpg pi@<ip_address_raspi>:~/opencv-data/

The python script in the github repo (face_detection.py) opens a photo passed as argument, tries to find faces and saves the image with detected faces in the same folder. We can copy the script to the Raspberry with:

scp ~/Documents/fifth_iteration/face_detection.py pi@<ip_address_raspi>:~/opencv-data/

Now that everything is configured, let's start the script! Similarly to the previous iteration, we have to start the docker container first (docker start -i opencv), then we move to the correct directory (cd opencv-data/) and finally we run the script (python face_detection.py test.jpg).

We can open and see the modified photo by copying it to our pc:

scp pi@<ip_address_raspi>:~/opencv-data/test_ocv.jpg ~/Documents/

6th Iteration: putting it all together

In this iteration we are going to put together all the previous iterations.

scheme

This is the idea:

  • An Arduino sketch will run every 15 seconds a shell command to start a docker container;
  • The docker container which contains OpenCV will run a python script on startup. This script will take a photo using a webcam, will detect faces in that photo and will save the image (with detected faces) in a docker shared volume with the name opencv.png if at least one face is found. After all that, the container will stop;
  • In another docker container, a flow in Node-RED will watch the shared volume, waiting for changes in the file opencv.jpg. If the file has changed (i.e. the photo has been overwritten), the flow composes an e-mail (attaching the photo) and sends it to a specified e-mail address.

Arduino sketch

Let's start with the Arduino code. The upload procedure to the Raspberry has already been explained in the 3rd Iteration. The code can be found at the end of the tutorial, or directly here.

Docker container for OpenCV

The OpenCV docker container part is a bit more complex. First of all we have to create a docker volume to share data between the two containers:

docker volume create share

To check if everything is ok we can check the list of volumes: we should see the volume we just created with docker volume ls.

We can create a docker image to simplify the procedure, but we could also build the image by ourselves as explained in the section Docker build OpenCV below. With Docker already installed on the Raspberry, just download this image by running the command:

    docker run -it --privileged -v share:/data --name face_detection umbobaldi/opencv_face_detection:1.0

Some notes:

  • -it let us see a short log and check if everything is ok;
  • -v share:/data mounts the docker volume share and binds it with /opencv-data folder inside the container.

The downloaded image is already configured to run on startup a python script which uses OpenCV. This script does the following:

  • It takes a photo using the webcam;
  • It does some face detection, as described in the 5th Iteration;
  • It saves the photo (opencv.png) in the opencv-data folder, which is connected to the shared volume, if at least one face is found. Otherwise nothing is saved.
  • When the script finish the execution, the docker container is stopped.

Docker container for Node-Red

In the end we can import flows.json as described in the 3rd Iteration. Let's open the file and copy all its content to the clipboard, then use Node-RED import function (toast menu > import > clipboard). To make it work we need to add the properties inside the e-mail node (recipient, e-mail address and password).

Here's the resulting flow:

This flow does the following:

  • The first node waits for changes in the file opencv.png located in the shared volume. When there is a change it means the old file has been overwritten by a newer file, so it's time to send the e-mail;
  • The delay node has to filter the input: when the OpenCV container overwrites the file the process is not immediate, so the messages have to be limited to one;
  • The function node composes the e-mail attaching the photo and writing a proper message;
  • Finally, the e-mail is sent by the e-mail node.

Docker build for OpenCV

In case we want to build the docker image by ourselves, we have to move to the folder where the Dockerfile is located, and run the command:

docker build -t myopencv:<version> .

This command will build our image. To check if everything is ok we can list all docker images with docker images -a. If the new image is listed, we can run it the first time using: $ docker run -it --priviliged -v share:/

docker run -it --privileged -v share:/data --name face_detection myopencv:<version>

If it is not the first time, just run it using

$ docker start -i face_detection`

The image is modeled after this one which has already OpenCV installed. The Dockerfile creates two folders: one for the code and the other one for sharing the photo. Then it moves camera.py and haarcascade_frontalface_alt.xml inside the newly created folder inside the container (/usr/scr/ocv_face_detection). In the end it starts the python script.

7th Iteration: Mqtt meassages from an Arduino MKR WiFi 1010

The goal in this last iteration is to send and receive Mqtt messages from an Arduino MKR WiFi 1010. The Arduino has to publish Mqtt messages on a topic when a pushbutton connected to it is pressed. Another Arduino (or the same, in this case) is subscribed to another topic and moves the shaft of a servomotor.

The Arduino is connected via WiFi to our Raspberry which is now configured as an access point. The Raspberry has to run an Mqtt Broker (Mosquitto in our case). We can install and run this broker using a docker. In the end, a Node-RED flow is responsible for the logic.

Arduino sketch

The Arduino sketch simply connects to the Raspberry via WiFi, then connects to the Mqtt Broker in the docker container running on the Raspberry and finally publishes a message on /arduino/button topic when the pushbutton is pressed.

The same Arduino is subscribed to /arduino/servo topic: when a message is published on this topic, the arduino activates a servomotor. If the WiFi connection is working the built-in LED is turned on.

To program and flash the code, use the Arduino Web Editor. For uploading the sketch to the Arduino, we need to connect the board to our pc, install the Arduino Create plugin and follow the 3rd Iteration to import the sketch arduino_button_servo_mqtt.ino (direct link to the sketch here).

The circuit chematics is shown below:

wiring

Raspberry Pi configuration

In this iteration we need to configure the Raspberry in such a way that the Arduino board can connect to it via WiFi. The idea is to make the onboard wireless card act like an access point and not as a client. To achieve that we can follow this guide or, better, use this tool which simplifies the process a lot. First of all we need to install all its dependencies. Then we can clone the repo and install the script. To set up the access point, we use this command:

$ sudo create_ap -n --no-virt  wlan0 RaspberryPi3b+_Net RaspberryPi

Here's a brief explanation of the above command:

  • -n is for disabling Internet sharing;
  • --no-virt is for not creating virtual interface;
  • wlan0 is the name of the interface;
  • RaspberryPi3b+_Net is the SSID of the wireless network;
  • RaspberryPi is the passphrase.

Node-RED flow

Node-RED is responsible for the logic behind this iteration. The idea is to make a flow which is able to publish on /arduino/servo when it receives a message on another topic (/arduino/button). The flow must be subscribed to this second topic to achieve that. We can import flows.json as described in the 3rd Iteration in order to define the Node-RED flow. It should look like this:

Mosquitto docker

Finally we can install our Mqtt Broker. We used Mosquitto beacuse it is opensource, easy to setup and simple. We can install it through a docker container since it's easier this way, using this image which comes preconfigured. The installation is similar to the one described in the 1st Iteration. To run it the first time, use:

$ docker run -d -p 1883:1883 --name mqtt mjenz/rp-mosquitto

while the other times, use:

$ docker start mqtt

At this point, all that remains to do is to deploy the flow and test the project!

About

use a edge device to gather data managed by edge services in containers

Topics

Resources

License

Security policy

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published

Contributors 3

  •  
  •  
  •