DIY Camera with Raspberry PI & WebCam

DIY Camera with Raspberry PI & WebCam

Do you want to convert your old web camera as a smart surveillance Camera? Learn how you can connect your old webcam to Raspberry Pi and make it a smart surveillance system here.

Using USB Camera/Webcam with Home Assistant

This article explains how you can connect a USB based camera module to your Raspberry Pi, and bring in motion sensing capabilities/real-time streaming into your home automation system. The software used to interact with USB camera is motion. The motion program has so many features - including saving images when there is motion detected, run an external program indicating the same, create video clips of all recordings (by stitching all the images together), make time-lapse pictures, and also has a built-in web server for easier access to the camera via a web url…and more! There are so many advanced features that are available, unfortunately this article covers only some of it.

The design goes something like the following picture. The camera is connected to a Raspberry Pi using USB interace, and the Raspberry Pi will be running motion program. The program is run as daemon, so that when Raspberry Pi boots, the program starts automatically. The motion program will be constantly streaming the video (via Web URL) and whenever the program detects significant movement (difference between frames), it not only saves the images, but also calls an external python program - which in turn drops a message into MQTT signaling that the motion has been detected. The home automation system (Home Assistant) allows for easy integration with MQTT, and has the capability to create automatic binary_sensors and automations. In this case, MQTT makes it easy to integrate between motion and Home Assistant.

Camera Setup

Now that you know how this will all come together, it is time to install and configure basic Raspberry Pi stuff. If you are unfamiliar with setting up Raspberry Pi, you can find it online at Raspberry Pi’s web site. After installing Raspbian Operating System on your Raspberry Pi, connect USB camera to Raspberry Pi and run the following command.

lsusb

The above command shows you a list of USB devices connected to your Raspberry Pi. If you see the camera there, you are good to go. If you don’t your camera may not be supported.

Now that you know the camera is recognized by your Raspberry Pi, it is timeto install software. The sofware that is commonly used is called Motion. To install, run the command.

sudo apt-get install motion

After the installation, you need make a few modifications to the settings, and the configuration settings for motion is in /etc/motion/motion.conf file. So, go ahead and open the file.

  • Make sure daemon is ON.
  • Set framerate anywhere in between 1000 to 1500.
  • Keep stream_port to 8081.
  • stream_quality should be 100.
  • Change stream_localhost to OFF.
  • Change webcontrol_localhost to OFF.
  • Set quality to 100.
  • Set width & height to 1280 & 720.
  • Set pre_capture to 2.
  • Set post_capture to 5.
  • set target_directory to the location of your choice - I use /home/pi/Pictures

Save and exit the file. By the way, you can change the resolution to what ever you need depending on your camera’s maximum supported resolution. Here is a copy of my motion.config for your reference: click here to view my motion.conf file

After making changes, you also need to make changes to one more file. Type in the command sudo nano /etc/default/motion and press enter.

sudo vi /etc/default/motion
  • Set start_motion_daemon to yes. Save and exit.

Use the folowing commands to start, stop, and restart motion service.

To start motion service, run the following command.

sudo service motion start

To stop motion service, run the following command.

sudo service motion stop

To restart motion service, run the following command.

sudo service motion restart

To check the status of motion service, run the following command:

sudo systemctl status motion

After running the service using sudo motion, if you run into any errors, it may be the permissions error. See Permissions section below.

Checking the camera images online:

Go to the ip address of the Raspberry Pi, and access the url as below:

http://192.168.x.xxx:8081

The port number 8081 can be changed to whatever you need. It is specified in the /etc/motion/motion.conf file.

Permissions

The motion service requires write access to several folders. The service also runs under user motion and group motion. Make sure you give the following folders access by running the command.

sudo chown motion:motion /var/run/motion
sudo chown motion:motion /var/log/motion
sudo chown motion:motion /var/log/motion/motion.log
sudo chown motion:motion /var/lib/motion 

Since I used /home/pi/Pictures folder as the target_directory, I had to give permissions to the folder using the following command.

sudo chown motion:motion /home/pi/Pictures
sudo service motion restart

I also installed Samba on the Raspberry Pi and shared the /home as a share, so that I can access the images and the videos using UNC path from Windows. for ex: \\192.168.xxx.xxx\pi\Pictures\

Home Assistant Integration

When you use the platform mjpeg in Home Assistant, by default, it is only limited to bringing camera image into the Home Assistant UI/dashboard. To bring the camera image into Home Assistant, add the following to your configuation.yaml file. This should show you the camera image on the Home Assistant dashboard.

camera:
  - platform: mjpeg
    mjpeg_url: http://192.168.xxx.xxx:8081
    name: Garage Camera

Creating [Motion] Sensors in Home Assistant

To create sensors/alerts whenever motion is detected in Home Assistant, we need to configure some things. The motion component has built-in hooks that you can leverage to identify when the motion is detected. Please note that it is not a PIR (passive infra red) sensor, it only checks for differences from frame to frame, and whenever there is a “significant” difference is noticed, the software can notify you in the form of hooks. The /etc/motion/motion.conf file contains those hooks. The on_motion_detected is a hook, where you can run your own command, and motion makes sure it runs it. In this case, we will write a python code that simply drops “on” message into a mqtt topic whenever there is a motion detected.

The configuration goes as below:

 on_motion_detected /usr/bin/python3 /home/pi/motion_on_mqtt.py

The python component that publishes “on” message to a specified mqtt topic is below:

import os
import time
import paho.mqtt.client as mqtt

MQTT_SERVER = "192.168.xxx.xxx"
MQTT_SERVER_PORT = 1883
MQTT_TOPIC = "/garage/motion"
MQTT_USERNAME = "xxx"
MQTT_PASSWORD = "xxx"

# Define event callbacks
def on_connect(mosq, obj, rc):
    print("Return Code On Connect: " + str(rc))

def on_log(mosq, obj, level, string):
    print(string)

mqttc = mqtt.Mosquitto()

# Assign event callbacks
mqttc.on_connect = on_connect

# Uncomment to enable debug messages
#mqttc.on_log = on_log

# Connect
mqttc.username_pw_set(MQTT_USERNAME, MQTT_PASSWORD)
mqttc.connect(MQTT_SERVER, MQTT_SERVER_PORT)

# publish the message
mqttc.publish(MQTT_TOPIC, 'on')

before you move on, make sure you change the ip address, port, username and password of your MQTT server in the above code.

Copy the contents above and place in a file motion_on_mqtt.py and place that file in /home/pi folder. Since the motion program runs under motion user, you need to give permissions to that file for motion user. You can do so by running:

sudo chown motion:motion /home/pi/motion_on_mqtt.py

If you do not have MQTT python libraries (not the MQTT client software) installed, you might need that for your python program to work. The following commands will let you install paho mqtt libraries on your machine.

sudo apt-get install mosquitto mosquitto-clients -y
pip install paho-mqtt
pip3 install paho-mqtt python-etcd
sudo -u motion pip install paho-mqtt

After installing the software, make sure you restart motion program by running the command

sudo service motion restart

You can now switch the context to Home Assistant. In Home Assistant code, add the following code:

binary_sensor:
  - platform: mqtt
    state_topic: "/garage/motion"
    name: "Garage Motion"
    device_class: occupancy
    payload_on: 'on'
    payload_off: 'off'
    value_template: "{{ value }} "

automation:

  - alias: Garage Motion Reset
    trigger:
      - platform: state
        entity_id: binary_sensor.garage_motion
        to: 'on'
        from: 'off'
    action:
      - delay: '00:00:30'
      - service: mqtt.publish
        data:
          topic: "/garage/motion"
          payload: 'off'
          retain: false

Save the changes, and restart Home Assistant. That should show you a binary sensor on you dashboard, that will change its status whenever the camera detects motion.

Clearing Files

The motion program by default saves all the images. You can change the setting output_pictures on in the config file. If not, you can write a simple script that runs on a fixed interval that clears all the images that the program stored.

I use a simple script (below) that I schedule a cron job */5 * * * * /home/pi/delete_pics.sh to run on a fixed interval to clear disk space.

#!/bin/bash
fileCount="$(ls -1 /home/pi/Pictures | wc -l)"
if [ $fileCount -gt 100 ]
then
sudo rm -r /home/pi/Pictures/*.*
fi

Troubleshooting

There are many places where things can go wrong, please be careful in ensuring you have the software installed and configured properly. Also make sure the motion program that rung under motion user has access to the folders, paths, files.

The python program doesn’t seem to be working

The python program requires Paho MQTT libraries installed on the system, and the user motion should be able to run it without any problems. If you can make sure the user motion can run the python program in the CLI (Command Line Interface), it should work as a daemon. To verify that, run the program as following:

sudo -u motion /usr/bin/python3 /home/pi/motion_on_mqtt.py

The command above should run successfully and you should see a “on” message in the MQTT - provided you configured MQTT server settings correctly.

MQTT Server

In case of not seeing messages in MQTT, make sure the MQTT server is set up and configured correctly. The user name, password, server Ip address, port all should be double checked.

If possible, run a test command where you can drop a simple message into your mqtt using the following command.

mosquitto_pub -h 192.168.xxx.xxx -p 1883 -t /test -m "on" -u USERNAME -P PASSWORD

Still having issues?

Instead of running the python program, try running a shell script that internaly calls the python program. Inside the shell script, put >/tmp/motion.log 2>&1 at the end of command to log all the error messages and direct them into a log file. for ex:

Before running the command, make sure the user motion has write access to the /tmp/mtion.log file. If not, run the command sudo chown motion:motion /tmp/motion.log

Sample script.sh contents

#!/bin/sh

echo "Running python program" >> /tmp/motion.log
/usr/bin/python3 /home/pi/motion_on_mqtt.py >> /tmp/motion.log 2>&1

Run the script as follows, and check the /tmp/motion.log for any errors.

sudo chown motion:motion script.sh
sudo chmod +x script.sh

sudo -u motion script.sh

Some things to remember:

  • Make sure you pay extra attention to the permissions!
  • Restart motion service after every change to the /etc/motion/motion.conf file

Hope by now, you got it all working…

After making changes, restart the Home Assistant, and if everything runs as expected, enjoy! If not, you may want to go back to googling stuff!

comments powered by Disqus