Hola muy buenas, tengo un problema con este codigo el cual se supone que tiene que sacar una foto con un esp32-cam, almacenarla en la SD y luego subirla a firebase storage lo cual no realiza correctamente, solamente la almacena en la tarjeta de memoria. He probado tambien invertir el orden de almacenamiento, primero almacenar en firebase storage y luego en la SD, teniendo el mismo problema, se almacena priMero en firebase y no en la SD, es decir la imagen se guarda en alguno de ambos almacenamientos pero no en ambos, cualquier ayuda seria de gran ayuda, gracias.
CODIGO:
#include "Arduino.h"
#include "WiFi.h"
#include "esp_camera.h"
#include "soc/soc.h" // Disable brownout problems
#include "soc/rtc_cntl_reg.h" // Disable brownout problems
#include "driver/rtc_io.h"
#include <LittleFS.h>
#include <FS.h>
#include <Firebase_ESP_Client.h>
#include <addons/TokenHelper.h>
#include <time.h> // Para obtener el timestamp
const char* ssid = "";
const char* password = "";
#define API_KEY ""
#define USER_EMAIL ""
#define USER_PASSWORD ""
#define STORAGE_BUCKET_ID ""
#define FILE_PHOTO_PATH "/photo.jpg"
String bucketPhoto; // Variable para almacenar el nombre único del archivo en Firebase
#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
boolean takeNewPhoto = false;
const int buttonPin = 13;
FirebaseData fbdo;
FirebaseAuth auth;
FirebaseConfig configF;
void fcsUploadCallback(FCS_UploadStatusInfo info);
bool taskCompleted = false;
int photoCounter = 0; // Contador de fotos para dar un nombre único
void capturePhotoSaveLittleFS() {
camera_fb_t* fb = NULL;
for (int i = 0; i < 4; i++) {
fb = esp_camera_fb_get();
esp_camera_fb_return(fb);
fb = NULL;
}
fb = esp_camera_fb_get();
if(!fb) {
Serial.println("Camera capture failed");
delay(1000);
ESP.restart();
}
Serial.printf("Picture file name: %s\n", FILE_PHOTO_PATH);
File file = LittleFS.open(FILE_PHOTO_PATH, FILE_WRITE);
if (!file) {
Serial.println("Failed to open file in writing mode");
} else {
file.write(fb->buf, fb->len);
Serial.print("The picture has been saved in ");
Serial.print(FILE_PHOTO_PATH);
Serial.print(" - Size: ");
Serial.print(fb->len);
Serial.println(" bytes");
}
file.close();
esp_camera_fb_return(fb);
}
void initWiFi(){
WiFi.begin(ssid, password);
while (WiFi.status() != WL_CONNECTED) {
delay(1000);
Serial.println("Connecting to WiFi...");
}
}
void initLittleFS(){
if (!LittleFS.begin(true)) {
Serial.println("An Error has occurred while mounting LittleFS");
ESP.restart();
}
else {
delay(500);
Serial.println("LittleFS mounted successfully");
}
}
void initCamera(){
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;
config.grab_mode = CAMERA_GRAB_LATEST;
if (psramFound()) {
config.frame_size = FRAMESIZE_UXGA;
config.jpeg_quality = 10;
config.fb_count = 1;
} else {
config.frame_size = FRAMESIZE_SVGA;
config.jpeg_quality = 12;
config.fb_count = 1;
}
esp_err_t err = esp_camera_init(&config);
if (err != ESP_OK) {
Serial.printf("Camera init failed with error 0x%x", err);
ESP.restart();
}
}
void setup() {
Serial.begin(115200);
pinMode(buttonPin, INPUT_PULLUP); // Configura el pin del pulsador como entrada
initWiFi();
initLittleFS();
WRITE_PERI_REG(RTC_CNTL_BROWN_OUT_REG, 0);
initCamera();
configF.api_key = API_KEY;
auth.user.email = USER_EMAIL;
auth.user.password = USER_PASSWORD;
configF.token_status_callback = tokenStatusCallback;
Firebase.begin(&configF, &auth);
Firebase.reconnectWiFi(true);
}
void loop() {
// Esperar hasta que se presione el botón para tomar la foto
if (digitalRead(buttonPin) == HIGH) {
// Se presionó el botón, tomar una nueva foto
takeNewPhoto = true;
delay(500); // Anti-rebote
}
if (takeNewPhoto) {
photoCounter++; // Incrementa el contador de fotos
String timeStamp = String(photoCounter); // Usa el contador como parte del nombre del archivo
bucketPhoto = "/data/photo_" + timeStamp + ".jpg"; // Asigna un nombre único a cada foto
capturePhotoSaveLittleFS();
takeNewPhoto = false; // Solo se tomará una foto al presionar el botón, se desactiva la captura para evitar repetición
taskCompleted = false; // Resetea la bandera de tarea para permitir nuevas subidas
}
if (Firebase.ready() && !taskCompleted && !takeNewPhoto) {
taskCompleted = true;
Serial.print("Uploading picture... ");
if (Firebase.Storage.upload(&fbdo, STORAGE_BUCKET_ID, FILE_PHOTO_PATH, mem_storage_type_flash, bucketPhoto.c_str(), "image/jpeg", fcsUploadCallback)){
Serial.printf("\nDownload URL: %s\n", fbdo.downloadURL().c_str());
} else {
Serial.println(fbdo.errorReason());
}
}
}
void fcsUploadCallback(FCS_UploadStatusInfo info){
if (info.status == firebase_fcs_upload_status_init){
Serial.printf("Uploading file %s (%d) to %s\n", info.localFileName.c_str(), info.fileSize, info.remoteFileName.c_str());
}
else if (info.status == firebase_fcs_upload_status_upload) {
Serial.printf("Uploaded %d%s, Elapsed time %d ms\n", (int)info.progress, "%", info.elapsedTime);
}
else if (info.status == firebase_fcs_upload_status_complete) {
Serial.println("Upload completed\n");
}
else if (info.status == firebase_fcs_upload_status_error) {
Serial.printf("Upload failed, %s\n", info.errorMsg.c_str());
}
}