server_help.cpp 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150
  1. #include "server_help.h"
  2. #include <stdio.h>
  3. #include <string.h>
  4. #include <sys/param.h>
  5. #include <sys/unistd.h>
  6. #include <sys/stat.h>
  7. #ifdef __cplusplus
  8. extern "C" {
  9. #endif
  10. #include <dirent.h>
  11. #ifdef __cplusplus
  12. }
  13. #endif
  14. #include "esp_err.h"
  15. #include "esp_log.h"
  16. #include "Helper.h"
  17. #include "esp_http_server.h"
  18. #include "../../include/defines.h"
  19. static const char *TAG = "SERVER HELP";
  20. char scratch[SERVER_HELPER_SCRATCH_BUFSIZE];
  21. bool endsWith(std::string const &str, std::string const &suffix) {
  22. if (str.length() < suffix.length()) {
  23. return false;
  24. }
  25. return str.compare(str.length() - suffix.length(), suffix.length(), suffix) == 0;
  26. }
  27. esp_err_t send_file(httpd_req_t *req, std::string filename)
  28. {
  29. FILE *fd = fopen(filename.c_str(), "r");
  30. if (!fd) {
  31. ESP_LOGE(TAG, "Failed to read file: %s", filename.c_str());
  32. /* Respond with 404 Error */
  33. httpd_resp_send_err(req, HTTPD_404_NOT_FOUND, get404());
  34. return ESP_FAIL;
  35. }
  36. ESP_LOGD(TAG, "Sending file: %s ...", filename.c_str());
  37. // httpd_resp_set_hdr(req, "Access-Control-Allow-Origin", "*");
  38. /* For all files with the following file extention tell
  39. the webbrowser to cache them for 24h */
  40. if (endsWith(filename, ".html") ||
  41. endsWith(filename, ".htm") ||
  42. endsWith(filename, ".css") ||
  43. endsWith(filename, ".js") ||
  44. endsWith(filename, ".map") ||
  45. endsWith(filename, ".jpg") ||
  46. endsWith(filename, ".jpeg") ||
  47. endsWith(filename, ".ico") ||
  48. endsWith(filename, ".png")) {
  49. httpd_resp_set_hdr(req, "Cache-Control", "max-age=86400");
  50. }
  51. set_content_type_from_file(req, filename.c_str());
  52. /* Retrieve the pointer to scratch buffer for temporary storage */
  53. char *chunk = scratch;
  54. size_t chunksize;
  55. do {
  56. /* Read file in chunks into the scratch buffer */
  57. chunksize = fread(chunk, 1, SERVER_HELPER_SCRATCH_BUFSIZE, fd);
  58. /* Send the buffer contents as HTTP response chunk */
  59. if (httpd_resp_send_chunk(req, chunk, chunksize) != ESP_OK) {
  60. fclose(fd);
  61. ESP_LOGE(TAG, "File sending failed!");
  62. /* Abort sending file */
  63. httpd_resp_sendstr_chunk(req, NULL);
  64. /* Respond with 500 Internal Server Error */
  65. httpd_resp_send_err(req, HTTPD_500_INTERNAL_SERVER_ERROR, "Failed to send file");
  66. return ESP_FAIL;
  67. }
  68. /* Keep looping till the whole file is sent */
  69. } while (chunksize != 0);
  70. /* Close file after sending complete */
  71. fclose(fd);
  72. ESP_LOGD(TAG, "File sending complete");
  73. return ESP_OK;
  74. }
  75. /* Copies the full path into destination buffer and returns
  76. * pointer to path (skipping the preceding base path) */
  77. const char* get_path_from_uri(char *dest, const char *base_path, const char *uri, size_t destsize)
  78. {
  79. const size_t base_pathlen = strlen(base_path);
  80. size_t pathlen = strlen(uri);
  81. const char *quest = strchr(uri, '?');
  82. if (quest) {
  83. pathlen = MIN(pathlen, quest - uri);
  84. }
  85. const char *hash = strchr(uri, '#');
  86. if (hash) {
  87. pathlen = MIN(pathlen, hash - uri);
  88. }
  89. if (base_pathlen + pathlen + 1 > destsize) {
  90. /* Full path string won't fit into destination buffer */
  91. return NULL;
  92. }
  93. /* Construct full path (base + path) */
  94. strcpy(dest, base_path);
  95. strlcpy(dest + base_pathlen, uri, pathlen + 1);
  96. /* Return pointer to path, skipping the base */
  97. return dest + base_pathlen;
  98. }
  99. /* Set HTTP response content type according to file extension */
  100. esp_err_t set_content_type_from_file(httpd_req_t *req, const char *filename)
  101. {
  102. if (IS_FILE_EXT(filename, ".pdf")) {
  103. return httpd_resp_set_type(req, "application/pdf");
  104. } else if (IS_FILE_EXT(filename, ".html")) {
  105. return httpd_resp_set_type(req, "text/html");
  106. } else if (IS_FILE_EXT(filename, ".jpeg")) {
  107. return httpd_resp_set_type(req, "image/jpeg");
  108. } else if (IS_FILE_EXT(filename, ".jpg")) {
  109. return httpd_resp_set_type(req, "image/jpeg");
  110. } else if (IS_FILE_EXT(filename, ".ico")) {
  111. return httpd_resp_set_type(req, "image/x-icon");
  112. } else if (IS_FILE_EXT(filename, ".js")) {
  113. return httpd_resp_set_type(req, "text/javascript");
  114. } else if (IS_FILE_EXT(filename, ".css")) {
  115. return httpd_resp_set_type(req, "text/css");
  116. }
  117. /* This is a limited set only */
  118. /* For any other type always set as plain text */
  119. return httpd_resp_set_type(req, "text/plain");
  120. }