- Author: Lex Gallon
- Adapted from code by: Malika Nisal Ratnayake
- Supervised by: Alan Dorin, Adel Nadjaran Toosi (Monash University)
To provide a computationally cheap way of identifying bees based on camera input. We use this identification to black out regions of the video where there are no bees, with the aim of reducing the filesize of the output video, which can then be sent on to an accurate machine learning model to track the possible bees.
Ideally, we want to minimise the transmission of pixels that aren't part of a bee (or other insect of interest), but want to be very sure that we are not accidentally omitting any bee pixels. I.e. we lean towards false positives to be safe.
Although we could use something like YOLOv4-Tiny, we've found that simply detecting changing pixels seems to recall almost all bees in a video. That is, even though it will also detect moving flowers and other regions we don't care so much about, we can be sure that almost all bees are detected.
Using such a simple method also has the benefit of not requiring training on a particular dataset. There are a few hyperparameters that need to be tuned, but these can be done manually with little effort.
Most of this section is dedicated how to get this running on a raspberry pi. If you just want to run this on your personal computer, don't be intimidated!
This is how I personally setup the raspberry pi. You may need to make small changes here and there.
All of this was provided by Malika:
- Raspberry Pi 4
- Camera attached to the Pi
- Sandisk ultra microSD card (128 GB)
- Sandisk microSD adapter (since my laptop only has normal-sized SD slot)
- Cygnett 20,000 mAh power bank (ChargeUp Quad)
- Charging cables (USB-A to USB-C for powering the Pi, my own USB-A to USB-A for charging power bank)
Most of these steps are only for setting up the Raspberry Pi for the first time. If you don't need to setup a raspberry pi and just want to run this on a normal computer, just follow the steps to
- clone this repo (17),
- create the conda environment (22),
- create the config file (23).
I'd recommend using miniconda if you haven't installed conda before.
If you're on Windows, you probably want to use git bash.
For simplicity, I'll assume you're using the Lite version of the Raspberry Pi OS, which then requires you to ssh to connect to the Pi. Some steps can be omitted if you are using the desktop version, mainly any steps related to ssh and wifi connection.
- Download and install Raspberry Pi Imager.
- Open Raspberry Pi Imager.
- Select the OS. I used Raspberry Pi OS Lite (64-bit), specifically version 6.1.21-v8+. "Lite" means no desktop GUI, so select the normal OS if you want to use a desktop on the Pi.
- Plug in the SD card and select it ("Choose Storage") in Imager.
- Ensure ssh is enabled.
- Set a password for the Pi.
- Optionally change the hostname of the Pi. By default, it's
raspberrypi.local
. - Select "Write" to write the OS to the SD card. Wait for this to finish.
- Create a new empty file called
ssh
to thebootfs
partition of the SD card. - Create a new file called
wpa_supplicant.conf
to thebootfs
partition. Refer to the section below to see what the contents of this file should be. - Edit
cmdline.txt
in thebootfs
partition. Addsystemd.restore_state=0 rfkill.default_state=1
to the end of this file (make sure you add a space!). - Eject the SD card from your computer, and plug it into the Pi.
- Connect the Pi to power. Give it a minute to ensure it has booted.
ssh [email protected]
to connect via ssh (if you changed the Pi's hostname earlier, make sure it matches here). Accept any prompts and enter the password you set earlier when asked. You should now be logged in as the userpi
on the Pi!sudo apt-get update && sudo apt-get upgrade
to update any existing software.sudo apt-get install git
to install git.git clone https://github.com/byebrid/bees-edge.git
to clone this repo. You may need to enter your GitHub username and password (use an access token!).wget https://github.com/conda-forge/miniforge/releases/latest/download/Miniforge3-Linux-aarch64.sh
to download the miniforge3 installer. Note you may need to select a different installer if you chose to install a different OS.chmod u+x Miniforge*
to make the installer exectuable../Miniforge*
to run the installer. Follow all of its prompts. One prompt will ask if you wantconda
to be initialised whenever you login. Select yes, but note that this makes your shell a little slower to start up.sudo raspi-config
to ensure the camera plays nicely withopencv
. Go to "Interface Options" and enable "Legacy Camera". Reboot as required.cd
into this repo and enterconda env create -f environment.yml
to create the conda environment. Enter yes if a confirmation prompt appears.cp eg_config.json config.json
to create a config file. Editconfig.json
to change any parameters for the program.
This is only needed for a headless OS. If using a desktop Pi OS, you can skip this. Instead, connect your keyboard and monitor to the Pi and connect to Wifi like a normal person.
Replace the ssid
and psk
values with your wifi network name and password (without the "<" and ">", but keep the quotation marks).
You may want to change the country
value too. You must use a ISO 3166-1 alpha-2 country code.
This worked for me, but you may require a more complicated wpa_supplicant.conf
. I can only point you to this Arch Linux page, this man page, and wish you good luck!
country=AU
ctrl_interface=DIR=/var/run/wpa_supplicant GROUP=netdev
update_config=1
network={
ssid="<my (5GHz) WIFI SSID>"
psk="<my WIFI password>"
}
Read this if you're using a headless OS on the raspberry pi and want to connect to multiple wifi networks.
One option I didn't test is simply adding multiple networks to the wpa_supplicant.conf
file (see this stackoverflow post). I didn't do this myself because this file is removed from the boot partition upon booting the Pi, and I wasn't sure if I could edit it in the main filesystem after the fact or create a new copy in the boot partition.
I relied on this ArchLinux wiki page for a lot of this.
Instead, I connected to my Pi via ssh. I then ran
$ wpa_cli -i wlan0
I found that if I didn't specify the wlan0 interface, wpa_cli
would choose the wrong interface and so couldn't detect any wifi networks (see this ArchLinux thread).
Start your mobile phone's hotspot.
At this point, scan for all available wifi networks using
> scan
OK
Wait for the message confirming that the scan finished. Then list all networks using
> scan_results
Search for your phone's hotspot's SSID. If you notice encoded bytes in the SSID (e.g. "\xC3\xA9"), I would recommend just changing your hotspot's name to use safe/boring characters, and do the scan again. Otherwise, you'll want to encode your SSID into hex.
Once you've verified your hotspot is available, do
> add_network
1
> set_network 1 ssid "<hotspot's SSID>"
> set_network 1 psk "<hotspot's password>"
> enable_network 1
add_network
creates a new empty network configuration. You then edit this configuration using those set_network
commands. Note that the ID following set_network
should match that returned by add_network
!
I didn't actually confirm if you need to do enable_network
when already connected to a wifi network, but better to be safe than sorry! This may be needed to automatically connect to your hotspot if your normal wifi network is unavailable.
To finish up,
> save_config
OK
> quit
Now test that everything by disabling your normal wifi network, connecting your personal computer to your mobile hotspot, and verifying that you can still ssh into the Pi. Another way of verifying the Pi has connected is by checking your phone to see how many devices are connected to your hotspot.
Ensure your conda environment is activated:
(base) $ conda activate bees-edge
Then just run app.py
:
(bees-edge) $ python bees_edge/app.py
You should soon see the program tell you where any output will go, amongst other information. If you don't see any error messages, then things are probably working! To make sure things are working, you can wait for confirmation that 'n' frames have been read, and perhaps check the output directory to verify that a video file has been created and has some data (e.g. filesize > 0).
If using a camera as input, enter Ctrl-C to stop at any time. Give it a minute to gracefully exit, else the output video
- I've used the lite OS because it was just easier for me to SSH into the raspberry pi rather than rely on an external keyboard and monitor. The lite version presumably runs a little quicker too.
- I've used the 64-bit OS so I have the option of using more than 3 GB memory for a single process (see Wikipedia - 3 GB barrier).
- You don't need to use conda. You could just use pip, preferably with a virtual environment (e.g. venv). conda is just easy for me to document.
You can either use a live webcam or an existing video file as input. Simply change video_source
in your config to either an integer (i.e. for live camera input, starts from 0) or string (for filepath). For filepaths, you can either specify a single video file, OR a directory of video files, in which case the script will run on each video file found in that directory.
To use a webcam, you must use its "index". This is probably 0 for an inbuilt webcam (that's what I use by default in eg_config.json). For me, the in-built webcam actually registers itself as two devices (see this stackoverflow post for why), so if I want to use my USB webcam, I actually need to provide an index of 2 (not 1).
Files should be easier, just pass in a filepath relative to your current working directory (which should always be the root of this repo!) or an absolute filepath.