ESPHome 2026.5.0-dev
Loading...
Searching...
No Matches
api_frame_helper.cpp
Go to the documentation of this file.
1#include "api_frame_helper.h"
2#ifdef USE_API
4#include "esphome/core/hal.h"
6#include "esphome/core/log.h"
7#include "proto.h"
8#include <cstring>
9#include <cinttypes>
10
11namespace esphome::api {
12
13static const char *const TAG = "api.frame_helper";
14
15// Maximum bytes to log in hex format (168 * 3 = 504, under TX buffer size of 512)
16static constexpr size_t API_MAX_LOG_BYTES = 168;
17
18#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERY_VERBOSE
19#define HELPER_LOG(msg, ...) \
20 do { \
21 char peername_buf[socket::SOCKADDR_STR_LEN]; \
22 this->get_peername_to(peername_buf); \
23 ESP_LOGVV(TAG, "%s (%s): " msg, this->client_name_, peername_buf, ##__VA_ARGS__); \
24 } while (0)
25#else
26#define HELPER_LOG(msg, ...) ((void) 0)
27#endif
28
29#ifdef HELPER_LOG_PACKETS
30#define LOG_PACKET_RECEIVED(buffer) \
31 do { \
32 char hex_buf_[format_hex_pretty_size(API_MAX_LOG_BYTES)]; \
33 ESP_LOGVV(TAG, "Received frame: %s", \
34 format_hex_pretty_to(hex_buf_, (buffer).data(), \
35 (buffer).size() < API_MAX_LOG_BYTES ? (buffer).size() : API_MAX_LOG_BYTES)); \
36 } while (0)
37#define LOG_PACKET_SENDING(data, len) \
38 do { \
39 char hex_buf_[format_hex_pretty_size(API_MAX_LOG_BYTES)]; \
40 ESP_LOGVV(TAG, "Sending raw: %s", \
41 format_hex_pretty_to(hex_buf_, data, (len) < API_MAX_LOG_BYTES ? (len) : API_MAX_LOG_BYTES)); \
42 } while (0)
43#else
44#define LOG_PACKET_RECEIVED(buffer) ((void) 0)
45#define LOG_PACKET_SENDING(data, len) ((void) 0)
46#endif
47
48const LogString *api_error_to_logstr(APIError err) {
49 // not using switch to ensure compiler doesn't try to build a big table out of it
50 if (err == APIError::OK) {
51 return LOG_STR("OK");
52 } else if (err == APIError::WOULD_BLOCK) {
53 return LOG_STR("WOULD_BLOCK");
54 } else if (err == APIError::BAD_INDICATOR) {
55 return LOG_STR("BAD_INDICATOR");
56 } else if (err == APIError::BAD_DATA_PACKET) {
57 return LOG_STR("BAD_DATA_PACKET");
58 } else if (err == APIError::TCP_NODELAY_FAILED) {
59 return LOG_STR("TCP_NODELAY_FAILED");
60 } else if (err == APIError::TCP_NONBLOCKING_FAILED) {
61 return LOG_STR("TCP_NONBLOCKING_FAILED");
62 } else if (err == APIError::CLOSE_FAILED) {
63 return LOG_STR("CLOSE_FAILED");
64 } else if (err == APIError::SHUTDOWN_FAILED) {
65 return LOG_STR("SHUTDOWN_FAILED");
66 } else if (err == APIError::BAD_STATE) {
67 return LOG_STR("BAD_STATE");
68 } else if (err == APIError::BAD_ARG) {
69 return LOG_STR("BAD_ARG");
70 } else if (err == APIError::SOCKET_READ_FAILED) {
71 return LOG_STR("SOCKET_READ_FAILED");
72 } else if (err == APIError::SOCKET_WRITE_FAILED) {
73 return LOG_STR("SOCKET_WRITE_FAILED");
74 } else if (err == APIError::OUT_OF_MEMORY) {
75 return LOG_STR("OUT_OF_MEMORY");
76 } else if (err == APIError::CONNECTION_CLOSED) {
77 return LOG_STR("CONNECTION_CLOSED");
78 }
79#ifdef USE_API_NOISE
80 else if (err == APIError::BAD_HANDSHAKE_PACKET_LEN) {
81 return LOG_STR("BAD_HANDSHAKE_PACKET_LEN");
82 } else if (err == APIError::HANDSHAKESTATE_READ_FAILED) {
83 return LOG_STR("HANDSHAKESTATE_READ_FAILED");
84 } else if (err == APIError::HANDSHAKESTATE_WRITE_FAILED) {
85 return LOG_STR("HANDSHAKESTATE_WRITE_FAILED");
86 } else if (err == APIError::HANDSHAKESTATE_BAD_STATE) {
87 return LOG_STR("HANDSHAKESTATE_BAD_STATE");
88 } else if (err == APIError::CIPHERSTATE_DECRYPT_FAILED) {
89 return LOG_STR("CIPHERSTATE_DECRYPT_FAILED");
90 } else if (err == APIError::CIPHERSTATE_ENCRYPT_FAILED) {
91 return LOG_STR("CIPHERSTATE_ENCRYPT_FAILED");
92 } else if (err == APIError::HANDSHAKESTATE_SETUP_FAILED) {
93 return LOG_STR("HANDSHAKESTATE_SETUP_FAILED");
94 } else if (err == APIError::HANDSHAKESTATE_SPLIT_FAILED) {
95 return LOG_STR("HANDSHAKESTATE_SPLIT_FAILED");
96 } else if (err == APIError::BAD_HANDSHAKE_ERROR_BYTE) {
97 return LOG_STR("BAD_HANDSHAKE_ERROR_BYTE");
98 }
99#endif
100 return LOG_STR("UNKNOWN");
101}
102
103#ifdef HELPER_LOG_PACKETS
104void APIFrameHelper::log_packet_sending_(const void *data, uint16_t len) {
105 LOG_PACKET_SENDING(reinterpret_cast<const uint8_t *>(data), len);
106}
107#endif
108
110 if (this->overflow_buf_.try_drain(this->socket_.get()) == -1) {
111 int err = errno;
112 if (err != EWOULDBLOCK && err != EAGAIN) {
113 this->state_ = State::FAILED;
114 HELPER_LOG("Socket write failed with errno %d", err);
116 }
117 }
118 return APIError::OK;
119}
120
121// Single-buffer write path: wraps in iovec and delegates.
122APIError APIFrameHelper::write_raw_buf_(const void *data, uint16_t len, ssize_t sent) {
123 struct iovec iov = {const_cast<void *>(data), len};
124 APIError err = this->write_raw_iov_(&iov, 1, len, sent);
125#ifdef HELPER_LOG_PACKETS
126 // Log after write/enqueue so re-entrant log sends can't corrupt data before it's sent
127 if (err == APIError::OK)
128 LOG_PACKET_SENDING(reinterpret_cast<const uint8_t *>(data), len);
129#endif
130 return err;
131}
132
133// Handles partial writes, errors, and overflow buffering.
134// Called when the inline fast path couldn't complete the write,
135// or directly from cold paths (handshake, error handling).
136APIError APIFrameHelper::write_raw_iov_(const struct iovec *iov, int iovcnt, uint16_t total_write_len, ssize_t sent) {
137 if (sent <= 0) {
138 if (sent == WRITE_NOT_ATTEMPTED) {
139 // Cold path: no write attempted yet, drain overflow and try
140 if (!this->overflow_buf_.empty()) {
142 if (err != APIError::OK)
143 return err;
144 }
145 if (this->overflow_buf_.empty()) {
146 sent = this->write_iov_to_socket_(iov, iovcnt);
147 if (sent == static_cast<ssize_t>(total_write_len))
148 return APIError::OK;
149 // Partial write or -1: fall through to error check / enqueue below
150 } else {
151 // Overflow backlog remains after drain; skip socket write, enqueue everything
152 sent = 0;
153 }
154 }
155 // WRITE_FAILED (-1): fast path or retry write returned -1, check errno
156 if (sent == WRITE_FAILED) {
157 int err = errno;
158 if (err != EWOULDBLOCK && err != EAGAIN) {
159 this->state_ = State::FAILED;
160 HELPER_LOG("Socket write failed with errno %d", err);
162 }
163 sent = 0; // Treat WOULD_BLOCK as zero bytes sent
164 }
165 }
166
167 // Full write completed (possible when called directly, not via write_raw_fast_buf_)
168 if (sent == static_cast<ssize_t>(total_write_len))
169 return APIError::OK;
170
171 // Queue unsent data into overflow buffer
172 if (!this->overflow_buf_.enqueue_iov(iov, iovcnt, total_write_len, static_cast<uint16_t>(sent))) {
173 HELPER_LOG("Overflow buffer full, dropping connection");
174 this->state_ = State::FAILED;
176 }
177 return APIError::OK;
178}
179
180const char *APIFrameHelper::get_peername_to(std::span<char, socket::SOCKADDR_STR_LEN> buf) const {
181 if (this->socket_) {
182 this->socket_->getpeername_to(buf);
183 } else {
184 buf[0] = '\0';
185 }
186 return buf.data();
187}
188
190 if (state_ != State::INITIALIZE || this->socket_ == nullptr) {
191 HELPER_LOG("Bad state for init %d", (int) state_);
192 return APIError::BAD_STATE;
193 }
194 int err = this->socket_->setblocking(false);
195 if (err != 0) {
197 HELPER_LOG("Setting nonblocking failed with errno %d", errno);
199 }
200
201 int enable = 1;
202 err = this->socket_->setsockopt(IPPROTO_TCP, TCP_NODELAY, &enable, sizeof(int));
203 if (err != 0) {
205 HELPER_LOG("Setting nodelay failed with errno %d", errno);
207 }
208 return APIError::OK;
209}
210
212 if (received == -1) {
213 const int err = errno;
214 if (err == EWOULDBLOCK || err == EAGAIN) {
216 }
218 HELPER_LOG("Socket read failed with errno %d", err);
220 } else if (received == 0) {
222 HELPER_LOG("Connection closed");
224 }
225 return APIError::OK;
226}
227
228} // namespace esphome::api
229#endif
APIError handle_socket_read_result_(ssize_t received)
void log_packet_sending_(const void *data, uint16_t len)
APIError write_raw_buf_(const void *data, uint16_t len, ssize_t sent=WRITE_NOT_ATTEMPTED)
static constexpr ssize_t WRITE_FAILED
APIError write_raw_iov_(const struct iovec *iov, int iovcnt, uint16_t total_write_len, ssize_t sent=WRITE_NOT_ATTEMPTED)
static constexpr ssize_t WRITE_NOT_ATTEMPTED
const char * get_peername_to(std::span< char, socket::SOCKADDR_STR_LEN > buf) const
std::unique_ptr< socket::Socket > socket_
ssize_t ESPHOME_ALWAYS_INLINE write_iov_to_socket_(const struct iovec *iov, int iovcnt)
bool empty() const
True when no backlogged data is waiting.
bool enqueue_iov(const struct iovec *iov, int iovcnt, uint16_t total_len, uint16_t skip)
Enqueue unsent IOV data into the backlog.
ssize_t try_drain(socket::Socket *socket)
Try to drain queued data to the socket.
__int64 ssize_t
Definition httplib.h:178
const LogString * api_error_to_logstr(APIError err)
std::string size_t len
Definition helpers.h:1045