tools_http_utils.c 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148
  1. #include "tools_http_utils.h"
  2. #include "esp_http_client.h"
  3. #include "esp_http_server.h"
  4. #include "esp_log.h"
  5. #include "tools.h"
  6. #include "esp_tls.h"
  7. #include "pb_decode.h"
  8. #include "pb_encode.h"
  9. static const char* TAG = "http_utils";
  10. /****************************************************************************************
  11. * URL download
  12. */
  13. typedef struct {
  14. void* user_context;
  15. http_download_cb_t callback;
  16. size_t max, bytes;
  17. bool abort;
  18. uint8_t* data;
  19. esp_http_client_handle_t client;
  20. } http_context_t;
  21. static void http_downloader(void* arg);
  22. static esp_err_t http_event_handler(esp_http_client_event_t* evt);
  23. void http_download(char* url, size_t max, http_download_cb_t callback, void* context) {
  24. http_context_t* http_context =
  25. (http_context_t*)heap_caps_calloc(sizeof(http_context_t), 1, MALLOC_CAP_SPIRAM);
  26. esp_http_client_config_t config = {
  27. .url = url,
  28. .event_handler = http_event_handler,
  29. .user_data = http_context,
  30. };
  31. http_context->callback = callback;
  32. http_context->user_context = context;
  33. http_context->max = max;
  34. http_context->client = esp_http_client_init(&config);
  35. xTaskCreateEXTRAM(
  36. http_downloader, "downloader", 8 * 1024, http_context, ESP_TASK_PRIO_MIN + 1, NULL);
  37. }
  38. static void http_downloader(void* arg) {
  39. http_context_t* http_context = (http_context_t*)arg;
  40. esp_http_client_perform(http_context->client);
  41. esp_http_client_cleanup(http_context->client);
  42. free(http_context);
  43. vTaskDeleteEXTRAM(NULL);
  44. }
  45. static esp_err_t http_event_handler(esp_http_client_event_t* evt) {
  46. http_context_t* http_context = (http_context_t*)evt->user_data;
  47. if (http_context->abort) return ESP_FAIL;
  48. switch (evt->event_id) {
  49. case HTTP_EVENT_ERROR:
  50. http_context->callback(NULL, 0, http_context->user_context);
  51. http_context->abort = true;
  52. break;
  53. case HTTP_EVENT_ON_HEADER:
  54. if (!strcasecmp(evt->header_key, "Content-Length")) {
  55. size_t len = atoi(evt->header_value);
  56. if (!len || len > http_context->max) {
  57. ESP_LOGI(TAG, "content-length null or too large %zu / %zu", len, http_context->max);
  58. http_context->abort = true;
  59. }
  60. }
  61. break;
  62. case HTTP_EVENT_ON_DATA: {
  63. size_t len = esp_http_client_get_content_length(evt->client);
  64. if (!http_context->data) {
  65. if ((http_context->data = (uint8_t*)malloc(len)) == NULL) {
  66. http_context->abort = true;
  67. ESP_LOGE(TAG, "failed to allocate memory for output buffer %zu", len);
  68. return ESP_FAIL;
  69. }
  70. }
  71. memcpy(http_context->data + http_context->bytes, evt->data, evt->data_len);
  72. http_context->bytes += evt->data_len;
  73. break;
  74. }
  75. case HTTP_EVENT_ON_FINISH:
  76. http_context->callback(http_context->data, http_context->bytes, http_context->user_context);
  77. break;
  78. case HTTP_EVENT_DISCONNECTED: {
  79. int mbedtls_err = 0;
  80. esp_err_t err = esp_tls_get_and_clear_last_error(evt->data, &mbedtls_err, NULL);
  81. if (err != ESP_OK) {
  82. ESP_LOGE(TAG, "HTTP download disconnect %d", err);
  83. if (http_context->data) free(http_context->data);
  84. http_context->callback(NULL, 0, http_context->user_context);
  85. return ESP_FAIL;
  86. }
  87. break;
  88. }
  89. default:
  90. break;
  91. }
  92. return ESP_OK;
  93. }
  94. /****************************************************************************************
  95. * URL tools
  96. */
  97. static inline char from_hex(char ch) { return isdigit(ch) ? ch - '0' : tolower(ch) - 'a' + 10; }
  98. void url_decode(char* url) {
  99. char *p, *src = strdup(url);
  100. for (p = src; *src; url++) {
  101. *url = *src++;
  102. if (*url == '%') {
  103. *url = from_hex(*src++) << 4;
  104. *url |= from_hex(*src++);
  105. } else if (*url == '+') {
  106. *url = ' ';
  107. }
  108. }
  109. *url = '\0';
  110. free(p);
  111. }
  112. bool out_http_binding(pb_ostream_t* stream, const uint8_t* buf, size_t count) {
  113. httpd_req_t* req = (httpd_req_t*)stream->state;
  114. ESP_LOGD(TAG, "Writing %d bytes to file", count);
  115. return httpd_resp_send_chunk(req, (const char*)buf, count) == ESP_OK;
  116. }
  117. bool in_http_binding(pb_istream_t* stream, pb_byte_t* buf, size_t count) {
  118. httpd_req_t* req = (httpd_req_t*)stream->state;
  119. ESP_LOGV(TAG, "Reading %d bytes from http stream", count);
  120. int received = httpd_req_recv(req, (char*)buf, count);
  121. if (received <= 0) {
  122. stream->errmsg = "Not all data received";
  123. return false;
  124. }
  125. return received==count;
  126. }