Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

More cleanup #31

Open
wants to merge 3 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
78 changes: 78 additions & 0 deletions lib/app_states/src/app_states.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@

#pragma once

#include <Arduino.h>

enum QueryState {
WAITING_TO_QUERY,
DNS_NOT_FOUND,
SSL_CONNECTION_FAILURE,
LAST_RESPONSE_PASS,
LAST_RESPONSE_FAIL,
LAST_RESPONSE_UNSURE,
NOT_AUTHENTICATED,
};

enum NotificationState {
NOTIFICATION_NOT_ATTEMPTED,
NOTIFICATIONS_SENT,
NOTIFICATION_ATTEMPT_FAILED,
};

String queryStateToString (QueryState state) {
switch (state) {
case WAITING_TO_QUERY:
return "WAITING_TO_QUERY";
case SSL_CONNECTION_FAILURE:
return "CONNECTION_FAILURE";
case DNS_NOT_FOUND:
return "DNS_NOT_FOUND";
case LAST_RESPONSE_PASS:
return "LAST_RESPONSE_PASS";
case LAST_RESPONSE_FAIL:
return "LAST_RESPONSE_FAIL";
case LAST_RESPONSE_UNSURE:
return "LAST_RESPONSE_UNSURE";
case NOT_AUTHENTICATED:
return "NOT_AUTHENTICATED";
default:
return "UNKNOWN";
}
}

String notificationStateToString (NotificationState state) {
switch (state) {
case NOTIFICATION_NOT_ATTEMPTED:
return "NOTIFICATION_NOT_ATTEMPTED";
case NOTIFICATIONS_SENT:
return "NOTIFICATIONS_SENT";
case NOTIFICATION_ATTEMPT_FAILED:
return "NOTIFICATION_ATTEMPT_FAILED";
default:
return "UNKNOWN";
}
}

/**
* @brief A class to lock the ESP32 frame buffer
*
* This class is used to lock the ESP32 frame buffer. It is used mainly for RAII, so that the frame buffer is properly released when the object goes out of scope.
* It's not a true lock, the pointer to the frame is freely accessbile
*
* @todo Implement this class
*/
class EspFrameLock {
public:
camera_fb_t* frame = NULL;

EspFrameLock() {
frame = esp_camera_fb_get();
}

~EspFrameLock() {
esp_camera_fb_return(frame);
delete frame;
}
EspFrameLock& operator=(const EspFrameLock&) = delete; // disallow assignment
EspFrameLock(const EspFrameLock&) = delete; // disallow copying
};
102 changes: 13 additions & 89 deletions src/deployable_example.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -24,14 +24,15 @@ SOFTWARE.

*/

#include <Arduino.h>

#include <Preferences.h>
#include "WiFi.h"
#include <WiFi.h>
#include <esp_camera.h>
#include <time.h>
#include "ArduinoJson.h"
#include "groundlight.h"
#include <ArduinoJson.h>

#include "groundlight.h"
#include "app_states.h"
#include "camera_pins.h" // thank you seeedstudio for this file
#include "integrations.h"
#include "stacklight.h"
Expand All @@ -56,80 +57,10 @@ uint8_t *frame_565_old;
#define ALPHA FRAME_ARR_LEN / ALPHA_DIVISOR
#define BETA 20 // out of 31

enum QueryState {
WAITING_TO_QUERY,
DNS_NOT_FOUND,
SSL_CONNECTION_FAILURE,
LAST_RESPONSE_PASS,
LAST_RESPONSE_FAIL,
LAST_RESPONSE_UNSURE,
NOT_AUTHENTICATED,
};

String queryStateToString (QueryState state) {
switch (state) {
case WAITING_TO_QUERY:
return "WAITING_TO_QUERY";
case SSL_CONNECTION_FAILURE:
return "CONNECTION_FAILURE";
case DNS_NOT_FOUND:
return "DNS_NOT_FOUND";
case LAST_RESPONSE_PASS:
return "LAST_RESPONSE_PASS";
case LAST_RESPONSE_FAIL:
return "LAST_RESPONSE_FAIL";
case LAST_RESPONSE_UNSURE:
return "LAST_RESPONSE_UNSURE";
case NOT_AUTHENTICATED:
return "NOT_AUTHENTICATED";
default:
return "UNKNOWN";
}
}

enum NotificationState {
NOTIFICATION_NOT_ATTEMPTED,
NOTIFICATIONS_SENT,
NOTIFICATION_ATTEMPT_FAILED,
};

String notificationStateToString (NotificationState state) {
switch (state) {
case NOTIFICATION_NOT_ATTEMPTED:
return "NOTIFICATION_NOT_ATTEMPTED";
case NOTIFICATIONS_SENT:
return "NOTIFICATIONS_SENT";
case NOTIFICATION_ATTEMPT_FAILED:
return "NOTIFICATION_ATTEMPT_FAILED";
default:
return "UNKNOWN";
}
}

enum StacklightState {
STACKLIGHT_NOT_FOUND,
STACKLIGHT_ONLINE,
STACKLIGHT_PAIRED,
};

String stacklightStateToString (StacklightState state) {
switch (state) {
case STACKLIGHT_NOT_FOUND:
return "STACKLIGHT_NOT_FOUND";
case STACKLIGHT_ONLINE:
return "STACKLIGHT_ONLINE";
case STACKLIGHT_PAIRED:
return "STACKLIGHT_PAIRED";
default:
return "UNKNOWN";
}
}

QueryState queryState = WAITING_TO_QUERY;
NotificationState notificationState = NOTIFICATION_NOT_ATTEMPTED;
StacklightState stacklightState = STACKLIGHT_NOT_FOUND;

camera_fb_t *frame = NULL;
int *last_frame_buffer = NULL;
char groundlight_endpoint[60] = "api.groundlight.ai";

Expand Down Expand Up @@ -248,7 +179,7 @@ bool decodeWorkingHoursString(String working_hours);

bool should_deep_sleep() {
return (query_delay > 29) && !disable_deep_sleep_for_notifications && !disable_deep_sleep_until_reset;
}
}

void deep_sleep() {
int time_elapsed = millis() - last_upload_time;
Expand Down Expand Up @@ -532,7 +463,7 @@ void setup() {
preferences.getString("api_key", groundlight_API_key, 75);
preferences.getString("det_id", groundlight_det_id, 100);
query_delay = preferences.getInt("query_delay", query_delay);

WiFi.begin(ssid, password);
wifi_configured = true;
}
Expand Down Expand Up @@ -737,32 +668,28 @@ void loop () {
vTaskDelay(1000 / portTICK_PERIOD_MS);
#endif

frame = esp_camera_fb_get();
// Testing how to get the latest image
esp_camera_fb_return(frame);
frame = esp_camera_fb_get();
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As far as I can tell, there isn't a clear reason to get and release before getting the frame we'll actually use. If there is, we can incorporate that into the frameLock class

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this may be to clear a buffered frame. test with an actual camera and an image that is changing to confirm ?

EspFrameLock frameLock;

#if defined(GPIO_LED_FLASH)
digitalWrite(GPIO_LED_FLASH, LOW);
#endif

if (!frame)
if (!frameLock.frame)
{
debug_printf("Camera capture failed! Restarting system in 3 seconds!\n");
delay(3000);
ESP.restart(); // some boards are less reliable for camera captures and will everntually just start working
}

debug_printf("encoded size is %d bytes\n", frame->len);
debug_printf("encoded size is %d bytes\n", frameLock.frame->len);

preferences.begin("config");
if (preferences.isKey("motion") && preferences.getBool("motion") && preferences.isKey("mot_a") && preferences.isKey("mot_b")) {
int alpha = round(preferences.getString("mot_a", "0.0").toFloat() * (float) FRAME_ARR_LEN);
int beta = round(preferences.getString("mot_b", "0.0").toFloat() * (float) COLOR_VAL_MAX);
if (is_motion_detected(frame, alpha, beta)) {
if (is_motion_detected(frameLock.frame, alpha, beta)) {
debug_println("Motion detected!");
} else {
esp_camera_fb_return(frame);
if (should_deep_sleep()) {
vTaskDelay(500 / portTICK_PERIOD_MS);
deep_sleep();
Expand All @@ -789,14 +716,13 @@ void loop () {

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Immediately above is a return statement without a esp_camera_fb_return call. Using the frameLock class should change the behavior there as we now will implicitly call esp_camera_fb_return in the class destructor

debug_printf("Submitting image query to Groundlight...");

queryResults = submit_image_query(frame, groundlight_endpoint, groundlight_det_id, groundlight_API_key);
queryResults = submit_image_query(frameLock.frame, groundlight_endpoint, groundlight_det_id, groundlight_API_key);
queryID = get_query_id(queryResults);

debug_printf("Query ID: %s\n", queryID.c_str());

if (queryID == "NONE" || queryID == "") {
debug_println("Failed to get query ID");
esp_camera_fb_return(frame);
return;
}

Expand Down Expand Up @@ -850,7 +776,7 @@ void loop () {
}
}
if (shouldDoNotification(queryResults)) {
if (sendNotifications(last_label, frame)) {
if (sendNotifications(last_label, frameLock.frame)) {
notificationState = NOTIFICATIONS_SENT;
} else {
notificationState = NOTIFICATION_ATTEMPT_FAILED;
Expand All @@ -875,8 +801,6 @@ void loop () {
debug_println(error.c_str());
}

esp_camera_fb_return(frame);

if (should_deep_sleep()) {
vTaskDelay(500 / portTICK_PERIOD_MS);
deep_sleep();
Expand Down
35 changes: 28 additions & 7 deletions src/stacklight.h
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
#pragma once

#include <Arduino.h>
#include "WiFi.h"
#include <HTTPClient.h>
Expand Down Expand Up @@ -117,12 +119,12 @@ namespace Stacklight

/**
* @brief Try to connect to Stacklight AP and push WiFi credentials
*
*
* This happens over time, and not for initialization. Must be connected to the stacklight first.
*
*
* @param ssid WiFi SSID
* @param password WiFi password
*
*
* @return Stacklight IP address (empty string if not connected)
*/
String tryConnectToStacklight(const char * ssid, const char * password) {
Expand All @@ -133,18 +135,18 @@ namespace Stacklight
vTaskDelay(1000 / portTICK_PERIOD_MS);
pushWiFiCredToStacklight(ssid, password);
}

return ip;
}

/**
* @brief Try to connect to Stacklight AP and push WiFi credentials
*
*
* This is for initialization. Must be connected to the stacklight first.
*
*
* @param ssid WiFi SSID
* @param password WiFi password
*
*
* @return Stacklight IP address (empty string if not connected)
*/
String initStacklight(const char * ssid, const char * password, int retries = 10) {
Expand Down Expand Up @@ -184,3 +186,22 @@ namespace Stacklight
return ip;
}
}

enum StacklightState {
STACKLIGHT_NOT_FOUND,
STACKLIGHT_ONLINE,
STACKLIGHT_PAIRED,
};

String stacklightStateToString (StacklightState state) {
switch (state) {
case STACKLIGHT_NOT_FOUND:
return "STACKLIGHT_NOT_FOUND";
case STACKLIGHT_ONLINE:
return "STACKLIGHT_ONLINE";
case STACKLIGHT_PAIRED:
return "STACKLIGHT_PAIRED";
default:
return "UNKNOWN";
}
}