To Boil or Not to Boil? That Is the Question!

by madmcu in Circuits > Arduino

114 Views, 1 Favorites, 0 Comments

To Boil or Not to Boil? That Is the Question!

VID20240604102754-ezgif.com-optimize (1).gif

When preparing the lunch, everyone has probably encountered the moment when you let the pan on the fire and the water or milk start boiling and coming out of the pan.


To prevent it you could set the fire lower but that is no fun, let me introduce you the overengineered way:

  1. Take an Arduino board and an accelerometer
  2. Use Embedded AI to detect when the water is boiling thanks to the vibration
  3. Start a fan to blow on the pan to prevent the water from coming out of the pan

Pretty simple right :)

Supplies

Hardware:

  • Arduino Uno R4 WiFi
  • Adafruit LIS3DH
  • Blank pcb board
  • micro-USB connector for PCB+ micro-USB to USB-A (or USB-A connector for PCB)
  • TIP120 transistor
  • 2.2KΩ resistor
  • Jumper wires
  • A USB-A fan


Software:

SETUP

IMG20240531165714.jpg
IMG20240531165607.jpg
IMG20240531165622.jpg
IMG20240531165637.jpg
IMG20240531165645.jpg

First, we need to connect the accelerometer to the Arduino board.

Use jumper wires to connect:

  • VIN (mic) to 3.3V (board)
  • GND to one of the GND on the board
  • SDA to SDA
  • SCL to SCL


Then we replicate this mintage: https://adam-meyer.com/arduino/images/2011/03/tip120-lightbulb1.png

(link to the tutorial: https://adam-meyer.com/arduino/TIP120)

The goal of the TIP120 transistor is to be controlled as a switch. Logical 1 outputted by Arduino pin should power on the Fan and logical 0 should stop it. Here we use the A0 pin with the resistor (look at the orange wire in the pictures).


Then you can fix the setup on the pan with hot glue and you need to find a way to fix the fan. I used plastic clamps (serflex).


In Arduino IDE:

Make sure you selected the right COM port: Tools > Port and select the right one.

Select the right board:

  • Tools > Boards > Arduino Renesas UNO R4 boards > Arduino UNO R4 WIFI
  • If you don't find it, click on Tools > Boards > Boards Manager..., look for the UNO R4 and install the package

Collect Accelerometer Data

arduino-coffee-datalogger-generator.PNG

To collect data with the accelerometer, we will use NanoEdge AI Studio and its tool, the data logger generator:

  1. Open NanoEdge
  2. Go the Data Logger
  3. Select Arduino boards
  4. Select the LIS3DH

Because the vibration made by the boiling water are pretty small and that we don't need a lot of data, here is the parameter to select:

  • Data rate (Hz): 1600Hz
  • Range: 2g
  • Sample per axis: 512

I didn't tested other parameters, feel free to do it!


Click Generate Data Logger and you will obtain a .zip containing a .ino file with the code ready to be flashed on the board.

The code will print buffer via serial. In our case it will send buffer of size 3*512 at a frequency of 1.6kHz.

In the next step, we will read these printed buffer and log datasets to then train the model.


Warning:

You may need to install the library for the accelerometer:

  • Go to sketch > Include Library > Manage libraries... > adafruit LIS3DH


In the data logging code, some part related to NanoEdge are commented. We will use later, after using NanoEdge AI Studio to get an AI library.


The code from NanoEdge is attached below

NanoEdge AI Studio

arduino-boiling-water-signals.PNG
arduino-boiling-water-benchmark.PNG
arduino-boiling-water-validation.PNG

Now that we are able to collect data using the accelerometer, the next step is to use NanoEdge AI Studio to log signals and use these signals to create an AI able to recognize boiling water signals from other ones.


Open NanoEdge AI Studio

Create a N class classification project

In project settings:

  • Select the Arduino R4 WiF as target
  • Select Accelerometer 3 axes as sensor
  • Everything else is optional
  • Click Next


In Signals:

Here we will add data of both not boiling water and boiling water and do it multiple times with different water levels.

We first start simple, I choose to stick with a half full pan and do the following:

  • Put the pan on the electric hob or induction hob (at maximum power)
  • log non boiling data while it is heating up
  • Stop logging non boiling when you think it is close/starting to boil
  • Wait a little to make sure it is really boiling
  • Start logging boiling water

Every 10 signals logged, I was pausing the log and moving a little bit the pan on the fire to add some variety to the data.

I logged around 150 signals of non boiling water (the time it took to start boiling) and around 50 to 100 boiling water.


I then changed the water level and repeated the process multiple time.


To log data in NanoEdge, here how to proceed:

  1. Click on Add Signal
  2. From Serial (USB)
  3. Select the right COM port
  4. Click Start/Stop in NanoEdge to collect signals
  5. Stop when satisfied


I collected multiple boiling classes and multiple not boiling classes. But to launch a benchmark, you need to concatenate all files to ends up with only 2: boiling and not boiling.


To do so, you can:

  • Download all files that you logged
  • Go back to the NanoEdge AI Studio home page
  • Go to Data Manipulation
  • Import all the not boiling files
  • Click concatenate above
  • Click extract line and click run
  • Then save the file containing all the signals
  • Do the same for the boiling
  • Go back to the project and import our 2 datasets containing everything


In Benchmark:

The benchmark is the heart of NanoEdge AI Studio. Here we will use both kinds of signals to try to find the best combination of preprocessing of the data and the best model able to distinguish the data.

  1. Click on New Benchmark
  2. Select all datasets
  3. Click start

The benchmark will iteratively try combination of preprocessing and model on the data and evaluate the performances. It split the data into a training and testing set multiple times to have robust results.

The benchmark can take hours to finish depending on the amount of data used. If the score reach 90% or more, you can pause/stop it and continue. The benchmark will try to optimize the model as much as possible but you can stop it when satisfied.


In Validation:

Validation is made to test models on new data. What can happen when training machine learning or AI model is that the models overfit. It means that a model will have good results during the training but will not be able to work well on new data because it learned the training dataset by heart instead of learning how to classify the data...

To use validation, you need new datasets of both boiling and not boiling signals. You can go back to the previous step to collect more data and download the csv to use them in validation.

To use validation:

  • Select up to 10 models that you want to test (the first ones are the best ones)
  • Click New Experiment
  • Select a new dataset of not boiling water and boiling water
  • Click start

You will get the performance of all the selected model on your new data.


I passed quite some time in this part to train a model that works well. Meaning that I deleted some data and logged new ones, did other benchmark and validation until I found a model that seemed to work as expected. Start simple, and add more and more variety while checking that the model works.


In Compilation:

Simply click compile to get the trained model.

Add NanoEdge AI Model to Arduino

To add the AI to our Arduino code, it is pretty easy, here is what we need to do:

  1. Include the model and the knowledge
  2. initialize the model
  3. Do detections

That it, and everything is done using functions!

Additionally I added some printing on the Led matrix and started a fan when the boiling class is detected to prevent the water from leaking out of the pan.


Now that we have the AI library, we need to add it to our Arduino code:

  • Open the .zip obtained, there is an Arduino folder containing another zip
  • Import the library in Arduino IDE: Sketch > Include library > Add .ZIP library... and select the .zip in the Arduino folder

IF YOU ALREADY USE A NANOEDGE AI LIBRARY IN ARDUINO IDE:

go to document/arduino/library and delete the nanoedge one. Then follow the instruction above to import the new library.


IMPORTANT:

If you get an error because of RAM, it may be because of the library in NanoEdge. Go back to the VALIDATION STEP in NanoEdge and select a smaller library (click on the crown on the right), then compile it and replace it in Arduino IDE.


The main code is attached, find below explanation of each import part:


NanoEdge:

#include <NanoEdgeAI.h>
#include "knowledge.h"

//DON T FORGET TO CHANGE NEAI_MODE TO 1 to do detections
#define NEAI_MODE 1

/* Global variables definitions */
static uint16_t neai_ptr = 0; //pointers to fill for sound buffer
static float neai_buffer[SENSOR_SAMPLES * AXIS] = {0.0}; //souhnd buffer


uint8_t neai_code = 0; //initialization code
uint16_t id_class = 0; // Point to id class (see argument of neai_classification fct)
float output_class_buffer[CLASS_NUMBER]; // Buffer of class probabilities
const char *id2class[CLASS_NUMBER + 1] = { // Buffer for mapping class id to class name
  "unknown",
  "boiling_test3",
  "not_boiling_test3",
};

void setup(){
//some code

  /* Initialize NanoEdgeAI AI */
  neai_code = neai_classification_init(knowledge);
  if (neai_code != NEAI_OK) {
    Serial.print("Not supported board.\n");
  }
}

void loop(){
//some code

 if (NEAI_MODE) {
    neai_classification(neai_buffer, output_class_buffer, &id_class);
    switch (id_class) {
      case 1:
Serial.println("!!! Boiling !!!");
        break;
      case 2:
        Serial.println(" Not Boiling ");
      default:
        Serial.println(" Error ");
        break;
    }
}

The id2class variable depends on the name of the classes that you used in NanoEdge. The order of boiling or not boiling classes can change, you need to copy the one in NanoEdgeAI.h for the compilation .zip:

  • YOUR_LIBRARY.zip\arduino\nanoedgeai_for_arduino.zip\nanoedgeai\src\NanoEdgeIA.h


To use the matrix to display messages:

#include "ArduinoGraphics.h"
#include "Arduino_LED_Matrix.h"

/* Declare matrix to display */
byte frame[8][12] = {
  { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
  { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
  { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
  { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
  { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
  { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
  { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 },
  { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }
};

//text to display
char text[30];


ArduinoLEDMatrix matrix;

void setup(){
//some code
matrix.begin();
}

void loop(){
//some code
// if we want to write something we set our text
strcpy (text, " OUR TEXT ");
// then display it and set some parameters
    matrix.beginDraw();
    matrix.stroke(0xFFFFFFFF);
    matrix.textScrollSpeed(50);
    matrix.textFont(Font_5x7);
    matrix.beginText(0, 1, 0xFFFFFF);
    matrix.println(text);
    matrix.endText(SCROLL_LEFT);
    matrix.endDraw();
}


We also use a fan to blow on the pan when the water is boiling, to do so, here is what you need to do:

...
void setup{
//some code
pinMode(A0, OUTPUT); //A0 is our pin, you can change it if you use another pin
}

void loop{

// if you want to turn the fan on
digitalWrite(A0, HIGH);

// if you want to stop it
digitalWrite(A0, LOW);
}

In our case, we turn the fan ON/OFF in the switch case after neai_classification()


If you use everything explained here you will end up exactly with the main code attached!


Thank you for reading :)