ESP32-Cam Headless Controller

ESP32-Cam Headless Controller

Brian Lough created a video on YouTube titled “WiFiManager – An Essential ESP32 library!”

This video pointed out that hard coded settings on the ESP32 can be a limitation. It went on to demonstrate this with wireless network connectivity and the inherent dependency when preset values are used. The solution rested on a library that gives the ESP32 the ability to operate as a WiFi access point and for the ESP32 to provide a web portal for an attached device. The web page could then take form entries to set variables on the ESP32.

In this post, I’ll be using this concept with the intent to set the time and date on the ESP32-Cam module. There are serveral demonstrations of setting time on the ESP32 online that utilize the network time protocol (here are two examples, see next video clip and https://randomnerdtutorials.com/esp32-date-time-ntp-client-server-arduino/). However, this is dependent on an internet connection to a NTP host. I’ll be using the ESP32Time library, https://www.arduino.cc/reference/en/libraries/esp32time/, which has no need for an external RTC module or NTP time synchronization. This will allow the ESP32 module to be operated offline with a minimal set of hardware.

The ability to control the ESP32 module using a webpage on a mobile device also reduces the need for control and indication hardware on the ESP32, which was a requirement on my earlier flight data logger, https://www.cloudacm.com/?p=3713

Much of the working code is based off of the work of Rui Santos, https://randomnerdtutorials.com/esp32-email-alert-temperature-threshold/. There have been some adaptations, such as the webpage content and its appearence. The data logging function also expands on the base code, with the timestamp information available in both the file properties and also its content.

Here is an example of the code used in this project.

// Title and Description

/** 
ESP32-Cam_AP_WebServer_Head-Controller_Example

Code Base for ESP32-Cam module to allow it to operate in AP mode
and provide a web portal used as a control interface
via a mobile device, such as a phone or tablet.
Thu May 26 2022 13:01:55 GMT+0000 (1653570115)
**/


// Notes
/** 
This code contains serial debug which will need to be removed before final deploy

**/



// Credit and Sources

/*********
  Rui Santos
  Complete project details at https://RandomNerdTutorials.com/esp32-email-alert-temperature-threshold/
  
  Permission is hereby granted, free of charge, to any person obtaining a copy
  of this software and associated documentation files.
  
  The above copyright notice and this permission notice shall be included in all
  copies or substantial portions of the Software.
*********/

/** 
  Unix Epoch Time Calculator  
  https://unixtime.org/
  NIST Offical U.S. Time
  https://time.gov/
**/

/** 
  WiFiManager - An Essential ESP32 library!
  
**/

// Libraries

#include <WiFi.h>
#include <AsyncTCP.h>
#include <ESPAsyncWebServer.h>
#include <ESP32Time.h>
#include <esp_wifi.h>
#include <SD_MMC.h>
#include "esp_camera.h"



// Definitions

ESP32Time rtc;
AsyncWebServer server(80);

// Pins for ESP32-CAM
#define FLASH_PIN         4

// Pins for Image Sensor
#define PWDN_GPIO_NUM     32
#define RESET_GPIO_NUM    -1
#define XCLK_GPIO_NUM      0
#define SIOD_GPIO_NUM     26
#define SIOC_GPIO_NUM     27
#define Y9_GPIO_NUM       35
#define Y8_GPIO_NUM       34
#define Y7_GPIO_NUM       39
#define Y6_GPIO_NUM       36
#define Y5_GPIO_NUM       21
#define Y4_GPIO_NUM       19
#define Y3_GPIO_NUM       18
#define Y2_GPIO_NUM        5
#define VSYNC_GPIO_NUM    25
#define HREF_GPIO_NUM     23
#define PCLK_GPIO_NUM     22



// Variables and Constants


const char* TIME_INPUT = "inputTime";
const char* TIME_COMMIT = "inputTimeCommit";
const char* SCAN_TIME = "ScanTime";
const char* SCAN_COMMIT = "ScanCommit";

// Interval between sensor readings. Learn more about timers: https://RandomNerdTutorials.com/esp32-pir-motion-sensor-interrupts-timers/
unsigned long previousMillis = 0;     
const long interval = 1000;  

unsigned long ScanSeconds = 1800;
unsigned long ScanMillis = 0;    
unsigned long previousScanMillis = 0;     
const long Scaninterval = 1000;  

const char* FileName = "/datalogger.csv";

// Set your new MAC Address
// Example mac address A1:B2:C3:D4:E5:F6
uint8_t newMACAddress[] = {0xA1, 0xB2, 0xC3, 0xD4, 0xE5, 0xF6};

// REPLACE WITH YOUR NETWORK CREDENTIALS
const char* ssid     = "ESP32-Cam-AP1";

// Default Value
// Thu May 26 2022 13:01:55 GMT+0000 https://unixtime.org/
int epoch = 1653570115;
String inputTime = "1653570115";
String CurrentTime;
String EpochValue;
String inputTimeCommit;
String ScanCommit;
String ScanTime;
String ScanEnabled;
bool DisableAP = 0;
bool EnableAP = 0;

// int epochInt = inputTime.toInt(); - Example of converting string to int
// String epochString = String(epoch); - Example of converting long to string

// HTML Code - This should remain in the Constants and Variables section of the code
// HTML web page to handle 3 input fields (email_input, enable_email_input, inputTime)
const char index_html[] PROGMEM = R"rawliteral(


<!DOCTYPE HTML><html>
<head>
  <title>Headend Controller</title>
  <meta name="viewport" content="width=device-width, initial-scale=1">
  <link rel="icon" href="data:,">


  <style>
  html {
    font-family: Arial, Helvetica, sans-serif; 
    text-align: center;
  }
  h1 {
    font-size: 1.8rem;
    color: rgba(238,238,238,1);
  }
  h2{
    font-size: 1.5rem;
    font-weight: bold;
    color: rgba(238,238,238,1);
  }
  .topnav {
    overflow: hidden;
    background-color: green;
  }
  body {
    margin: 0;
    background-color: rgba(71,71,71,1);
  }
  .content {
    padding: 30px;
    max-width: 600px;
    margin: 0 auto;
  }
  .card {
    background-color: rgba(51,51,51,1);
    border: 2px solid rgba(85,85,85,1);
    padding-top:10px;
    padding-bottom:20px;
  }
  .button {
    padding: 15px 50px;
    font-size: 24px;
    text-align: center;
    outline: none;
    color: #fff;
    background-color: grey;
    border: none;
    border-radius: 5px;
    -webkit-touch-callout: none;
    -webkit-user-select: none;
    -khtml-user-select: none;
    -moz-user-select: none;
    -ms-user-select: none;
    user-select: none;
    -webkit-tap-highlight-color: rgba(0,0,0,0);
   }
   /*.button:hover {background-color: green}*/
   .button:active {
     background-color: green;
     box-shadow: 2 2px #CDCDCD;
     transform: translateY(2px);
   }
   .state {
     font-size: 1.5rem;
     color:rgba(238,238,238,1);
     font-weight: bold;
   }
   .timeinfo {
     font-size: 1rem;
     color:rgba(238,238,238,1);
     font-weight: bold;
   }
  </style>
<title>Headend Controller</title>
<meta name="viewport" content="width=device-width, initial-scale=1">
<link rel="icon" href="data:,">
</head>
<body>

  <div class="topnav">
    <h1><img src=""><br>Headend Controller</h1>
  </div>

  <div class="content">
    <div class="card">
    
      <p class="timeinfo"><form action="/get"></p>
    
      <p class="state">Current Time (GMT): <span id="state">%EPOCH%</span></p>
    
      <p class="timeinfo">Enter New Unix Timestamp <input type="number" step="0.1" name="inputTime" value="%TIME%" required></p>  
      <p class="timeinfo">Entered Epoch: <span id="state">&zwj;%TIME%</span></p>    
      <p class="timeinfo">Use New Timestamp <input type="checkbox" name="inputTimeCommit" value="true" %TIME_COMMIT%></p>
    
      <p class="timeinfo">Scan Duration (Minutes) <input type="number" step="0.1" name="ScanTime" value="%SCAN_TIME%" max="120"></p>
      <p class="timeinfo">Entered Duration: <span id="state">&zwj;%SCAN_TIME%</span></p>    
        
      <p class="timeinfo">Start Wifi Scan <input type="checkbox" name="ScanCommit" value="true" %SCAN_COMMIT%></p>
      <p class="timeinfo"><input type="submit" value="Submit"></p>
    
      <p class="timeinfo"></form></p>
      <p class="timeinfo"><iframe style="display:none" name="hidden-form"></iframe></p>
    
      </span></p>
    </div>
  </div>
  
</body></html>

)rawliteral";


// Processor variable section to handle values from web form submitted by mobile device
String processor(const String& var){
  if(var == "EPOCH"){
    return EpochValue;
  }
  else if(var == "TIME"){
    return inputTime;
  }
  else if(var == "TIME_COMMIT"){
    return inputTimeCommit;
  }
  else if(var == "SCAN_COMMIT"){
    return ScanCommit;
  }
  else if(var == "SCAN_TIME"){
    return ScanTime;
  }
  return String();
}



// Routines and Subroutines



// Start MicroSD Subroutine
bool startMicroSD() {
  Serial.print("Starting microSD... ");

  // Pin 13 needs to be pulled-up
  // https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/peripherals/sd_pullup_requirements.html#pull-up-conflicts-on-gpio13
  pinMode(13, OUTPUT);
  digitalWrite(13, HIGH);

  if(SD_MMC.begin("/sdcard", true)) {
    Serial.println("OKAY");
    return true;
  } else {
    Serial.println("FAILED");
    return false;
  }
}



// Start Camera Subroutine
bool startCamera() {
  // Turn off the flash - The flash still occurs when files are read or written to the microSD
  pinMode(FLASH_PIN, OUTPUT);
  digitalWrite(FLASH_PIN, LOW);

  // Initialize the camera hardware
  camera_config_t config;
  config.ledc_channel = LEDC_CHANNEL_0;
  config.ledc_timer = LEDC_TIMER_0;
  config.pin_d0 = Y2_GPIO_NUM;
  config.pin_d1 = Y3_GPIO_NUM;
  config.pin_d2 = Y4_GPIO_NUM;
  config.pin_d3 = Y5_GPIO_NUM;
  config.pin_d4 = Y6_GPIO_NUM;
  config.pin_d5 = Y7_GPIO_NUM;
  config.pin_d6 = Y8_GPIO_NUM;
  config.pin_d7 = Y9_GPIO_NUM;
  config.pin_xclk = XCLK_GPIO_NUM;
  config.pin_pclk = PCLK_GPIO_NUM;
  config.pin_vsync = VSYNC_GPIO_NUM;
  config.pin_href = HREF_GPIO_NUM;
  config.pin_sscb_sda = SIOD_GPIO_NUM;
  config.pin_sscb_scl = SIOC_GPIO_NUM;
  config.pin_pwdn = PWDN_GPIO_NUM;
  config.pin_reset = RESET_GPIO_NUM;
  config.xclk_freq_hz = 20000000;
  config.pixel_format = PIXFORMAT_JPEG;
  
  // Set resolution based on whether we have extra memory
  if(psramFound()){
    Serial.println("PSRAM found. Maximum XGA resolution supported.");
    config.frame_size = FRAMESIZE_XGA;
    config.jpeg_quality = 10;
    config.fb_count = 2;
  } else {
    Serial.println("PSRAM not found. Maximum SVGA resolution supported.");
    config.frame_size = FRAMESIZE_SVGA;
    config.jpeg_quality = 12;
    config.fb_count = 1;
  }

  // Start the camera
  Serial.print("Starting camera... ");
  esp_err_t err = esp_camera_init(&config);
  if (err != ESP_OK) {
    Serial.println("FAILED");
    return false;
  } else {
    Serial.println("OKAY");
    return true;
  }
}


//Append to the end of file in SD card
void appendFile(fs::FS &fs, const char * path, const char * message){

    File file = fs.open(path, FILE_APPEND);
    if(!file){
        return;
    }
    if(file.print(message)){
    } else {
    }
}

// Printout Header Routine
void PrintHeader()
{
    appendFile(SD_MMC, FileName, "WifiScan-Results\n");
}



// WebServer Fault Subroutine
void notFound(AsyncWebServerRequest *request) {
  request->send(404, "text/plain", "Not found");
} 


// Camera Operate and Image Save Subroutine
void takePhoto(String filename) { 
  // Take a photo and get the data

  camera_fb_t *fb = esp_camera_fb_get();
  if (!fb) {
    Serial.println("Unable to take a photo");
    return;
  }

  // Make sure it is a JPEG
  if (fb->format != PIXFORMAT_JPEG) {
     Serial.println("Capture format not JPEG");
     esp_camera_fb_return(fb); // Return the photo data
     return;
  }

  // Save the picture to the SD card

  File file = SD_MMC.open(filename.c_str(), "w");
  if(file) {
    Serial.println("Saving " + filename);
    file.write(fb->buf, fb->len);
    file.close();
  } else {
    Serial.println("Unable to write " + filename);
  }

  // Return the picture data
  esp_camera_fb_return(fb);
}



// Capture Photo Routine
void CapturePhoto() {
  
    
        // Keep a count of the number of photos we have taken
        static int number = 0;
      
        // Keep a count of the number of photos we have taken
        number++;
      
        // Construct a filename that looks like "/1653482900.jpg"
        String filename = "/";
        filename += String(rtc.getEpoch());
        filename += ".jpg"; 
        takePhoto(filename);

        
        if (ScanSeconds > 0) {
           unsigned long ScanMillis = millis();
           if (ScanMillis - previousScanMillis >= Scaninterval) {
              previousScanMillis = ScanMillis;              
              
              String countdown = "Seconds remaining: ";
              countdown += String(ScanSeconds);
              Serial.println(countdown);
                            
                            
              // Partial Wifi Scan code
              if (DisableAP == 1) {
                 // Disconnect WiFi to allow scanning
                 WiFi.disconnect();
                 delay(100);    
                 DisableAP = 0;
                 EnableAP = 1;
              }
              // WiFi.scanNetworks will return the number of networks found
              
              String TimeStamp = String(rtc.getEpoch());
              appendFile(SD_MMC, FileName, TimeStamp.c_str()); 
              appendFile(SD_MMC, FileName, ",");
              TimeStamp = String(rtc.getDateTime());
              appendFile(SD_MMC, FileName, TimeStamp.c_str()); 
              appendFile(SD_MMC, FileName, ",");
              int n = WiFi.scanNetworks();  // This number can be staticly set to limit results          
              String StringWifiNetworks = String(n);
              appendFile(SD_MMC, FileName, StringWifiNetworks.c_str()); 
              appendFile(SD_MMC, FileName, ",");
              for (int i = 0; i < n; ++i) {
                  // Print SSID and RSSI for each network found - Maybe append to string so a single write occurs
                  String StringSSID = String(WiFi.SSID(i));
                  appendFile(SD_MMC, FileName, StringSSID.c_str()); 
                  appendFile(SD_MMC, FileName, ",");
                  String StringRSSI = String(WiFi.RSSI(i));
                  appendFile(SD_MMC, FileName, StringRSSI.c_str()); 
                  appendFile(SD_MMC, FileName, ",");
                  String StringEncryptionType = String((WiFi.encryptionType(i) == WIFI_AUTH_OPEN)?" ":"*");
                  appendFile(SD_MMC, FileName, StringEncryptionType.c_str()); 
                  appendFile(SD_MMC, FileName, ",");   
              }
              appendFile(SD_MMC, FileName, "\n");  
              /**
              String networksfound = "WiFi Networks Found: ";
              networksfound += String(n);
              Serial.println(networksfound);
              **/
              
              ScanSeconds--;
           }
          
        }

        if (ScanSeconds == 0) {
          if (EnableAP == 1)  {
            WiFi.softAP(ssid);
            delay(100); 
            EnableAP = 0;  
          }
          ScanEnabled = "Standby";
        }

        
}

// Setup Routine
void setup() {

  ScanEnabled = "Standby";
  
  rtc.setTime(epoch);
    
  // Initialize the peripherals
  Serial.begin(115200);
  while(!Serial) delay(100);
  
  startMicroSD();
  startCamera(); 
  PrintHeader();

  delay(5000); // Delay 5 seconds before first photo
  
  WiFi.mode(WIFI_AP);

  esp_wifi_set_mac(WIFI_IF_AP, &newMACAddress[0]);
  
  WiFi.softAP(ssid);
  delay(100);
  
  IPAddress IP = WiFi.softAPIP();

  IPAddress Ip(192, 168, 100, 1);
  IPAddress NMask(255, 255, 255, 0);
  WiFi.softAPConfig(Ip, Ip, NMask);

  IPAddress myIP = WiFi.softAPIP();

  // Send web page to client
  server.on("/", HTTP_GET, [](AsyncWebServerRequest *request){
    request->send_P(200, "text/html", index_html, processor);
  });

  // Receive an HTTP GET request at <ESP_IP>/get?email_input=<inputMessage>&enable_email_input=<inputMessage2>&inputTime=<inputTime>
  server.on("/get", HTTP_GET, [] (AsyncWebServerRequest *request) {
      // GET inputTime value on <ESP_IP>/get?inputTime=<inputTime>

  if (request->hasParam(TIME_INPUT)) {
        inputTime = request->getParam(TIME_INPUT)->value();
    if (request->hasParam(TIME_COMMIT)) {
      // Pass the input and set the RTC time if the commit check box is checked.
        rtc.setTime(inputTime.toInt()); 
      }
    }

  if (request->hasParam(SCAN_TIME)) {
         ScanTime = request->getParam(SCAN_TIME)->value();
         ScanSeconds = ScanTime.toInt();
         ScanSeconds = 10*ScanSeconds; // Change ScanTime minutes to actual seconds (was 60, changed because wifi scan takes 6 seconds per 1 second)
    if (request->hasParam(SCAN_COMMIT)) {
      // Call the wifiscan function, this will break the Headend Controller AP mode
      // The wifiscan function can have a nested image capture function and write to MicroSD function.
      // Start WifiScan if this variable changes from the default via the web interface controller.
      ScanEnabled = "Enabled";   
      int DisableAP = 1;         
      }
    }
    
    // request->send(200, "text/html", "HTTP GET request sent to your ESP.<br><a href=\"/\">Return to Home Page</a>");
    request->send_P(200, "text/html", index_html, processor);
  });
  server.onNotFound(notFound);
  server.begin();
}



// Super Loop Routine
void loop() {
  unsigned long currentMillis = millis();
  if (currentMillis - previousMillis >= interval) {
    previousMillis = currentMillis;
    EpochValue = String(rtc.getTime("%A, %B %d %Y %H:%M:%S"));
    // CurrentTime = String(rtc.getTime());

      if (ScanEnabled.equals("Enabled")) {
        CapturePhoto();
      }
  
      else {
        Serial.println(ScanEnabled);
      }
  }
}

Here is a video of what the mobile device setup looks like and its ease of use as a controller of the ESP32 module.

It demonstrates how to set the time and initiate the scan. It also shows the use of SensorLog, this is a separate app ran on the iphone to log other data. The timestamps and data from both devices can be coorelated to allow all of the data to be combined, which is the primary goal of this topic.

Here is a list of items that were changed in the code that are customizable for any needed application.

1. Set your new MAC Address
// Located in Variables and Constants Function
// Example mac address A1:B2:C3:D4:E5:F6
uint8_t newMACAddress[] = {0xA1, 0xB2, 0xC3, 0xD4, 0xE5, 0xF6};
// Located in Setup Function
esp_wifi_set_mac(WIFI_IF_AP, &newMACAddress[0]);

2. Set your IP address
// Located in Setup Function
IPAddress Ip(192, 168, 100, 1);
IPAddress NMask(255, 255, 255, 0);
WiFi.softAPConfig(Ip, Ip, NMask);

3. Set ESP32-Cam up as a wifi AP
// Located in Variables and Constants Function
// REPLACE WITH YOUR NETWORK CREDENTIALS
const char* ssid     = "ESP32-Cam-AP1";
// Located in Setup Function
WiFi.mode(WIFI_AP);  
WiFi.softAP(ssid);

4. File name of datascan log file
// Located in Variables and Constants Function
// This variable is called in several of the functions
const char* FileName = "/datalogger.csv";

5. File name that changes with time for photos
// Located in the Capture Photo Routine
String filename = "/";
filename += String(rtc.getEpoch());
filename += ".jpg"; 
takePhoto(filename);

6. Some numbers on mobile devices are seen as phone numbers which hyperlink
// Located in the HTML code
// &zwj; prevents the hyperlinking and just displays the numbers.
<p class="timeinfo">Entered Epoch: <span id="state">&zwj;%TIME%</span></p> 

7. Embedded image using Base64
// Located in the HTML code
<img src="data:image/jpg;base64,....
I used a linux system to encode in Base64 using this script.

Image2Base64.sh

#!/bin/bash
# https://stackoverflow.com/questions/32698451/how-do-i-convert-a-base64-image
# instead of uploading to a third party - https://www.base64-image.de/

# <img src="<the results here>">

{ echo "data:image/jpg;base64,"; openssl enc -base64 -in uvex.jpg; } > uvex.txt

# text file contents are pasted in HTML code, inside quotes of this tag <img src="">

Although not done on this code, the icon for mobile devices of the website can use an embedded icon file using a similar approach.  Here are a couple of examples of code that would be placed in the HTML code.

...
<link rel="icon" type="image/jpg" href="data:image/jpg;base64..." />
...

 or
 
...
<link rel="icon" type="image/png" href="data:image/png;base64..." />
...

Here is a video that was compiled from the series of images taken during the scan duration. The script to create it is below the video

# Image2Video.sh

#!/bin/bash

cd 'Path containing images'
ls *jpg > imagelist1.txt
sed "s/1653/file \'1653/" imagelist1.txt > imagelist2.txt
sed "s/jpg/jpg\'/" imagelist2.txt > imagelist3.txt
ffmpeg -y -r 10 -f concat -safe 0 -i "imagelist3.txt" -c:v libx264 -vf "fps=10,format=yuv420p" "ESP32-Cam_Timelapse.mp4"
rm *.txt

 

Comments are closed.