6#ifdef USE_API_PLAINTEXT
9#ifdef USE_API_USER_DEFINED_ACTIONS
31#ifdef USE_HOMEASSISTANT_TIME
34#ifdef USE_BLUETOOTH_PROXY
40#ifdef USE_VOICE_ASSISTANT
46#ifdef USE_WATER_HEATER
59static constexpr uint8_t MAX_MESSAGES_PER_LOOP = 10;
60static constexpr uint8_t MAX_PING_RETRIES = 60;
61static constexpr uint16_t PING_RETRY_INTERVAL = 1000;
62static constexpr uint32_t KEEPALIVE_DISCONNECT_TIMEOUT = (KEEPALIVE_TIMEOUT_MS * 5) / 2;
71static constexpr uint32_t HANDSHAKE_TIMEOUT_MS = 60000;
76static_assert(MAC_ADDRESS_PRETTY_BUFFER_SIZE - 1 == 17,
77 "Update max_data_length for mac_address/bluetooth_mac_address in api.proto");
79static_assert(
sizeof(ESPHOME_VERSION) - 1 <= 32,
"Update max_data_length for esphome_version in api.proto");
80static_assert(ESPHOME_DEVICE_NAME_MAX_LEN <= 31,
"Update max_data_length for name in api.proto");
81static_assert(ESPHOME_FRIENDLY_NAME_MAX_LEN <= 120,
"Update max_data_length for friendly_name in api.proto");
83static const char *
const TAG =
"api.connection";
85static const int CAMERA_STOP_STREAM = 5000;
91#define ENTITY_COMMAND_MAKE_CALL(entity_type, entity_var, getter_name) \
92 entity_type *entity_var = App.get_##getter_name##_by_key(msg.key, msg.device_id); \
93 if ((entity_var) == nullptr) \
95 auto call = (entity_var)->make_call();
99#define ENTITY_COMMAND_GET(entity_type, entity_var, getter_name) \
100 entity_type *entity_var = App.get_##getter_name##_by_key(msg.key, msg.device_id); \
101 if ((entity_var) == nullptr) \
106#define ENTITY_COMMAND_MAKE_CALL(entity_type, entity_var, getter_name) \
107 entity_type *entity_var = App.get_##getter_name##_by_key(msg.key); \
108 if ((entity_var) == nullptr) \
110 auto call = (entity_var)->make_call();
114#define ENTITY_COMMAND_GET(entity_type, entity_var, getter_name) \
115 entity_type *entity_var = App.get_##getter_name##_by_key(msg.key); \
116 if ((entity_var) == nullptr) \
121#if defined(USE_API_PLAINTEXT) && defined(USE_API_NOISE)
123 if (noise_ctx.has_psk()) {
124 this->helper_ = std::unique_ptr<APIFrameHelper>{new APINoiseFrameHelper(std::move(sock), noise_ctx)};
128#elif defined(USE_API_PLAINTEXT)
129 this->helper_ = std::unique_ptr<APIPlaintextFrameHelper>{
new APIPlaintextFrameHelper(std::move(sock))};
130#elif defined(USE_API_NOISE)
132 std::unique_ptr<APINoiseFrameHelper>{
new APINoiseFrameHelper(std::move(sock), parent->get_noise_ctx())};
134#error "No frame helper defined"
143void APIConnection::start() {
146 APIError err = this->helper_->init();
147 if (err != APIError::OK) {
148 this->fatal_error_with_log_(LOG_STR(
"Helper init failed"), err);
152 char peername[socket::SOCKADDR_STR_LEN];
153 this->helper_->set_client_name(this->helper_->get_peername_to(peername), strlen(peername));
156APIConnection::~APIConnection() {
157 this->destroy_active_iterator_();
158#ifdef USE_BLUETOOTH_PROXY
163#ifdef USE_VOICE_ASSISTANT
168#ifdef USE_ZWAVE_PROXY
173#ifdef USE_SERIAL_PROXY
175 if (proxy->get_api_connection() ==
this) {
176 proxy->serial_proxy_request(
this, enums::SERIAL_PROXY_REQUEST_TYPE_UNSUBSCRIBE);
182void APIConnection::destroy_active_iterator_() {
183 switch (this->active_iterator_) {
184 case ActiveIterator::LIST_ENTITIES:
185 this->iterator_storage_.list_entities.~ListEntitiesIterator();
187 case ActiveIterator::INITIAL_STATE:
188 this->iterator_storage_.initial_state.~InitialStateIterator();
190 case ActiveIterator::NONE:
193 this->active_iterator_ = ActiveIterator::NONE;
197 this->destroy_active_iterator_();
198 this->active_iterator_ =
type;
199 if (
type == ActiveIterator::LIST_ENTITIES) {
201 this->iterator_storage_.list_entities.
begin();
204 this->iterator_storage_.initial_state.
begin();
208void APIConnection::loop() {
209 if (this->flags_.next_close) {
212 this->flags_.remove =
true;
216 APIError err = this->helper_->loop();
217 if (err != APIError::OK) {
218 this->fatal_error_with_log_(LOG_STR(
"Socket operation failed"), err);
229 if (this->helper_->is_socket_ready() || this->flags_.may_have_remaining_data) {
230 this->flags_.may_have_remaining_data =
false;
232 uint8_t message_count = 0;
233 for (; message_count < MAX_MESSAGES_PER_LOOP; message_count++) {
235 err = this->helper_->read_packet(&buffer);
236 if (err == APIError::WOULD_BLOCK) {
239 }
else if (err != APIError::OK) {
240 this->fatal_error_with_log_(LOG_STR(
"Reading failed"), err);
246 if (this->is_authenticated()) {
247 this->last_traffic_ = now;
251 if (this->flags_.remove)
257 if (message_count == MAX_MESSAGES_PER_LOOP) {
258 this->flags_.may_have_remaining_data =
true;
263 if (this->flags_.batch_scheduled && now - this->deferred_batch_.batch_start_time >= this->get_batch_delay_ms_()) {
264 this->process_batch_();
267 if (this->active_iterator_ != ActiveIterator::NONE) {
268 this->process_active_iterator_();
274 if (!this->is_authenticated() && now - this->last_traffic_ > HANDSHAKE_TIMEOUT_MS) {
275 this->on_fatal_error();
276 this->log_client_(ESPHOME_LOG_LEVEL_WARN, LOG_STR(
"handshake timeout; disconnecting"));
283 if (now - this->last_traffic_ > KEEPALIVE_TIMEOUT_MS) {
284 this->check_keepalive_(now);
287#ifdef USE_API_HOMEASSISTANT_STATES
288 if (state_subs_at_ >= 0) {
289 this->process_state_subscriptions_();
296 this->try_send_camera_image_();
300void APIConnection::check_keepalive_(
uint32_t now) {
302 if (this->flags_.sent_ping) {
304 if (now - this->last_traffic_ > KEEPALIVE_DISCONNECT_TIMEOUT) {
306 this->log_client_(ESPHOME_LOG_LEVEL_WARN, LOG_STR(
"is unresponsive; disconnecting"));
308 }
else if (!this->flags_.remove) {
310 ESP_LOGVV(TAG,
"Sending keepalive PING");
312 this->flags_.sent_ping = this->send_message(req);
313 if (!this->flags_.sent_ping) {
316 ESP_LOGW(TAG,
"Buffer full, ping queued");
317 this->schedule_message_front_(
nullptr, PingRequest::MESSAGE_TYPE, PingRequest::ESTIMATED_SIZE);
318 this->flags_.sent_ping =
true;
323void APIConnection::process_active_iterator_() {
325 if (this->active_iterator_ == ActiveIterator::LIST_ENTITIES) {
326 if (this->iterator_storage_.list_entities.completed()) {
327 this->destroy_active_iterator_();
328 if (this->flags_.state_subscription) {
329 this->begin_iterator_(ActiveIterator::INITIAL_STATE);
331 this->finalize_iterator_sync_();
334 this->process_iterator_batch_(this->iterator_storage_.list_entities);
337 if (this->iterator_storage_.initial_state.completed()) {
338 this->destroy_active_iterator_();
339 this->finalize_iterator_sync_();
341 this->process_iterator_batch_(this->iterator_storage_.initial_state);
346void APIConnection::finalize_iterator_sync_() {
350 if (!this->deferred_batch_.empty()) {
351 this->process_batch_();
354 this->flags_.should_try_send_immediately =
true;
356 this->deferred_batch_.release_buffer();
357 this->helper_->release_buffers();
361 size_t initial_size = this->deferred_batch_.size();
362 size_t max_batch = this->get_max_batch_size_();
363 while (!iterator.
completed() && (this->deferred_batch_.size() - initial_size) < max_batch) {
369 if (this->deferred_batch_.size() >= max_batch) {
370 this->process_batch_();
374bool APIConnection::send_disconnect_response_() {
378 this->log_client_(ESPHOME_LOG_LEVEL_DEBUG, LOG_STR(
"disconnected"));
379 this->flags_.next_close =
true;
381 return this->send_message(resp);
383void APIConnection::on_disconnect_response() {
386 this->flags_.remove =
true;
396 return encode_to_buffer(size_fn(&msg), encode_fn, &msg, conn, remaining_size);
410 char object_id_buf[OBJECT_ID_MAX_LEN];
420#ifdef USE_ENTITY_ICON
421 char icon_buf[MAX_ICON_LENGTH];
429 return encode_to_buffer_slow(size_fn(&msg), encode_fn, &msg, conn, remaining_size);
437 char dc_buf[MAX_DEVICE_CLASS_LENGTH];
439 return fill_and_encode_entity_info(entity, msg, size_fn, encode_fn, conn, remaining_size);
442#ifdef USE_BINARY_SENSOR
444 return this->send_message_smart_(binary_sensor, BinarySensorStateResponse::MESSAGE_TYPE,
445 BinarySensorStateResponse::ESTIMATED_SIZE);
451 resp.
state = binary_sensor->state;
453 return fill_and_encode_entity_state(binary_sensor, resp, conn, remaining_size);
460 return fill_and_encode_entity_info_with_device_class(binary_sensor, msg, msg.
device_class, conn, remaining_size);
466 return this->send_message_smart_(cover, CoverStateResponse::MESSAGE_TYPE, CoverStateResponse::ESTIMATED_SIZE);
471 auto traits = cover->get_traits();
473 if (traits.get_supports_tilt())
474 msg.
tilt = cover->tilt;
476 return fill_and_encode_entity_state(cover, msg, conn, remaining_size);
481 auto traits = cover->get_traits();
486 return fill_and_encode_entity_info_with_device_class(cover, msg, msg.
device_class, conn, remaining_size);
493 call.set_tilt(msg.
tilt);
495 call.set_command_stop();
502 return this->send_message_smart_(fan, FanStateResponse::MESSAGE_TYPE, FanStateResponse::ESTIMATED_SIZE);
505 auto *fan =
static_cast<fan::Fan *
>(entity);
507 auto traits = fan->get_traits();
508 msg.
state = fan->state;
509 if (traits.supports_oscillation())
511 if (traits.supports_speed()) {
514 if (traits.supports_direction())
516 if (traits.supports_preset_modes() && fan->has_preset_mode())
518 return fill_and_encode_entity_state(fan, msg, conn, remaining_size);
521 auto *fan =
static_cast<fan::Fan *
>(entity);
523 auto traits = fan->get_traits();
529 return fill_and_encode_entity_info(fan, msg, conn, remaining_size);
532 ENTITY_COMMAND_MAKE_CALL(
fan::Fan, fan, fan)
534 call.set_state(msg.
state);
551 return this->send_message_smart_(light, LightStateResponse::MESSAGE_TYPE, LightStateResponse::ESTIMATED_SIZE);
556 auto values = light->remote_values;
557 auto color_mode = values.get_color_mode();
558 resp.
state = values.is_on();
562 resp.
red = values.get_red();
563 resp.
green = values.get_green();
564 resp.
blue = values.get_blue();
565 resp.
white = values.get_white();
569 if (light->supports_effects()) {
570 resp.
effect = light->get_effect_name();
572 return fill_and_encode_entity_state(light, resp, conn, remaining_size);
577 auto traits = light->get_traits();
578 auto supported_modes = traits.get_supported_color_modes();
587 if (light->supports_effects()) {
588 auto &light_effects = light->get_effects();
589 effects_list.
init(light_effects.size() + 1);
591 for (
auto *effect : light_effects) {
593 effects_list.
push_back(effect->get_name().c_str());
597 return fill_and_encode_entity_info(light, msg, conn, remaining_size);
602 call.set_state(msg.
state);
610 call.set_red(msg.
red);
611 call.set_green(msg.
green);
612 call.set_blue(msg.
blue);
615 call.set_white(msg.
white);
634 return this->send_message_smart_(sensor, SensorStateResponse::MESSAGE_TYPE, SensorStateResponse::ESTIMATED_SIZE);
640 resp.
state = sensor->state;
642 return fill_and_encode_entity_state(sensor, resp, conn, remaining_size);
652 return fill_and_encode_entity_info_with_device_class(sensor, msg, msg.
device_class, conn, remaining_size);
658 return this->send_message_smart_(a_switch, SwitchStateResponse::MESSAGE_TYPE, SwitchStateResponse::ESTIMATED_SIZE);
664 resp.
state = a_switch->state;
665 return fill_and_encode_entity_state(a_switch, resp, conn, remaining_size);
672 return fill_and_encode_entity_info_with_device_class(a_switch, msg, msg.
device_class, conn, remaining_size);
680 a_switch->turn_off();
685#ifdef USE_TEXT_SENSOR
687 return this->send_message_smart_(text_sensor, TextSensorStateResponse::MESSAGE_TYPE,
688 TextSensorStateResponse::ESTIMATED_SIZE);
696 return fill_and_encode_entity_state(text_sensor, resp, conn, remaining_size);
701 return fill_and_encode_entity_info_with_device_class(text_sensor, msg, msg.
device_class, conn, remaining_size);
707 return this->send_message_smart_(climate, ClimateStateResponse::MESSAGE_TYPE, ClimateStateResponse::ESTIMATED_SIZE);
712 auto traits = climate->get_traits();
724 if (traits.get_supports_fan_modes() && climate->fan_mode.has_value())
726 if (!traits.get_supported_custom_fan_modes().empty() && climate->has_custom_fan_mode()) {
729 if (traits.get_supports_presets() && climate->preset.has_value()) {
732 if (!traits.get_supported_custom_presets().empty() && climate->has_custom_preset()) {
735 if (traits.get_supports_swing_modes())
741 return fill_and_encode_entity_state(climate, resp, conn, remaining_size);
746 auto traits = climate->get_traits();
768 return fill_and_encode_entity_info(climate, msg, conn, remaining_size);
798 return this->send_message_smart_(number, NumberStateResponse::MESSAGE_TYPE, NumberStateResponse::ESTIMATED_SIZE);
804 resp.
state = number->state;
806 return fill_and_encode_entity_state(number, resp, conn, remaining_size);
814 msg.
min_value = number->traits.get_min_value();
815 msg.
max_value = number->traits.get_max_value();
816 msg.
step = number->traits.get_step();
817 return fill_and_encode_entity_info_with_device_class(number, msg, msg.
device_class, conn, remaining_size);
821 call.set_value(msg.
state);
826#ifdef USE_DATETIME_DATE
828 return this->send_message_smart_(date, DateStateResponse::MESSAGE_TYPE, DateStateResponse::ESTIMATED_SIZE);
834 resp.
year = date->year;
835 resp.
month = date->month;
836 resp.
day = date->day;
837 return fill_and_encode_entity_state(date, resp, conn, remaining_size);
842 return fill_and_encode_entity_info(date, msg, conn, remaining_size);
851#ifdef USE_DATETIME_TIME
853 return this->send_message_smart_(time, TimeStateResponse::MESSAGE_TYPE, TimeStateResponse::ESTIMATED_SIZE);
859 resp.
hour = time->hour;
860 resp.
minute = time->minute;
861 resp.
second = time->second;
862 return fill_and_encode_entity_state(time, resp, conn, remaining_size);
867 return fill_and_encode_entity_info(time, msg, conn, remaining_size);
876#ifdef USE_DATETIME_DATETIME
878 return this->send_message_smart_(datetime, DateTimeStateResponse::MESSAGE_TYPE,
879 DateTimeStateResponse::ESTIMATED_SIZE);
885 if (datetime->has_state()) {
889 return fill_and_encode_entity_state(datetime, resp, conn, remaining_size);
894 return fill_and_encode_entity_info(datetime, msg, conn, remaining_size);
905 return this->send_message_smart_(text, TextStateResponse::MESSAGE_TYPE, TextStateResponse::ESTIMATED_SIZE);
909 auto *text =
static_cast<text::Text *
>(entity);
913 return fill_and_encode_entity_state(text, resp, conn, remaining_size);
917 auto *text =
static_cast<text::Text *
>(entity);
920 msg.
min_length = text->traits.get_min_length();
921 msg.
max_length = text->traits.get_max_length();
922 msg.
pattern = text->traits.get_pattern_ref();
923 return fill_and_encode_entity_info(text, msg, conn, remaining_size);
926 ENTITY_COMMAND_MAKE_CALL(
text::Text, text, text)
934 return this->send_message_smart_(select, SelectStateResponse::MESSAGE_TYPE, SelectStateResponse::ESTIMATED_SIZE);
940 resp.
state = select->current_option();
942 return fill_and_encode_entity_state(select, resp, conn, remaining_size);
948 msg.
options = &select->traits.get_options();
949 return fill_and_encode_entity_info(select, msg, conn, remaining_size);
962 return fill_and_encode_entity_info_with_device_class(button, msg, msg.
device_class, conn, remaining_size);
972 return this->send_message_smart_(a_lock, LockStateResponse::MESSAGE_TYPE, LockStateResponse::ESTIMATED_SIZE);
976 auto *a_lock =
static_cast<lock::Lock *
>(entity);
979 return fill_and_encode_entity_state(a_lock, resp, conn, remaining_size);
983 auto *a_lock =
static_cast<lock::Lock *
>(entity);
988 return fill_and_encode_entity_info(a_lock, msg, conn, remaining_size);
994 case enums::LOCK_UNLOCK:
997 case enums::LOCK_LOCK:
1000 case enums::LOCK_OPEN:
1009 return this->send_message_smart_(valve, ValveStateResponse::MESSAGE_TYPE, ValveStateResponse::ESTIMATED_SIZE);
1016 return fill_and_encode_entity_state(valve, resp, conn, remaining_size);
1021 auto traits = valve->get_traits();
1025 return fill_and_encode_entity_info_with_device_class(valve, msg, msg.
device_class, conn, remaining_size);
1032 call.set_command_stop();
1037#ifdef USE_MEDIA_PLAYER
1039 return this->send_message_smart_(media_player, MediaPlayerStateResponse::MESSAGE_TYPE,
1040 MediaPlayerStateResponse::ESTIMATED_SIZE);
1047 : media_player->state;
1049 resp.
volume = media_player->volume;
1050 resp.
muted = media_player->is_muted();
1051 return fill_and_encode_entity_state(media_player, resp, conn, remaining_size);
1056 auto traits = media_player->get_traits();
1059 for (
auto &supported_format : traits.get_supported_formats()) {
1062 media_format.format =
StringRef(supported_format.format);
1063 media_format.sample_rate = supported_format.sample_rate;
1064 media_format.num_channels = supported_format.num_channels;
1066 media_format.sample_bytes = supported_format.sample_bytes;
1068 return fill_and_encode_entity_info(media_player, msg, conn, remaining_size);
1076 call.set_volume(msg.
volume);
1089void APIConnection::try_send_camera_image_() {
1090 if (!this->image_reader_)
1094 while (this->image_reader_->available()) {
1095 if (!this->helper_->can_write_without_blocking())
1098 uint32_t to_send = std::min((
size_t) MAX_BATCH_PACKET_SIZE, this->image_reader_->available());
1099 bool done = this->image_reader_->available() == to_send;
1103 msg.
set_data(this->image_reader_->peek_data_buffer(), to_send);
1109 if (!this->send_message(msg)) {
1112 this->image_reader_->consume_data(to_send);
1114 this->image_reader_->return_image();
1119void APIConnection::set_camera_state(std::shared_ptr<camera::CameraImage> image) {
1120 if (!this->flags_.state_subscription)
1122 if (!this->image_reader_)
1124 if (this->image_reader_->available())
1127 this->image_reader_->set_image(std::move(image));
1129 this->try_send_camera_image_();
1135 return fill_and_encode_entity_info(camera, msg, conn, remaining_size);
1146 App.
scheduler.set_timeout(this->parent_,
"api_camera_stop_stream", CAMERA_STOP_STREAM,
1152#ifdef USE_HOMEASSISTANT_TIME
1156#ifdef USE_TIME_TIMEZONE
1162 if (pt.std_offset_seconds != 0 || pt.dst_start.type != enums::DST_RULE_TYPE_NONE) {
1167 tz.
dst_start.
day =
static_cast<uint16_t
>(pt.dst_start.day);
1170 tz.
dst_start.
week =
static_cast<uint8_t
>(pt.dst_start.week);
1173 tz.
dst_end.
day =
static_cast<uint16_t
>(pt.dst_end.day);
1175 tz.
dst_end.
month =
static_cast<uint8_t
>(pt.dst_end.month);
1176 tz.
dst_end.
week =
static_cast<uint8_t
>(pt.dst_end.week);
1188#ifdef USE_BLUETOOTH_PROXY
1189void APIConnection::on_subscribe_bluetooth_le_advertisements_request(
1193void APIConnection::on_unsubscribe_bluetooth_le_advertisements_request() {
1219bool APIConnection::send_subscribe_bluetooth_connections_free_response_() {
1223void APIConnection::on_subscribe_bluetooth_connections_free_request() {
1224 if (!this->send_subscribe_bluetooth_connections_free_response_()) {
1225 this->on_fatal_error();
1231 msg.
mode == enums::BluetoothScannerMode::BLUETOOTH_SCANNER_MODE_ACTIVE);
1238#ifdef USE_VOICE_ASSISTANT
1239bool APIConnection::check_voice_assistant_api_connection_()
const {
1250 if (!this->check_voice_assistant_api_connection_()) {
1258 if (msg.
port == 0) {
1264 this->helper_->getpeername((
struct sockaddr *) &storage, &
len);
1269 if (this->check_voice_assistant_api_connection_()) {
1274 if (this->check_voice_assistant_api_connection_()) {
1279 if (this->check_voice_assistant_api_connection_()) {
1285 if (this->check_voice_assistant_api_connection_()) {
1292 if (!this->check_voice_assistant_api_connection_()) {
1293 return this->send_message(resp);
1297 for (
auto &wake_word : config.available_wake_words) {
1300 resp_wake_word.id =
StringRef(wake_word.id);
1301 resp_wake_word.wake_word =
StringRef(wake_word.wake_word);
1302 for (
const auto &lang : wake_word.trained_languages) {
1303 resp_wake_word.trained_languages.push_back(lang);
1309 if (wake_word.model_type !=
"micro") {
1316 resp_wake_word.id =
StringRef(wake_word.id);
1317 resp_wake_word.wake_word =
StringRef(wake_word.wake_word);
1318 for (
const auto &lang : wake_word.trained_languages) {
1319 resp_wake_word.trained_languages.push_back(lang);
1325 return this->send_message(resp);
1328 if (!this->send_voice_assistant_get_configuration_response_(msg)) {
1329 this->on_fatal_error();
1334 if (this->check_voice_assistant_api_connection_()) {
1340#ifdef USE_ZWAVE_PROXY
1350#ifdef USE_ALARM_CONTROL_PANEL
1352 return this->send_message_smart_(a_alarm_control_panel, AlarmControlPanelStateResponse::MESSAGE_TYPE,
1353 AlarmControlPanelStateResponse::ESTIMATED_SIZE);
1360 return fill_and_encode_entity_state(a_alarm_control_panel, resp, conn, remaining_size);
1367 msg.
requires_code = a_alarm_control_panel->get_requires_code();
1369 return fill_and_encode_entity_info(a_alarm_control_panel, msg, conn, remaining_size);
1374 case enums::ALARM_CONTROL_PANEL_DISARM:
1377 case enums::ALARM_CONTROL_PANEL_ARM_AWAY:
1380 case enums::ALARM_CONTROL_PANEL_ARM_HOME:
1383 case enums::ALARM_CONTROL_PANEL_ARM_NIGHT:
1386 case enums::ALARM_CONTROL_PANEL_ARM_VACATION:
1387 call.arm_vacation();
1389 case enums::ALARM_CONTROL_PANEL_ARM_CUSTOM_BYPASS:
1390 call.arm_custom_bypass();
1392 case enums::ALARM_CONTROL_PANEL_TRIGGER:
1401#ifdef USE_WATER_HEATER
1403 return this->send_message_smart_(water_heater, WaterHeaterStateResponse::MESSAGE_TYPE,
1404 WaterHeaterStateResponse::ESTIMATED_SIZE);
1414 resp.
state = wh->get_state();
1416 return fill_and_encode_entity_state(wh, resp, conn, remaining_size);
1421 auto traits = wh->get_traits();
1427 return fill_and_encode_entity_info(wh, msg, conn, remaining_size);
1432 if (msg.
has_fields & enums::WATER_HEATER_COMMAND_HAS_MODE)
1434 if (msg.
has_fields & enums::WATER_HEATER_COMMAND_HAS_TARGET_TEMPERATURE)
1436 if (msg.
has_fields & enums::WATER_HEATER_COMMAND_HAS_TARGET_TEMPERATURE_LOW)
1438 if (msg.
has_fields & enums::WATER_HEATER_COMMAND_HAS_TARGET_TEMPERATURE_HIGH)
1440 if ((msg.
has_fields & enums::WATER_HEATER_COMMAND_HAS_AWAY_STATE) ||
1441 (msg.
has_fields & enums::WATER_HEATER_COMMAND_HAS_STATE)) {
1444 if ((msg.
has_fields & enums::WATER_HEATER_COMMAND_HAS_ON_STATE) ||
1445 (msg.
has_fields & enums::WATER_HEATER_COMMAND_HAS_STATE)) {
1456 this->send_message_smart_(event, EventResponse::MESSAGE_TYPE, EventResponse::ESTIMATED_SIZE,
1463 return fill_and_encode_entity_state(event, resp, conn, remaining_size);
1470 return fill_and_encode_entity_info_with_device_class(event, msg, msg.
device_class, conn, remaining_size);
1490#ifdef USE_SERIAL_PROXY
1493 if (msg.
instance >= proxies.size()) {
1494 ESP_LOGW(TAG,
"Serial proxy instance %" PRIu32
" out of range (max %" PRIu32
")", msg.
instance,
1495 static_cast<uint32_t>(proxies.size()));
1504 if (msg.
instance >= proxies.size()) {
1505 ESP_LOGW(TAG,
"Serial proxy instance %" PRIu32
" out of range", msg.
instance);
1513 if (msg.
instance >= proxies.size()) {
1514 ESP_LOGW(TAG,
"Serial proxy instance %" PRIu32
" out of range", msg.
instance);
1522 if (msg.
instance >= proxies.size()) {
1523 ESP_LOGW(TAG,
"Serial proxy instance %" PRIu32
" out of range", msg.
instance);
1528 resp.line_states = proxies[msg.
instance]->get_modem_pins();
1529 this->send_message(resp);
1534 if (msg.
instance >= proxies.size()) {
1535 ESP_LOGW(TAG,
"Serial proxy instance %" PRIu32
" out of range", msg.
instance);
1539 case enums::SERIAL_PROXY_REQUEST_TYPE_SUBSCRIBE:
1540 case enums::SERIAL_PROXY_REQUEST_TYPE_UNSUBSCRIBE:
1541 proxies[msg.
instance]->serial_proxy_request(
this, msg.
type);
1543 case enums::SERIAL_PROXY_REQUEST_TYPE_FLUSH: {
1546 resp.type = enums::SERIAL_PROXY_REQUEST_TYPE_FLUSH;
1547 switch (proxies[msg.
instance]->flush_port()) {
1549 resp.status = enums::SERIAL_PROXY_STATUS_OK;
1552 resp.status = enums::SERIAL_PROXY_STATUS_ASSUMED_SUCCESS;
1555 resp.status = enums::SERIAL_PROXY_STATUS_TIMEOUT;
1558 resp.status = enums::SERIAL_PROXY_STATUS_ERROR;
1561 this->send_message(resp);
1565 ESP_LOGW(TAG,
"Unknown serial proxy request type: %" PRIu32,
static_cast<uint32_t>(msg.
type));
1579 return fill_and_encode_entity_info(infrared, msg, conn, remaining_size);
1585 return this->send_message_smart_(update, UpdateStateResponse::MESSAGE_TYPE, UpdateStateResponse::ESTIMATED_SIZE);
1591 if (update->has_state()) {
1593 if (update->update_info.has_progress) {
1595 resp.
progress = update->update_info.progress;
1603 return fill_and_encode_entity_state(update, resp, conn, remaining_size);
1608 return fill_and_encode_entity_info_with_device_class(update, msg, msg.
device_class, conn, remaining_size);
1614 case enums::UPDATE_COMMAND_UPDATE:
1617 case enums::UPDATE_COMMAND_CHECK:
1620 case enums::UPDATE_COMMAND_NONE:
1621 ESP_LOGE(TAG,
"UPDATE_COMMAND_NONE not handled; confirm command is correct");
1624 ESP_LOGW(TAG,
"Unknown update command: %" PRIu32, msg.
command);
1630bool APIConnection::try_send_log_message(
int level,
const char *
tag,
const char *
line,
size_t message_len) {
1633 msg.
set_message(
reinterpret_cast<const uint8_t *
>(
line), message_len);
1634 return this->send_message(msg);
1637void APIConnection::complete_authentication_() {
1639 if (this->flags_.connection_state ==
static_cast<uint8_t
>(ConnectionState::AUTHENTICATED)) {
1643 this->flags_.connection_state =
static_cast<uint8_t
>(ConnectionState::AUTHENTICATED);
1646 this->log_client_(ESPHOME_LOG_LEVEL_DEBUG, LOG_STR(
"connected"));
1647#ifdef USE_API_CLIENT_CONNECTED_TRIGGER
1649 char peername[socket::SOCKADDR_STR_LEN];
1650 this->parent_->get_client_connected_trigger()->trigger(std::string(this->helper_->get_client_name()),
1651 std::string(this->helper_->get_peername_to(peername)));
1654#ifdef USE_HOMEASSISTANT_TIME
1656 this->send_time_request();
1659#ifdef USE_ZWAVE_PROXY
1671 char peername[socket::SOCKADDR_STR_LEN];
1672 ESP_LOGV(TAG,
"Hello from client: '%s' | %s | API Version %" PRIu16
".%" PRIu16, this->helper_->get_client_name(),
1673 this->helper_->get_peername_to(peername), this->client_api_version_major_, this->client_api_version_minor_);
1676 if (!this->client_supports_api_version(1, 14)) {
1677 ESP_LOGW(TAG,
"'%s' using outdated API %" PRIu16
".%" PRIu16
", update to 1.14+", this->helper_->get_client_name(),
1678 this->client_api_version_major_, this->client_api_version_minor_);
1689 this->complete_authentication_();
1691 return this->send_message(resp);
1694bool APIConnection::send_ping_response_() {
1696 return this->send_message(resp);
1699bool APIConnection::send_device_info_response_() {
1707 char mac_address[18];
1721#if defined(USE_ESP8266) || defined(USE_ESP32)
1722#define ESPHOME_MANUFACTURER "Espressif"
1723#elif defined(USE_RP2040)
1724#define ESPHOME_MANUFACTURER "Raspberry Pi"
1725#elif defined(USE_BK72XX)
1726#define ESPHOME_MANUFACTURER "Beken"
1727#elif defined(USE_LN882X)
1728#define ESPHOME_MANUFACTURER "Lightning"
1729#elif defined(USE_NRF52)
1730#define ESPHOME_MANUFACTURER "Nordic Semiconductor"
1731#elif defined(USE_RTL87XX)
1732#define ESPHOME_MANUFACTURER "Realtek"
1733#elif defined(USE_HOST)
1734#define ESPHOME_MANUFACTURER "Host"
1739 static const char MANUFACTURER_PROGMEM[]
PROGMEM = ESPHOME_MANUFACTURER;
1740 char manufacturer_buf[
sizeof(MANUFACTURER_PROGMEM)];
1741 memcpy_P(manufacturer_buf, MANUFACTURER_PROGMEM,
sizeof(MANUFACTURER_PROGMEM));
1747 static_assert(
sizeof(ESPHOME_MANUFACTURER) - 1 <= 20,
"Update max_data_length for manufacturer in api.proto");
1748#undef ESPHOME_MANUFACTURER
1751 static const char MODEL_PROGMEM[]
PROGMEM = ESPHOME_BOARD;
1752 char model_buf[
sizeof(MODEL_PROGMEM)];
1753 memcpy_P(model_buf, MODEL_PROGMEM,
sizeof(MODEL_PROGMEM));
1759#ifdef USE_DEEP_SLEEP
1762#ifdef ESPHOME_PROJECT_NAME
1764 static const char PROJECT_NAME_PROGMEM[]
PROGMEM = ESPHOME_PROJECT_NAME;
1765 static const char PROJECT_VERSION_PROGMEM[]
PROGMEM = ESPHOME_PROJECT_VERSION;
1766 char project_name_buf[
sizeof(PROJECT_NAME_PROGMEM)];
1767 char project_version_buf[
sizeof(PROJECT_VERSION_PROGMEM)];
1768 memcpy_P(project_name_buf, PROJECT_NAME_PROGMEM,
sizeof(PROJECT_NAME_PROGMEM));
1769 memcpy_P(project_version_buf, PROJECT_VERSION_PROGMEM,
sizeof(PROJECT_VERSION_PROGMEM));
1782#ifdef USE_BLUETOOTH_PROXY
1785 char bluetooth_mac[18];
1789#ifdef USE_VOICE_ASSISTANT
1792#ifdef USE_ZWAVE_PROXY
1796#ifdef USE_SERIAL_PROXY
1797 size_t serial_proxy_index = 0;
1799 if (serial_proxy_index >= SERIAL_PROXY_COUNT)
1802 info.name =
StringRef(proxy->get_name());
1803 info.port_type = proxy->get_port_type();
1810 size_t device_index = 0;
1812 if (device_index >= ESPHOME_DEVICE_COUNT)
1814 auto &device_info = resp.
devices[device_index++];
1815 device_info.device_id = device->get_device_id();
1816 device_info.name =
StringRef(device->get_name());
1817 device_info.area_id = device->get_area_id();
1821 size_t area_index = 0;
1823 if (area_index >= ESPHOME_AREA_COUNT)
1825 auto &area_info = resp.
areas[area_index++];
1826 area_info.area_id = area->get_area_id();
1827 area_info.name =
StringRef(area->get_name());
1831 return this->send_message(resp);
1834 if (!this->send_hello_response_(msg)) {
1835 this->on_fatal_error();
1838void APIConnection::on_disconnect_request() {
1839 if (!this->send_disconnect_response_()) {
1840 this->on_fatal_error();
1843void APIConnection::on_ping_request() {
1844 if (!this->send_ping_response_()) {
1845 this->on_fatal_error();
1848void APIConnection::on_device_info_request() {
1849 if (!this->send_device_info_response_()) {
1850 this->on_fatal_error();
1854#ifdef USE_API_HOMEASSISTANT_STATES
1870 for (
auto &it : this->parent_->get_state_subs()) {
1881 it.callback(msg.
state);
1885#ifdef USE_API_USER_DEFINED_ACTIONS
1893 if (!arg.string_.empty()) {
1894 const_cast<char *
>(arg.string_.c_str())[arg.string_.size()] =
'\0';
1898#ifdef USE_API_USER_DEFINED_ACTION_RESPONSES
1903 action_call_id = this->parent_->register_active_action_call(msg.
call_id,
this);
1906 for (
auto *service : this->parent_->get_user_services()) {
1907 if (service->execute_service(msg, action_call_id)) {
1912 for (
auto *service : this->parent_->get_user_services()) {
1913 if (service->execute_service(msg)) {
1919 ESP_LOGV(TAG,
"Could not find service");
1925#ifdef USE_API_USER_DEFINED_ACTION_RESPONSES
1926void APIConnection::send_execute_service_response(
uint32_t call_id,
bool success,
StringRef error_message) {
1931 this->send_message(resp);
1933#ifdef USE_API_USER_DEFINED_ACTION_RESPONSES_JSON
1934void APIConnection::send_execute_service_response(
uint32_t call_id,
bool success,
StringRef error_message,
1935 const uint8_t *response_data,
size_t response_data_len) {
1942 this->send_message(resp);
1948#ifdef USE_API_HOMEASSISTANT_ACTION_RESPONSES
1950#ifdef USE_API_HOMEASSISTANT_ACTION_RESPONSES_JSON
1968 if (this->parent_->clear_noise_psk(
true)) {
1971 ESP_LOGW(TAG,
"Failed to clear encryption key");
1974 ESP_LOGW(TAG,
"Invalid encryption key length");
1975 }
else if (!this->parent_->save_noise_psk(psk,
true)) {
1976 ESP_LOGW(TAG,
"Failed to save encryption key");
1981 return this->send_message(resp);
1984 if (!this->send_noise_encryption_set_key_response_(msg)) {
1985 this->on_fatal_error();
1989#ifdef USE_API_HOMEASSISTANT_STATES
1990void APIConnection::on_subscribe_home_assistant_states_request() { state_subs_at_ = 0; }
1992bool APIConnection::try_to_clear_buffer_slow_(
bool log_out_of_space) {
1994 APIError err = this->helper_->loop();
1995 if (err != APIError::OK) {
1996 this->fatal_error_with_log_(LOG_STR(
"Socket operation failed"), err);
1999 if (this->helper_->can_write_without_blocking())
2001 if (log_out_of_space) {
2002 ESP_LOGV(TAG,
"Cannot send message because of TCP buffer space");
2008#ifdef HAS_PROTO_MESSAGE_DUMP
2010 if (message_type != SubscribeLogsResponse::MESSAGE_TYPE
2012 && message_type != CameraImageResponse::MESSAGE_TYPE
2015 auto *proto_msg =
static_cast<const ProtoMessage *
>(msg);
2017 this->log_send_message_(proto_msg->message_name(), proto_msg->dump_to(dump_buf));
2020 auto &shared_buf = this->parent_->get_shared_buffer_ref();
2021 this->prepare_first_message_buffer(shared_buf,
payload_size);
2022 size_t write_start = shared_buf.size();
2025 encode_fn(msg, buffer PROTO_ENCODE_DEBUG_INIT(&shared_buf));
2033 return encode_to_buffer(calculated_size, encode_fn, msg, conn, remaining_size);
2036 const bool is_log_message = (message_type == SubscribeLogsResponse::MESSAGE_TYPE);
2038 if (!this->try_to_clear_buffer(!is_log_message)) {
2043 this->helper_->set_nodelay_for_message(is_log_message);
2045 APIError err = this->helper_->write_protobuf_packet(message_type, buffer);
2046 if (err == APIError::WOULD_BLOCK)
2048 if (err != APIError::OK) {
2049 this->fatal_error_with_log_(LOG_STR(
"Packet write failed"), err);
2055void APIConnection::on_no_setup_connection() {
2056 this->on_fatal_error();
2057 this->log_client_(ESPHOME_LOG_LEVEL_DEBUG, LOG_STR(
"no connection setup"));
2059void APIConnection::on_fatal_error() {
2062 this->flags_.remove =
true;
2065bool APIConnection::schedule_message_front_(
EntityBase *entity, uint8_t message_type, uint8_t estimated_size) {
2066 this->deferred_batch_.add_item_front(entity, message_type, estimated_size);
2067 return this->schedule_batch_();
2070bool APIConnection::send_message_smart_(
EntityBase *entity, uint8_t message_type, uint8_t estimated_size,
2071 uint8_t aux_data_index) {
2072 if (this->should_send_immediately_(message_type) && this->helper_->can_write_without_blocking()) {
2073 auto &shared_buf = this->parent_->get_shared_buffer_ref();
2074 this->prepare_first_message_buffer(shared_buf, estimated_size);
2076 if (this->dispatch_message_(item, MAX_BATCH_PACKET_SIZE,
true) &&
2078#ifdef HAS_PROTO_MESSAGE_DUMP
2079 this->log_batch_item_(item);
2084 return this->schedule_message_(entity, message_type, estimated_size, aux_data_index);
2087bool APIConnection::schedule_batch_() {
2088 if (!this->flags_.batch_scheduled) {
2089 this->flags_.batch_scheduled =
true;
2095void APIConnection::process_batch_() {
2096 if (this->deferred_batch_.empty()) {
2097 this->flags_.batch_scheduled =
false;
2106 this->helper_->set_nodelay_for_message(
false);
2109 if (!this->try_to_clear_buffer(
true)) {
2115 auto &shared_buf = this->parent_->get_shared_buffer_ref();
2116 size_t num_items = this->deferred_batch_.size();
2119 const uint8_t header_padding = this->helper_->frame_header_padding();
2120 const uint8_t footer_size = this->helper_->frame_footer_size();
2123 uint32_t total_estimated_size = num_items * (header_padding + footer_size);
2124 for (
size_t i = 0; i < num_items; i++) {
2125 total_estimated_size += this->deferred_batch_[i].estimated_size;
2128 if (total_estimated_size > MAX_BATCH_PACKET_SIZE) {
2129 total_estimated_size = MAX_BATCH_PACKET_SIZE;
2132 this->prepare_first_message_buffer(shared_buf, header_padding, total_estimated_size);
2135 if (num_items == 1) {
2136 const auto &item = this->deferred_batch_[0];
2138 uint16_t
payload_size = this->dispatch_message_(item, std::numeric_limits<uint16_t>::max(),
true);
2141#ifdef HAS_PROTO_MESSAGE_DUMP
2143 this->log_batch_item_(item);
2145 this->clear_batch_();
2148 ESP_LOGW(TAG,
"Message too large to send: type=%u", item.message_type);
2149 this->clear_batch_();
2155 this->process_batch_multi_(shared_buf, num_items, header_padding, footer_size);
2160void APIConnection::process_batch_multi_(
APIBuffer &shared_buf,
size_t num_items, uint8_t header_padding,
2161 uint8_t footer_size) {
2163 static_assert(std::is_trivially_destructible<MessageInfo>::value,
2164 "MessageInfo must remain trivially destructible with this placement-new approach");
2166 const size_t messages_to_process = std::min(num_items, MAX_MESSAGES_PER_BATCH);
2171 size_t items_processed = 0;
2172 uint16_t remaining_size = std::numeric_limits<uint16_t>::max();
2179 for (
size_t i = 0; i < messages_to_process; i++) {
2180 const auto &item = this->deferred_batch_[i];
2183 uint16_t
payload_size = this->dispatch_message_(item, remaining_size, i == 0);
2192 uint16_t proto_payload_size =
payload_size - this->batch_header_size_ - footer_size;
2197 new (&message_info[items_processed++])
2198 MessageInfo(item.message_type, current_offset, proto_payload_size, this->batch_header_size_);
2200 if (items_processed == 1) {
2201 remaining_size = MAX_BATCH_PACKET_SIZE;
2206 current_offset = shared_buf.
size() + footer_size;
2209 if (items_processed > 0) {
2211 if (footer_size > 0) {
2212 shared_buf.
resize(shared_buf.
size() + footer_size);
2217 std::span<const MessageInfo>(message_info, items_processed));
2218 if (err != APIError::OK && err != APIError::WOULD_BLOCK) {
2219 this->fatal_error_with_log_(LOG_STR(
"Batch write failed"), err);
2222#ifdef HAS_PROTO_MESSAGE_DUMP
2225 for (
size_t i = 0; i < items_processed; i++) {
2226 const auto &item = this->deferred_batch_[i];
2227 this->log_batch_item_(item);
2232 if (items_processed < this->deferred_batch_.size()) {
2233 this->deferred_batch_.remove_front(items_processed);
2234 this->schedule_batch_();
2240 this->clear_batch_();
2247 this->flags_.batch_first_message = batch_first;
2251 if (item.
message_type == EventResponse::MESSAGE_TYPE) {
2257 this, remaining_size);
2265#define CASE_STATE_INFO(entity_name, StateResp, InfoResp) \
2266 case StateResp::MESSAGE_TYPE: \
2267 func = &try_send_##entity_name##_state; \
2269 case InfoResp::MESSAGE_TYPE: \
2270 func = &try_send_##entity_name##_info; \
2272#define CASE_INFO_ONLY(entity_name, InfoResp) \
2273 case InfoResp::MESSAGE_TYPE: \
2274 func = &try_send_##entity_name##_info; \
2278#ifdef USE_BINARY_SENSOR
2299#ifdef USE_TEXT_SENSOR
2308#ifdef USE_DATETIME_DATE
2311#ifdef USE_DATETIME_TIME
2314#ifdef USE_DATETIME_DATETIME
2329#ifdef USE_MEDIA_PLAYER
2332#ifdef USE_ALARM_CONTROL_PANEL
2335#ifdef USE_WATER_HEATER
2351 case ListEntitiesDoneResponse::MESSAGE_TYPE:
2352 func = &try_send_list_info_done;
2354 case DisconnectRequest::MESSAGE_TYPE:
2355 func = &try_send_disconnect_request;
2357 case PingRequest::MESSAGE_TYPE:
2358 func = &try_send_ping_request;
2364#undef CASE_STATE_INFO
2365#undef CASE_INFO_ONLY
2367 return func(item.
entity,
this, remaining_size);
2372 return encode_message_to_buffer(resp, conn, remaining_size);
2377 return encode_message_to_buffer(req, conn, remaining_size);
2382 return encode_message_to_buffer(req, conn, remaining_size);
2385#ifdef USE_API_HOMEASSISTANT_STATES
2386void APIConnection::process_state_subscriptions_() {
2387 const auto &subs = this->parent_->get_state_subs();
2388 if (this->state_subs_at_ >=
static_cast<int>(subs.size())) {
2389 this->state_subs_at_ = -1;
2393 const auto &it = subs[this->state_subs_at_];
2400 resp.
once = it.once;
2401 if (this->send_message(resp)) {
2402 this->state_subs_at_++;
2407void APIConnection::log_client_(
int level,
const LogString *
message) {
2408 char peername[socket::SOCKADDR_STR_LEN];
2409 esp_log_printf_(level, TAG, __LINE__, ESPHOME_LOG_FORMAT(
"%s (%s): %s"), this->helper_->get_client_name(),
2410 this->helper_->get_peername_to(peername), LOG_STR_ARG(
message));
2414 char peername[socket::SOCKADDR_STR_LEN];
2415 ESP_LOGW(TAG,
"%s (%s): %s %s errno=%d", this->helper_->get_client_name(), this->helper_->get_peername_to(peername),
const StringRef & get_name() const
Get the name of this Application set by pre_setup().
static constexpr size_t BUILD_TIME_STR_SIZE
Size of buffer required for build time string (including null terminator)
const StringRef & get_friendly_name() const
Get the friendly name of this Application set by pre_setup().
void get_build_time_string(std::span< char, BUILD_TIME_STR_SIZE > buffer)
Copy the build time string into the provided buffer Buffer must be BUILD_TIME_STR_SIZE bytes (compile...
const char * get_area() const
Get the area of this Application set by pre_setup().
const auto & get_devices()
auto & get_serial_proxies() const
uint32_t IRAM_ATTR HOT get_loop_component_start_time() const
Get the cached time in milliseconds from when the current component started its loop execution.
void begin(bool include_internal=false)
const char * get_device_class_to(std::span< char, MAX_DEVICE_CLASS_LENGTH > buffer) const
bool has_own_name() const
const StringRef & get_name() const
ESPDEPRECATED("Use get_unit_of_measurement_ref() instead for better performance (avoids string copy). Will be " "removed in ESPHome 2026.9.0", "2026.3.0") std const char * get_icon_to(std::span< char, MAX_ICON_LENGTH > buffer) const
Get the unit of measurement as std::string (deprecated, prefer get_unit_of_measurement_ref())
ESPDEPRECATED("Use get_icon_to() instead. Will be removed in ESPHome 2026.9.0", "2026.3.0") std uint32_t get_device_id() const
bool is_disabled_by_default() const
ESPDEPRECATED("object_id mangles names and all object_id methods are planned for removal " "(see https://github.com/esphome/backlog/issues/76). " "Now is the time to stop using object_id. If still needed, use get_object_id_to() " "which will remain available longer. get_object_id() will be removed in 2026.7.0", "2025.12.0") std uint32_t get_object_id_hash() const
EntityCategory get_entity_category() const
StringRef get_object_id_to(std::span< char, OBJECT_ID_MAX_LEN > buf) const
Get object_id with zero heap allocation For static case: returns StringRef to internal storage (buffe...
Fixed-capacity vector - allocates once at runtime, never reallocates This avoids std::vector template...
void push_back(const T &value)
Add element without bounds checking Caller must ensure sufficient capacity was allocated via init() S...
StringRef is a reference to a string owned by something else.
constexpr const char * c_str() const
constexpr bool empty() const
constexpr size_type size() const
static constexpr StringRef from_lit(const CharT(&s)[N])
static StringRef from_maybe_nullptr(const char *s)
Byte buffer that skips zero-initialization on resize().
void resize(size_t n) ESPHOME_ALWAYS_INLINE
void on_button_command_request(const ButtonCommandRequest &msg)
uint8_t *(*)(const void *, ProtoWriteBuffer &PROTO_ENCODE_DEBUG_PARAM) MessageEncodeFn
APIConnection(std::unique_ptr< socket::Socket > socket, APIServer *parent)
uint16_t(*)(EntityBase *, APIConnection *, uint32_t remaining_size) MessageCreatorPtr
bool client_supports_api_version(uint16_t major, uint16_t minor) const
uint32_t(*)(const void *) CalculateSizeFn
APINoiseContext & get_noise_ctx()
enums::AlarmControlPanelStateCommand command
enums::AlarmControlPanelState state
enums::BluetoothScannerMode mode
void set_data(const uint8_t *data, size_t len)
bool has_target_temperature_high
float target_temperature_low
bool has_target_temperature_low
StringRef custom_fan_mode
float target_temperature_high
enums::ClimateSwingMode swing_mode
enums::ClimateFanMode fan_mode
bool has_target_temperature
enums::ClimatePreset preset
enums::ClimateFanMode fan_mode
float target_temperature_low
enums::ClimateSwingMode swing_mode
enums::ClimateAction action
enums::ClimatePreset preset
StringRef custom_fan_mode
float current_temperature
float target_temperature_high
enums::CoverOperation current_operation
StringRef project_version
uint32_t zwave_proxy_feature_flags
StringRef esphome_version
std::array< AreaInfo, ESPHOME_AREA_COUNT > areas
StringRef bluetooth_mac_address
uint32_t bluetooth_proxy_feature_flags
StringRef compilation_time
uint32_t voice_assistant_feature_flags
bool api_encryption_supported
std::array< SerialProxyInfo, SERIAL_PROXY_COUNT > serial_proxies
std::array< DeviceInfo, ESPHOME_DEVICE_COUNT > devices
Fixed-size buffer for message dumps - avoids heap allocation.
uint16_t response_data_len
const uint8_t * response_data
enums::FanDirection direction
enums::FanDirection direction
ParsedTimezone parsed_timezone
uint32_t api_version_major
uint32_t api_version_minor
uint32_t api_version_minor
uint32_t api_version_major
const uint8_t * response_data
uint16_t response_data_len
enums::EntityCategory entity_category
const uint8_t * timings_data_
uint32_t carrier_frequency
bool has_color_temperature
enums::ColorMode color_mode
bool has_transition_length
uint32_t transition_length
bool has_color_brightness
enums::ColorMode color_mode
bool requires_code_to_arm
uint32_t supported_features
bool is_status_binary_sensor
const std::vector< const char * > * supported_custom_presets
const climate::ClimateSwingModeMask * supported_swing_modes
float visual_max_humidity
const std::vector< const char * > * supported_custom_fan_modes
bool supports_current_temperature
bool supports_current_humidity
bool supports_target_humidity
float visual_min_humidity
float visual_max_temperature
float visual_target_temperature_step
bool supports_two_point_target_temperature
const climate::ClimatePresetMask * supported_presets
const climate::ClimateFanModeMask * supported_fan_modes
const climate::ClimateModeMask * supported_modes
float visual_min_temperature
float visual_current_temperature_step
const FixedVector< const char * > * event_types
const std::vector< const char * > * supported_preset_modes
int32_t supported_speed_count
bool supports_oscillation
uint32_t receiver_frequency
const FixedVector< const char * > * effects
const light::ColorModeMask * supported_color_modes
StringRef unit_of_measurement
const FixedVector< const char * > * options
int32_t accuracy_decimals
StringRef unit_of_measurement
enums::SensorStateClass state_class
float target_temperature_step
const water_heater::WaterHeaterModeMask * supported_modes
uint32_t supported_features
enums::LockCommand command
enums::SerialProxyRequestType type
void set_message(const uint8_t *data, size_t len)
enums::UpdateCommand command
StringRef current_version
StringRef release_summary
enums::ValveOperation current_operation
std::vector< VoiceAssistantExternalWakeWord > external_wake_words
std::vector< VoiceAssistantWakeWord > available_wake_words
uint32_t max_active_wake_words
const std::vector< std::string > * active_wake_words
std::vector< std::string > active_wake_words
float target_temperature_low
enums::WaterHeaterMode mode
float target_temperature_high
float current_temperature
float target_temperature_low
float target_temperature_high
enums::WaterHeaterMode mode
enums::ZWaveProxyRequestType type
Base class for all binary_sensor-type classes.
void bluetooth_gatt_read(const api::BluetoothGATTReadRequest &msg)
void bluetooth_gatt_send_services(const api::BluetoothGATTGetServicesRequest &msg)
void bluetooth_device_request(const api::BluetoothDeviceRequest &msg)
void bluetooth_gatt_write_descriptor(const api::BluetoothGATTWriteDescriptorRequest &msg)
void bluetooth_scanner_set_mode(bool active)
void subscribe_api_connection(api::APIConnection *api_connection, uint32_t flags)
uint32_t get_feature_flags() const
void send_connections_free()
void unsubscribe_api_connection(api::APIConnection *api_connection)
void get_bluetooth_mac_address_pretty(std::span< char, 18 > output)
void bluetooth_set_connection_params(const api::BluetoothSetConnectionParamsRequest &msg)
void bluetooth_gatt_read_descriptor(const api::BluetoothGATTReadDescriptorRequest &msg)
void bluetooth_gatt_write(const api::BluetoothGATTWriteRequest &msg)
void bluetooth_gatt_notify(const api::BluetoothGATTNotifyRequest &msg)
Abstract camera base class.
virtual CameraImageReader * create_image_reader()=0
Returns a new camera image reader that keeps track of the JPEG data in the camera image.
virtual void start_stream(CameraRequester requester)=0
virtual void stop_stream(CameraRequester requester)=0
virtual void request_image(CameraRequester requester)=0
static Camera * instance()
The singleton instance of the camera implementation.
ClimateDevice - This is the base class for all climate integrations.
Base class for all cover devices.
uint8_t get_last_event_type_index() const
Return index of last triggered event type, or max uint8_t if no event triggered yet.
void set_epoch_time(uint32_t epoch)
Infrared - Base class for infrared remote control implementations.
This class represents the communication layer between the front-end MQTT layer and the hardware outpu...
Base class for all locks.
Base-class for all numbers.
Base-class for all selects.
Base-class for all sensors.
Base class for all switches.
Base-class for all text inputs.
void set_timezone(const char *tz)
Set the time zone from a POSIX TZ string.
Base class for all valve devices.
const Configuration & get_configuration()
void on_timer_event(const api::VoiceAssistantTimerEventResponse &msg)
void on_audio(const api::VoiceAssistantAudio &msg)
void client_subscription(api::APIConnection *client, bool subscribe)
void on_event(const api::VoiceAssistantEventResponse &msg)
void on_announce(const api::VoiceAssistantAnnounceRequest &msg)
api::APIConnection * get_api_connection() const
uint32_t get_feature_flags() const
void on_set_configuration(const std::vector< std::string > &active_wake_words)
void zwave_proxy_request(api::APIConnection *api_connection, api::enums::ZWaveProxyRequestType type)
void send_frame(const uint8_t *data, size_t length)
uint32_t get_feature_flags() const
void api_connection_authenticated(api::APIConnection *conn)
const LogString * api_error_to_logstr(APIError err)
std::array< uint8_t, 32 > psk_t
BluetoothProxy * global_bluetooth_proxy
@ CLIMATE_SUPPORTS_CURRENT_HUMIDITY
@ CLIMATE_SUPPORTS_TARGET_HUMIDITY
@ CLIMATE_SUPPORTS_TWO_POINT_TARGET_TEMPERATURE
@ CLIMATE_SUPPORTS_CURRENT_TEMPERATURE
@ CLIMATE_SUPPORTS_ACTION
@ CLIMATE_REQUIRES_TWO_POINT_TARGET_TEMPERATURE
ClimatePreset
Enum for all preset modes NOTE: If adding values, update ClimatePresetMask in climate_traits....
ClimateSwingMode
Enum for all modes a climate swing can be in NOTE: If adding values, update ClimateSwingModeMask in c...
ClimateMode
Enum for all modes a climate device can be in.
ClimateFanMode
NOTE: If adding values, update ClimateFanModeMask in climate_traits.h to use the new last value.
bool global_has_deep_sleep
FanDirection
Simple enum to represent the direction of a fan.
HomeassistantTime * global_homeassistant_time
ColorMode
Color modes are a combination of color capabilities that can be used at the same time.
@ COLOR_TEMPERATURE
Color temperature can be controlled.
@ COLD_WARM_WHITE
Brightness of cold and warm white output can be controlled.
void set_global_tz(const ParsedTimezone &tz)
Set the global timezone used by epoch_to_local_tm() when called without a timezone.
DSTRuleType
Type of DST transition rule.
@ UART_FLUSH_RESULT_ASSUMED_SUCCESS
Platform cannot report result; success is assumed.
@ UART_FLUSH_RESULT_SUCCESS
Confirmed: all bytes left the TX FIFO.
@ UART_FLUSH_RESULT_FAILED
Confirmed: driver or hardware error.
@ UART_FLUSH_RESULT_TIMEOUT
Confirmed: timed out before TX completed.
@ UPDATE_STATE_INSTALLING
VoiceAssistant * global_voice_assistant
@ WATER_HEATER_STATE_ON
Water heater is on (not in standby)
@ WATER_HEATER_STATE_AWAY
Away/vacation mode is currently active.
ZWaveProxy * global_zwave_proxy
const char int const __FlashStringHelper va_list args
void HOT esp_log_printf_(int level, const char *tag, int line, const char *format,...)
void get_mac_address_raw(uint8_t *mac)
Get the device MAC address as raw bytes, written into the provided byte array (6 bytes).
void HOT delay(uint32_t ms)
Application App
Global storage of Application pointer - only one Application can exist.
size_t base64_decode(const std::string &encoded_string, uint8_t *buf, size_t buf_len)
char * format_mac_addr_upper(const uint8_t *mac, char *output)
Format MAC address as XX:XX:XX:XX:XX:XX (uppercase, colon separators)
A more user-friendly version of struct tm from time.h.
uint16_t day
Day of year (for JULIAN_NO_LEAP and DAY_OF_YEAR)
DSTRuleType type
Type of rule.
uint8_t week
Week 1-5, 5 = last (for MONTH_WEEK_DAY)
int32_t time_seconds
Seconds after midnight (default 7200 = 2:00 AM)
uint8_t day_of_week
Day 0-6, 0 = Sunday (for MONTH_WEEK_DAY)
uint8_t month
Month 1-12 (for MONTH_WEEK_DAY)
Parsed POSIX timezone information (packed for 32-bit: 32 bytes)
DSTRule dst_end
When DST ends.
DSTRule dst_start
When DST starts.
int32_t dst_offset_seconds
DST offset from UTC in seconds.
int32_t std_offset_seconds
Standard time offset from UTC in seconds (positive = west)
const uint8_t ESPHOME_WEBSERVER_INDEX_HTML[] PROGMEM