ESPHome 2026.5.0-dev
Loading...
Searching...
No Matches
http_request_idf.cpp
Go to the documentation of this file.
1#include "http_request_idf.h"
2
3#ifdef USE_ESP32
4
7
9#include "esphome/core/log.h"
10
11#if CONFIG_MBEDTLS_CERTIFICATE_BUNDLE
12#include "esp_crt_bundle.h"
13#endif
14
15#include "esp_task_wdt.h"
16
17namespace esphome::http_request {
18
19static const char *const TAG = "http_request.idf";
20static constexpr uint32_t ERROR_DURATION_MS = 1000;
21
22struct UserData {
23 const std::vector<std::string> &lower_case_collect_headers;
24 std::vector<Header> &response_headers;
25};
26
29 ESP_LOGCONFIG(TAG,
30 " Buffer Size RX: %u\n"
31 " Buffer Size TX: %u\n"
32 " Custom CA Certificate: %s",
33 this->buffer_size_rx_, this->buffer_size_tx_, YESNO(this->ca_certificate_ != nullptr));
34}
35
36esp_err_t HttpRequestIDF::http_event_handler(esp_http_client_event_t *evt) {
37 UserData *user_data = (UserData *) evt->user_data;
38
39 switch (evt->event_id) {
40 case HTTP_EVENT_ON_HEADER: {
41 const std::string header_name = str_lower_case(evt->header_key);
42 if (should_collect_header(user_data->lower_case_collect_headers, header_name)) {
43 const std::string header_value = evt->header_value;
44 ESP_LOGD(TAG, "Received response header, name: %s, value: %s", header_name.c_str(), header_value.c_str());
45 user_data->response_headers.push_back({header_name, header_value});
46 }
47 break;
48 }
49 default: {
50 break;
51 }
52 }
53 return ESP_OK;
54}
55
56std::shared_ptr<HttpContainer> HttpRequestIDF::perform(const std::string &url, const std::string &method,
57 const std::string &body,
58 const std::vector<Header> &request_headers,
59 const std::vector<std::string> &lower_case_collect_headers) {
60 if (!network::is_connected()) {
61 this->status_momentary_error("failed", ERROR_DURATION_MS);
62 ESP_LOGE(TAG, "HTTP Request failed; Not connected to network");
63 return nullptr;
64 }
65
66 esp_http_client_method_t method_idf;
67 if (method == "GET") {
68 method_idf = HTTP_METHOD_GET;
69 } else if (method == "POST") {
70 method_idf = HTTP_METHOD_POST;
71 } else if (method == "PUT") {
72 method_idf = HTTP_METHOD_PUT;
73 } else if (method == "DELETE") {
74 method_idf = HTTP_METHOD_DELETE;
75 } else if (method == "PATCH") {
76 method_idf = HTTP_METHOD_PATCH;
77 } else {
78 this->status_momentary_error("failed", ERROR_DURATION_MS);
79 ESP_LOGE(TAG, "HTTP Request failed; Unsupported method");
80 return nullptr;
81 }
82
83 bool secure = url.find("https:") != std::string::npos;
84
85 esp_http_client_config_t config = {};
86
87 config.url = url.c_str();
88 config.method = method_idf;
89 config.timeout_ms = this->timeout_;
90 config.disable_auto_redirect = !this->follow_redirects_;
91 config.max_redirection_count = this->redirect_limit_;
92 config.auth_type = HTTP_AUTH_TYPE_BASIC;
93 if (secure && this->verify_ssl_) {
94 if (this->ca_certificate_ != nullptr) {
95 config.cert_pem = this->ca_certificate_;
96#if CONFIG_MBEDTLS_CERTIFICATE_BUNDLE
97 } else {
98 config.crt_bundle_attach = esp_crt_bundle_attach;
99#endif
100 }
101 }
102
103 if (this->useragent_ != nullptr) {
104 config.user_agent = this->useragent_;
105 }
106
107 config.buffer_size = this->buffer_size_rx_;
108 config.buffer_size_tx = this->buffer_size_tx_;
109
110 const uint32_t start = millis();
112
113 config.event_handler = http_event_handler;
114
115 esp_http_client_handle_t client = esp_http_client_init(&config);
116 if (client == nullptr) {
117 this->status_momentary_error("failed", ERROR_DURATION_MS);
118 ESP_LOGE(TAG, "HTTP Request failed; client could not be initialized");
119 return nullptr;
120 }
121
122 std::shared_ptr<HttpContainerIDF> container = std::make_shared<HttpContainerIDF>(client);
123 container->set_parent(this);
124
125 container->set_secure(secure);
126
127 auto user_data = UserData{lower_case_collect_headers, container->response_headers_};
128 esp_http_client_set_user_data(client, static_cast<void *>(&user_data));
129
130 for (const auto &header : request_headers) {
131 esp_http_client_set_header(client, header.name.c_str(), header.value.c_str());
132 }
133
134 const int body_len = body.length();
135
136 esp_err_t err = esp_http_client_open(client, body_len);
137 if (err != ESP_OK) {
138 this->status_momentary_error("failed", ERROR_DURATION_MS);
139 ESP_LOGE(TAG, "HTTP Request failed: %s", esp_err_to_name(err));
140 esp_http_client_cleanup(client);
141 return nullptr;
142 }
143
144 if (body_len > 0) {
145 int write_left = body_len;
146 int write_index = 0;
147 const char *buf = body.c_str();
148 while (write_left > 0) {
149 int written = esp_http_client_write(client, buf + write_index, write_left);
150 if (written < 0) {
151 err = ESP_FAIL;
152 break;
153 }
154 write_left -= written;
155 write_index += written;
156 }
157 }
158
159 if (err != ESP_OK) {
160 this->status_momentary_error("failed", ERROR_DURATION_MS);
161 ESP_LOGE(TAG, "HTTP Request failed: %s", esp_err_to_name(err));
162 esp_http_client_cleanup(client);
163 return nullptr;
164 }
165
166 container->feed_wdt();
167 // esp_http_client_fetch_headers() returns 0 for chunked transfer encoding (no Content-Length header).
168 // The read() method handles content_length == 0 specially to support chunked responses.
169 container->content_length = esp_http_client_fetch_headers(client);
170 container->set_chunked(esp_http_client_is_chunked_response(client));
171 container->feed_wdt();
172 container->status_code = esp_http_client_get_status_code(client);
173 container->feed_wdt();
174 container->duration_ms = millis() - start;
175 if (is_success(container->status_code)) {
176 return container;
177 }
178
179 if (this->follow_redirects_) {
180 auto num_redirects = this->redirect_limit_;
181 while (is_redirect(container->status_code) && num_redirects > 0) {
182 err = esp_http_client_set_redirection(client);
183 if (err != ESP_OK) {
184 ESP_LOGE(TAG, "esp_http_client_set_redirection failed: %s", esp_err_to_name(err));
185 this->status_momentary_error("failed", ERROR_DURATION_MS);
186 esp_http_client_cleanup(client);
187 return nullptr;
188 }
189#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
190 char redirect_url[256]{};
191 if (esp_http_client_get_url(client, redirect_url, sizeof(redirect_url) - 1) == ESP_OK) {
192 ESP_LOGV(TAG, "redirecting to url: %s", redirect_url);
193 }
194#endif
195 err = esp_http_client_open(client, 0);
196 if (err != ESP_OK) {
197 ESP_LOGE(TAG, "esp_http_client_open failed: %s", esp_err_to_name(err));
198 this->status_momentary_error("failed", ERROR_DURATION_MS);
199 esp_http_client_cleanup(client);
200 return nullptr;
201 }
202
203 container->feed_wdt();
204 container->content_length = esp_http_client_fetch_headers(client);
205 container->set_chunked(esp_http_client_is_chunked_response(client));
206 container->feed_wdt();
207 container->status_code = esp_http_client_get_status_code(client);
208 container->feed_wdt();
209 container->duration_ms = millis() - start;
210 if (is_success(container->status_code)) {
211 return container;
212 }
213
214 num_redirects--;
215 }
216
217 if (num_redirects == 0) {
218 ESP_LOGW(TAG, "Reach redirect limit count=%d", this->redirect_limit_);
219 }
220 }
221
222 ESP_LOGE(TAG, "HTTP Request failed; URL: %s; Code: %d", url.c_str(), container->status_code);
223 this->status_momentary_error("failed", ERROR_DURATION_MS);
224 return container;
225}
226
228 // Base class handles no-body status codes and non-chunked content_length completion
230 return true;
231 }
232 // For chunked responses, use the authoritative ESP-IDF completion check
233 return this->is_chunked_ && esp_http_client_is_complete_data_received(this->client_);
234}
235
236// ESP-IDF HTTP read implementation (blocking mode)
237//
238// WARNING: Return values differ from BSD sockets! See http_request.h for full documentation.
239//
240// esp_http_client_read() in blocking mode returns:
241// > 0: bytes read
242// 0: all chunked data received (is_chunk_complete true) or connection closed
243// -ESP_ERR_HTTP_EAGAIN: transport timeout, no data available yet
244// < 0: error
245//
246// We normalize to HttpContainer::read() contract:
247// > 0: bytes read
248// 0: all content read (for both content_length-based and chunked completion)
249// < 0: error/connection closed
250//
251// Note on chunked transfer encoding:
252// esp_http_client_fetch_headers() returns 0 for chunked responses (no Content-Length header).
253// When esp_http_client_read() returns 0 for a chunked response, is_read_complete() calls
254// esp_http_client_is_complete_data_received() to distinguish successful completion from
255// connection errors. Callers use http_read_loop_result() which checks is_read_complete()
256// to return COMPLETE for successful chunked EOF.
257//
258// Streaming chunked responses are not supported (see http_request.h for details).
259// When data stops arriving, esp_http_client_read() returns -ESP_ERR_HTTP_EAGAIN
260// after its internal transport timeout (configured via timeout_ms) expires.
261// This is passed through as a negative return value, which callers treat as an error.
262int HttpContainerIDF::read(uint8_t *buf, size_t max_len) {
263 const uint32_t start = millis();
264 watchdog::WatchdogManager wdm(this->parent_->get_watchdog_timeout());
265
266 // Check if we've already read all expected content (non-chunked and no-body only).
267 // Use the base class check here, NOT the override: esp_http_client_is_complete_data_received()
268 // returns true as soon as all data arrives from the network, but data may still be in
269 // the client's internal buffer waiting to be consumed by esp_http_client_read().
271 return 0; // All content read successfully
272 }
273
274 this->feed_wdt();
275 int read_len_or_error = esp_http_client_read(this->client_, (char *) buf, max_len);
276 this->feed_wdt();
277
278 this->duration_ms += (millis() - start);
279
280 if (read_len_or_error > 0) {
281 this->bytes_read_ += read_len_or_error;
282 return read_len_or_error;
283 }
284
285 // esp_http_client_read() returns 0 when:
286 // - Known content_length: connection closed before all data received (error)
287 // - Chunked encoding: all chunks received (is_chunk_complete true, genuine EOF)
288 //
289 // Return 0 in both cases. Callers use http_read_loop_result() which calls
290 // is_read_complete() to distinguish these:
291 // - Chunked complete: is_read_complete() returns true (via
292 // esp_http_client_is_complete_data_received()), caller gets COMPLETE
293 // - Non-chunked incomplete: is_read_complete() returns false, caller
294 // eventually gets TIMEOUT (since no more data arrives)
295 if (read_len_or_error == 0) {
296 return 0;
297 }
298
299 // Negative value - error, return the actual error code for debugging
300 return read_len_or_error;
301}
302
304 if (this->client_ == nullptr) {
305 return; // Already cleaned up
306 }
307 watchdog::WatchdogManager wdm(this->parent_->get_watchdog_timeout());
308
309 esp_http_client_close(this->client_);
310 esp_http_client_cleanup(this->client_);
311 this->client_ = nullptr;
312}
313
315 // Tests to see if the executing task has a watchdog timer attached
316 if (esp_task_wdt_status(nullptr) == ESP_OK) {
317 App.feed_wdt();
318 }
319}
320
321} // namespace esphome::http_request
322
323#endif // USE_ESP32
void feed_wdt()
Feed the task watchdog.
void status_momentary_error(const char *name, uint32_t length=5000)
Set error status flag and automatically clear it after a timeout.
virtual bool is_read_complete() const
Check if all expected content has been read.
bool is_chunked_
True if response uses chunked transfer encoding.
int read(uint8_t *buf, size_t max_len) override
void feed_wdt()
Feeds the watchdog timer if the executing task has one attached.
std::shared_ptr< HttpContainer > start(const std::string &url, const std::string &method, const std::string &body, const std::vector< Header > &request_headers)
std::shared_ptr< HttpContainer > perform(const std::string &url, const std::string &method, const std::string &body, const std::vector< Header > &request_headers, const std::vector< std::string > &lower_case_collect_headers) override
static esp_err_t http_event_handler(esp_http_client_event_t *evt)
Monitors the http client events to gather response headers.
bool should_collect_header(const std::vector< std::string > &lower_case_collect_headers, const std::string &lower_header_name)
Check if a header name should be collected (linear scan, fine for small lists)
bool is_success(int const status)
Checks if the given HTTP status code indicates a successful request.
bool is_redirect(int const status)
Returns true if the HTTP status code is a redirect.
ESPHOME_ALWAYS_INLINE bool is_connected()
Return whether the node is connected to the network (through wifi, eth, ...)
Definition util.h:27
std::string str_lower_case(const std::string &str)
Convert the string to lower case.
Definition helpers.cpp:240
uint32_t IRAM_ATTR HOT millis()
Definition core.cpp:26
int written
Definition helpers.h:1089
Application App
Global storage of Application pointer - only one Application can exist.
static void uint32_t