server_camera.cpp 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. #include "server_camera.h"
  2. #include <string>
  3. #include "string.h"
  4. #include "esp_camera.h"
  5. #include "ClassControllCamera.h"
  6. #include "MainFlowControl.h"
  7. #include "ClassLogFile.h"
  8. #include "esp_log.h"
  9. #include "basic_auth.h"
  10. #include "../../include/defines.h"
  11. static const char *TAG = "server_cam";
  12. void PowerResetCamera()
  13. {
  14. #if CAM_PIN_PWDN == GPIO_NUM_NC // Use reset only if pin is available
  15. LogFile.WriteToFile(ESP_LOG_DEBUG, TAG, "No power down pin availbale to reset camera");
  16. #else
  17. ESP_LOGD(TAG, "Resetting camera by power down line");
  18. gpio_config_t conf;
  19. conf.intr_type = GPIO_INTR_DISABLE;
  20. conf.pin_bit_mask = 1LL << CAM_PIN_PWDN;
  21. conf.mode = GPIO_MODE_OUTPUT;
  22. conf.pull_down_en = GPIO_PULLDOWN_DISABLE;
  23. conf.pull_up_en = GPIO_PULLUP_DISABLE;
  24. gpio_config(&conf);
  25. // carefull, logic is inverted compared to reset pin
  26. gpio_set_level(CAM_PIN_PWDN, 1);
  27. vTaskDelay(1000 / portTICK_PERIOD_MS);
  28. gpio_set_level(CAM_PIN_PWDN, 0);
  29. vTaskDelay(1000 / portTICK_PERIOD_MS);
  30. #endif
  31. }
  32. esp_err_t handler_lightOn(httpd_req_t *req)
  33. {
  34. #ifdef DEBUG_DETAIL_ON
  35. LogFile.WriteHeapInfo("handler_lightOn - Start");
  36. ESP_LOGD(TAG, "handler_lightOn uri: %s", req->uri);
  37. #endif
  38. if (Camera.getCameraInitSuccessful())
  39. {
  40. Camera.LightOnOff(true);
  41. const char *resp_str = (const char *)req->user_ctx;
  42. httpd_resp_send(req, resp_str, strlen(resp_str));
  43. }
  44. else
  45. {
  46. httpd_resp_send_err(req, HTTPD_403_FORBIDDEN, "Camera not initialized: REST API /lighton not available!");
  47. return ESP_ERR_NOT_FOUND;
  48. }
  49. #ifdef DEBUG_DETAIL_ON
  50. LogFile.WriteHeapInfo("handler_lightOn - Done");
  51. #endif
  52. return ESP_OK;
  53. }
  54. esp_err_t handler_lightOff(httpd_req_t *req)
  55. {
  56. #ifdef DEBUG_DETAIL_ON
  57. LogFile.WriteHeapInfo("handler_lightOff - Start");
  58. ESP_LOGD(TAG, "handler_lightOff uri: %s", req->uri);
  59. #endif
  60. if (Camera.getCameraInitSuccessful())
  61. {
  62. Camera.LightOnOff(false);
  63. const char *resp_str = (const char *)req->user_ctx;
  64. httpd_resp_send(req, resp_str, strlen(resp_str));
  65. }
  66. else
  67. {
  68. httpd_resp_send_err(req, HTTPD_403_FORBIDDEN, "Camera not initialized: REST API /lightoff not available!");
  69. return ESP_ERR_NOT_FOUND;
  70. }
  71. #ifdef DEBUG_DETAIL_ON
  72. LogFile.WriteHeapInfo("handler_lightOff - Done");
  73. #endif
  74. return ESP_OK;
  75. }
  76. esp_err_t handler_capture(httpd_req_t *req)
  77. {
  78. #ifdef DEBUG_DETAIL_ON
  79. LogFile.WriteHeapInfo("handler_capture - Start");
  80. #endif
  81. if (Camera.getCameraInitSuccessful())
  82. {
  83. // If the camera settings were changed by creating a new reference image, they must be reset
  84. if (CFstatus.changedCameraSettings)
  85. {
  86. Camera.setSensorDatenFromCCstatus(); // CCstatus >>> Kamera
  87. Camera.SetQualityZoomSize(CCstatus.ImageQuality, CCstatus.ImageFrameSize, CCstatus.ImageZoomEnabled, CCstatus.ImageZoomOffsetX, CCstatus.ImageZoomOffsetY, CCstatus.ImageZoomSize, CCstatus.ImageVflip);
  88. Camera.LedIntensity = CCstatus.ImageLedIntensity;
  89. CFstatus.changedCameraSettings = false;
  90. }
  91. #ifdef DEBUG_DETAIL_ON
  92. ESP_LOGD(TAG, "Size: %d, Quality: %d", CCstatus.ImageFrameSize, CCstatus.ImageQuality);
  93. #endif
  94. esp_err_t result;
  95. result = Camera.CaptureToHTTP(req);
  96. #ifdef DEBUG_DETAIL_ON
  97. LogFile.WriteHeapInfo("handler_capture - Done");
  98. #endif
  99. return result;
  100. }
  101. else
  102. {
  103. httpd_resp_send_err(req, HTTPD_403_FORBIDDEN, "Camera not initialized: REST API /capture not available!");
  104. return ESP_ERR_NOT_FOUND;
  105. }
  106. }
  107. esp_err_t handler_capture_with_light(httpd_req_t *req)
  108. {
  109. #ifdef DEBUG_DETAIL_ON
  110. LogFile.WriteHeapInfo("handler_capture_with_light - Start");
  111. #endif
  112. if (Camera.getCameraInitSuccessful())
  113. {
  114. char _query[100];
  115. char _delay[10];
  116. int delay = 2500;
  117. if (httpd_req_get_url_query_str(req, _query, 100) == ESP_OK)
  118. {
  119. ESP_LOGD(TAG, "Query: %s", _query);
  120. if (httpd_query_key_value(_query, "delay", _delay, 10) == ESP_OK)
  121. {
  122. #ifdef DEBUG_DETAIL_ON
  123. ESP_LOGD(TAG, "Delay: %s", _delay);
  124. #endif
  125. delay = atoi(_delay);
  126. if (delay < 0)
  127. {
  128. delay = 0;
  129. }
  130. }
  131. }
  132. // If the camera settings were changed by creating a new reference image, they must be reset
  133. if (CFstatus.changedCameraSettings)
  134. {
  135. Camera.setSensorDatenFromCCstatus(); // CCstatus >>> Kamera
  136. Camera.SetQualityZoomSize(CCstatus.ImageQuality, CCstatus.ImageFrameSize, CCstatus.ImageZoomEnabled, CCstatus.ImageZoomOffsetX, CCstatus.ImageZoomOffsetY, CCstatus.ImageZoomSize, CCstatus.ImageVflip);
  137. Camera.LedIntensity = CCstatus.ImageLedIntensity;
  138. CFstatus.changedCameraSettings = false;
  139. }
  140. #ifdef DEBUG_DETAIL_ON
  141. ESP_LOGD(TAG, "Size: %d, Quality: %d", CCstatus.ImageFrameSize, CCstatus.ImageQuality);
  142. #endif
  143. Camera.LightOnOff(true);
  144. const TickType_t xDelay = delay / portTICK_PERIOD_MS;
  145. vTaskDelay(xDelay);
  146. esp_err_t result;
  147. result = Camera.CaptureToHTTP(req);
  148. Camera.LightOnOff(false);
  149. #ifdef DEBUG_DETAIL_ON
  150. LogFile.WriteHeapInfo("handler_capture_with_light - Done");
  151. #endif
  152. return result;
  153. }
  154. else
  155. {
  156. httpd_resp_send_err(req, HTTPD_403_FORBIDDEN, "Camera not initialized: REST API /capture_with_flashlight not available!");
  157. return ESP_ERR_NOT_FOUND;
  158. }
  159. }
  160. esp_err_t handler_capture_save_to_file(httpd_req_t *req)
  161. {
  162. #ifdef DEBUG_DETAIL_ON
  163. LogFile.WriteHeapInfo("handler_capture_save_to_file - Start");
  164. #endif
  165. if (Camera.getCameraInitSuccessful())
  166. {
  167. char _query[100];
  168. char _delay[10];
  169. int delay = 0;
  170. char filename[100];
  171. std::string fn = "/sdcard/";
  172. if (httpd_req_get_url_query_str(req, _query, 100) == ESP_OK)
  173. {
  174. ESP_LOGD(TAG, "Query: %s", _query);
  175. if (httpd_query_key_value(_query, "filename", filename, 100) == ESP_OK)
  176. {
  177. fn.append(filename);
  178. #ifdef DEBUG_DETAIL_ON
  179. ESP_LOGD(TAG, "Filename: %s", fn.c_str());
  180. #endif
  181. }
  182. else
  183. {
  184. fn.append("noname.jpg");
  185. }
  186. if (httpd_query_key_value(_query, "delay", _delay, 10) == ESP_OK)
  187. {
  188. #ifdef DEBUG_DETAIL_ON
  189. ESP_LOGD(TAG, "Delay: %s", _delay);
  190. #endif
  191. delay = atoi(_delay);
  192. if (delay < 0)
  193. {
  194. delay = 0;
  195. }
  196. }
  197. }
  198. else
  199. {
  200. fn.append("noname.jpg");
  201. }
  202. // If the camera settings were changed by creating a new reference image, they must be reset
  203. if (CFstatus.changedCameraSettings)
  204. {
  205. Camera.setSensorDatenFromCCstatus(); // CCstatus >>> Kamera
  206. Camera.SetQualityZoomSize(CCstatus.ImageQuality, CCstatus.ImageFrameSize, CCstatus.ImageZoomEnabled, CCstatus.ImageZoomOffsetX, CCstatus.ImageZoomOffsetY, CCstatus.ImageZoomSize, CCstatus.ImageVflip);
  207. Camera.LedIntensity = CCstatus.ImageLedIntensity;
  208. CFstatus.changedCameraSettings = false;
  209. }
  210. #ifdef DEBUG_DETAIL_ON
  211. ESP_LOGD(TAG, "Size: %d, Quality: %d", CCstatus.ImageFrameSize, CCstatus.ImageQuality);
  212. #endif
  213. esp_err_t result;
  214. result = Camera.CaptureToFile(fn, delay);
  215. const char *resp_str = (const char *)fn.c_str();
  216. httpd_resp_send(req, resp_str, strlen(resp_str));
  217. #ifdef DEBUG_DETAIL_ON
  218. LogFile.WriteHeapInfo("handler_capture_save_to_file - Done");
  219. #endif
  220. return result;
  221. }
  222. else
  223. {
  224. httpd_resp_send_err(req, HTTPD_403_FORBIDDEN, "Camera not initialized: REST API /save not available!");
  225. return ESP_ERR_NOT_FOUND;
  226. }
  227. }
  228. void register_server_camera_uri(httpd_handle_t server)
  229. {
  230. #ifdef DEBUG_DETAIL_ON
  231. ESP_LOGI(TAG, "server_part_camera - Registering URI handlers");
  232. #endif
  233. httpd_uri_t camuri = {};
  234. camuri.method = HTTP_GET;
  235. camuri.uri = "/lighton";
  236. camuri.handler = APPLY_BASIC_AUTH_FILTER(handler_lightOn);
  237. camuri.user_ctx = (void *)"Light On";
  238. httpd_register_uri_handler(server, &camuri);
  239. camuri.uri = "/lightoff";
  240. camuri.handler = APPLY_BASIC_AUTH_FILTER(handler_lightOff);
  241. camuri.user_ctx = (void *)"Light Off";
  242. httpd_register_uri_handler(server, &camuri);
  243. camuri.uri = "/capture";
  244. camuri.handler = APPLY_BASIC_AUTH_FILTER(handler_capture);
  245. camuri.user_ctx = NULL;
  246. httpd_register_uri_handler(server, &camuri);
  247. camuri.uri = "/capture_with_flashlight";
  248. camuri.handler = APPLY_BASIC_AUTH_FILTER(handler_capture_with_light);
  249. camuri.user_ctx = NULL;
  250. httpd_register_uri_handler(server, &camuri);
  251. camuri.uri = "/save";
  252. camuri.handler = APPLY_BASIC_AUTH_FILTER(handler_capture_save_to_file);
  253. camuri.user_ctx = NULL;
  254. httpd_register_uri_handler(server, &camuri);
  255. }