esphome 2025.4.1__py3-none-any.whl → 2025.5.0__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (457) hide show
  1. esphome/__main__.py +16 -14
  2. esphome/components/ac_dimmer/ac_dimmer.cpp +3 -2
  3. esphome/components/adc/__init__.py +51 -34
  4. esphome/components/airthings_wave_base/__init__.py +1 -1
  5. esphome/components/alarm_control_panel/__init__.py +37 -2
  6. esphome/components/am43/cover/__init__.py +4 -5
  7. esphome/components/analog_threshold/analog_threshold_binary_sensor.cpp +6 -4
  8. esphome/components/analog_threshold/analog_threshold_binary_sensor.h +4 -5
  9. esphome/components/analog_threshold/binary_sensor.py +10 -8
  10. esphome/components/anova/climate.py +4 -5
  11. esphome/components/api/__init__.py +25 -8
  12. esphome/components/api/api_connection.cpp +416 -662
  13. esphome/components/api/api_connection.h +256 -57
  14. esphome/components/api/api_frame_helper.cpp +232 -177
  15. esphome/components/api/api_frame_helper.h +61 -8
  16. esphome/components/api/api_noise_context.h +13 -4
  17. esphome/components/api/api_pb2.cpp +1422 -1
  18. esphome/components/api/api_pb2.h +255 -1
  19. esphome/components/api/api_pb2_service.cpp +162 -49
  20. esphome/components/api/api_pb2_service.h +90 -51
  21. esphome/components/api/api_pb2_size.h +361 -0
  22. esphome/components/api/api_server.cpp +110 -34
  23. esphome/components/api/api_server.h +8 -0
  24. esphome/components/api/proto.h +86 -17
  25. esphome/components/as3935_i2c/as3935_i2c.h +0 -3
  26. esphome/components/as7341/as7341.h +1 -1
  27. esphome/components/at581x/at581x.h +4 -4
  28. esphome/components/atm90e32/__init__.py +1 -0
  29. esphome/components/atm90e32/atm90e32.cpp +576 -199
  30. esphome/components/atm90e32/atm90e32.h +128 -31
  31. esphome/components/atm90e32/atm90e32_reg.h +4 -2
  32. esphome/components/atm90e32/button/__init__.py +62 -10
  33. esphome/components/atm90e32/button/atm90e32_button.cpp +63 -4
  34. esphome/components/atm90e32/button/atm90e32_button.h +36 -4
  35. esphome/components/atm90e32/number/__init__.py +130 -0
  36. esphome/components/atm90e32/number/atm90e32_number.h +16 -0
  37. esphome/components/atm90e32/sensor.py +21 -4
  38. esphome/components/atm90e32/text_sensor/__init__.py +48 -0
  39. esphome/components/audio/__init__.py +96 -49
  40. esphome/components/audio/audio.h +48 -0
  41. esphome/components/audio/audio_decoder.cpp +1 -1
  42. esphome/components/audio/audio_resampler.cpp +2 -0
  43. esphome/components/audio/audio_resampler.h +1 -0
  44. esphome/components/ballu/climate.py +2 -9
  45. esphome/components/bang_bang/climate.py +5 -6
  46. esphome/components/bedjet/bedjet_hub.cpp +1 -0
  47. esphome/components/bedjet/climate/__init__.py +3 -8
  48. esphome/components/bedjet/fan/__init__.py +2 -11
  49. esphome/components/binary/fan/__init__.py +13 -16
  50. esphome/components/binary_sensor/__init__.py +13 -10
  51. esphome/components/bl0906/constants.h +16 -16
  52. esphome/components/ble_client/text_sensor/__init__.py +3 -5
  53. esphome/components/bluetooth_proxy/bluetooth_connection.cpp +4 -6
  54. esphome/components/bluetooth_proxy/bluetooth_proxy.cpp +136 -21
  55. esphome/components/bluetooth_proxy/bluetooth_proxy.h +7 -0
  56. esphome/components/button/__init__.py +11 -8
  57. esphome/components/canbus/canbus.cpp +3 -0
  58. esphome/components/canbus/canbus.h +16 -0
  59. esphome/components/ccs811/sensor.py +9 -6
  60. esphome/components/climate/__init__.py +35 -2
  61. esphome/components/climate/climate_mode.h +1 -1
  62. esphome/components/climate/climate_traits.h +63 -57
  63. esphome/components/climate_ir/__init__.py +57 -17
  64. esphome/components/climate_ir_lg/climate.py +2 -5
  65. esphome/components/climate_ir_lg/climate_ir_lg.cpp +7 -7
  66. esphome/components/climate_ir_lg/climate_ir_lg.h +1 -1
  67. esphome/components/color/__init__.py +2 -0
  68. esphome/components/const/__init__.py +5 -0
  69. esphome/components/coolix/climate.py +2 -9
  70. esphome/components/copy/cover/__init__.py +10 -9
  71. esphome/components/copy/fan/__init__.py +11 -9
  72. esphome/components/copy/lock/__init__.py +11 -9
  73. esphome/components/copy/text/__init__.py +9 -6
  74. esphome/components/cover/__init__.py +37 -2
  75. esphome/components/cse7766/cse7766.cpp +2 -1
  76. esphome/components/cst226/binary_sensor/__init__.py +28 -0
  77. esphome/components/cst226/binary_sensor/cs226_button.h +22 -0
  78. esphome/components/cst226/binary_sensor/cstt6_button.cpp +19 -0
  79. esphome/components/cst226/touchscreen/cst226_touchscreen.cpp +27 -5
  80. esphome/components/cst226/touchscreen/cst226_touchscreen.h +10 -10
  81. esphome/components/current_based/cover.py +37 -36
  82. esphome/components/current_based/current_based_cover.cpp +2 -1
  83. esphome/components/daikin/climate.py +2 -9
  84. esphome/components/daikin/daikin.cpp +15 -9
  85. esphome/components/daikin/daikin.h +5 -5
  86. esphome/components/daikin_arc/climate.py +2 -7
  87. esphome/components/daikin_brc/climate.py +3 -5
  88. esphome/components/dallas_temp/dallas_temp.cpp +17 -24
  89. esphome/components/dallas_temp/dallas_temp.h +0 -1
  90. esphome/components/daly_bms/daly_bms.cpp +2 -1
  91. esphome/components/debug/debug_component.cpp +6 -1
  92. esphome/components/debug/debug_component.h +8 -0
  93. esphome/components/debug/debug_esp32.cpp +109 -254
  94. esphome/components/debug/sensor.py +14 -0
  95. esphome/components/deep_sleep/deep_sleep_esp32.cpp +13 -1
  96. esphome/components/delonghi/climate.py +2 -9
  97. esphome/components/demo/__init__.py +18 -20
  98. esphome/components/dfrobot_sen0395/switch/__init__.py +21 -22
  99. esphome/components/display/rect.cpp +4 -9
  100. esphome/components/display/rect.h +1 -1
  101. esphome/components/dps310/sensor.py +6 -6
  102. esphome/components/ee895/sensor.py +9 -9
  103. esphome/components/emmeti/climate.py +2 -9
  104. esphome/components/endstop/cover.py +17 -16
  105. esphome/components/endstop/endstop_cover.cpp +2 -1
  106. esphome/components/ens160_base/__init__.py +12 -9
  107. esphome/components/esp32/__init__.py +60 -3
  108. esphome/components/esp32/core.cpp +11 -5
  109. esphome/components/esp32/gpio.cpp +86 -24
  110. esphome/components/esp32/gpio.py +15 -16
  111. esphome/components/esp32/gpio_esp32.py +1 -2
  112. esphome/components/esp32/gpio_esp32_c2.py +1 -1
  113. esphome/components/esp32/gpio_esp32_c3.py +1 -1
  114. esphome/components/esp32/gpio_esp32_c6.py +1 -1
  115. esphome/components/esp32/gpio_esp32_h2.py +1 -1
  116. esphome/components/esp32_ble/ble.cpp +1 -8
  117. esphome/components/esp32_ble/ble.h +5 -3
  118. esphome/components/esp32_ble/ble_advertising.cpp +2 -1
  119. esphome/components/esp32_ble/ble_advertising.h +1 -0
  120. esphome/components/esp32_ble_server/__init__.py +3 -0
  121. esphome/components/esp32_ble_tracker/__init__.py +7 -1
  122. esphome/components/esp32_ble_tracker/esp32_ble_tracker.cpp +192 -118
  123. esphome/components/esp32_ble_tracker/esp32_ble_tracker.h +29 -3
  124. esphome/components/esp32_camera/__init__.py +1 -1
  125. esphome/components/esp32_camera/esp32_camera.cpp +2 -10
  126. esphome/components/esp32_camera/esp32_camera.h +1 -1
  127. esphome/components/esp32_can/esp32_can.cpp +1 -1
  128. esphome/components/esp32_improv/esp32_improv_component.cpp +1 -1
  129. esphome/components/esp32_rmt_led_strip/led_strip.cpp +1 -1
  130. esphome/components/esp32_rmt_led_strip/led_strip.h +7 -5
  131. esphome/components/esp32_rmt_led_strip/light.py +9 -1
  132. esphome/components/esp32_touch/esp32_touch.cpp +1 -1
  133. esphome/components/esp8266/gpio.cpp +69 -8
  134. esphome/components/ethernet/ethernet_component.cpp +1 -1
  135. esphome/components/event/__init__.py +13 -10
  136. esphome/components/factory_reset/switch/__init__.py +7 -21
  137. esphome/components/fan/__init__.py +52 -5
  138. esphome/components/fastled_base/__init__.py +1 -4
  139. esphome/components/fastled_base/fastled_light.cpp +1 -1
  140. esphome/components/feedback/cover.py +38 -33
  141. esphome/components/feedback/feedback_cover.cpp +2 -1
  142. esphome/components/fujitsu_general/climate.py +2 -9
  143. esphome/components/gcja5/gcja5.cpp +2 -1
  144. esphome/components/gpio/one_wire/gpio_one_wire.cpp +45 -43
  145. esphome/components/gpio/one_wire/gpio_one_wire.h +2 -1
  146. esphome/components/gpio_expander/cached_gpio.h +22 -7
  147. esphome/components/gps/__init__.py +47 -17
  148. esphome/components/gps/gps.cpp +42 -23
  149. esphome/components/gps/gps.h +17 -13
  150. esphome/components/graph/__init__.py +1 -2
  151. esphome/components/gree/climate.py +4 -6
  152. esphome/components/gree/gree.cpp +16 -2
  153. esphome/components/gree/gree.h +2 -2
  154. esphome/components/growatt_solar/growatt_solar.cpp +2 -1
  155. esphome/components/haier/climate.py +37 -34
  156. esphome/components/hbridge/fan/__init__.py +19 -17
  157. esphome/components/he60r/cover.py +4 -5
  158. esphome/components/heatpumpir/climate.py +3 -6
  159. esphome/components/hitachi_ac344/climate.py +2 -9
  160. esphome/components/hitachi_ac424/climate.py +2 -9
  161. esphome/components/hlw8012/hlw8012.cpp +1 -1
  162. esphome/components/hm3301/hm3301.h +1 -1
  163. esphome/components/hte501/sensor.py +6 -6
  164. esphome/components/http_request/__init__.py +39 -6
  165. esphome/components/http_request/http_request.cpp +20 -0
  166. esphome/components/http_request/http_request.h +57 -15
  167. esphome/components/http_request/http_request_arduino.cpp +22 -6
  168. esphome/components/http_request/http_request_arduino.h +4 -3
  169. esphome/components/http_request/http_request_host.cpp +141 -0
  170. esphome/components/http_request/http_request_host.h +37 -0
  171. esphome/components/http_request/http_request_idf.cpp +35 -3
  172. esphome/components/http_request/http_request_idf.h +10 -3
  173. esphome/components/http_request/httplib.h +9691 -0
  174. esphome/components/http_request/update/__init__.py +11 -8
  175. esphome/components/hyt271/sensor.py +6 -6
  176. esphome/components/i2c/i2c.h +4 -0
  177. esphome/components/i2c/i2c_bus_esp_idf.cpp +1 -1
  178. esphome/components/i2s_audio/__init__.py +131 -22
  179. esphome/components/i2s_audio/i2s_audio.h +44 -4
  180. esphome/components/i2s_audio/media_player/__init__.py +19 -9
  181. esphome/components/i2s_audio/microphone/__init__.py +63 -5
  182. esphome/components/i2s_audio/microphone/i2s_audio_microphone.cpp +351 -61
  183. esphome/components/i2s_audio/microphone/i2s_audio_microphone.h +40 -6
  184. esphome/components/i2s_audio/speaker/__init__.py +31 -5
  185. esphome/components/i2s_audio/speaker/i2s_audio_speaker.cpp +155 -19
  186. esphome/components/i2s_audio/speaker/i2s_audio_speaker.h +17 -4
  187. esphome/components/ili9xxx/ili9xxx_init.h +1 -1
  188. esphome/components/image/__init__.py +37 -17
  189. esphome/components/image/image.cpp +25 -8
  190. esphome/components/internal_temperature/internal_temperature.cpp +6 -4
  191. esphome/components/key_collector/__init__.py +35 -0
  192. esphome/components/key_collector/key_collector.cpp +8 -0
  193. esphome/components/key_collector/key_collector.h +10 -0
  194. esphome/components/kuntze/kuntze.cpp +2 -1
  195. esphome/components/ld2410/ld2410.h +1 -1
  196. esphome/components/ld2450/ld2450.h +1 -1
  197. esphome/components/light/__init__.py +57 -0
  198. esphome/components/lock/__init__.py +51 -4
  199. esphome/components/lock/automation.h +2 -13
  200. esphome/components/logger/__init__.py +22 -0
  201. esphome/components/logger/logger.cpp +154 -103
  202. esphome/components/logger/logger.h +211 -36
  203. esphome/components/logger/task_log_buffer.cpp +138 -0
  204. esphome/components/logger/task_log_buffer.h +69 -0
  205. esphome/components/lvgl/__init__.py +13 -5
  206. esphome/components/lvgl/automation.py +50 -1
  207. esphome/components/lvgl/defines.py +0 -1
  208. esphome/components/lvgl/lv_validation.py +10 -1
  209. esphome/components/lvgl/lvgl_esphome.cpp +5 -1
  210. esphome/components/lvgl/schemas.py +14 -14
  211. esphome/components/lvgl/text/__init__.py +1 -2
  212. esphome/components/lvgl/widgets/arc.py +7 -6
  213. esphome/components/lvgl/widgets/buttonmatrix.py +3 -3
  214. esphome/components/lvgl/widgets/checkbox.py +2 -2
  215. esphome/components/lvgl/widgets/dropdown.py +2 -1
  216. esphome/components/lvgl/widgets/img.py +15 -12
  217. esphome/components/mapping/__init__.py +134 -0
  218. esphome/components/matrix_keypad/matrix_keypad.cpp +2 -1
  219. esphome/components/max7219digit/max7219digit.cpp +28 -27
  220. esphome/components/mdns/__init__.py +11 -5
  221. esphome/components/mdns/mdns_component.cpp +11 -5
  222. esphome/components/mdns/mdns_component.h +3 -2
  223. esphome/components/mdns/mdns_esp32.cpp +4 -3
  224. esphome/components/mdns/mdns_esp8266.cpp +4 -2
  225. esphome/components/mdns/mdns_libretiny.cpp +4 -2
  226. esphome/components/mdns/mdns_rp2040.cpp +4 -2
  227. esphome/components/media_player/__init__.py +40 -6
  228. esphome/components/mhz19/sensor.py +11 -7
  229. esphome/components/micro_wake_word/__init__.py +99 -31
  230. esphome/components/micro_wake_word/automation.h +54 -0
  231. esphome/components/micro_wake_word/micro_wake_word.cpp +331 -319
  232. esphome/components/micro_wake_word/micro_wake_word.h +58 -105
  233. esphome/components/micro_wake_word/preprocessor_settings.h +19 -2
  234. esphome/components/micro_wake_word/streaming_model.cpp +158 -41
  235. esphome/components/micro_wake_word/streaming_model.h +85 -13
  236. esphome/components/microphone/__init__.py +139 -9
  237. esphome/components/microphone/automation.h +14 -2
  238. esphome/components/microphone/microphone.cpp +21 -0
  239. esphome/components/microphone/microphone.h +14 -5
  240. esphome/components/microphone/microphone_source.cpp +95 -0
  241. esphome/components/microphone/microphone_source.h +80 -0
  242. esphome/components/mics_4514/sensor.py +25 -14
  243. esphome/components/midea/climate.py +3 -4
  244. esphome/components/midea_ir/climate.py +3 -5
  245. esphome/components/mipi_spi/__init__.py +15 -0
  246. esphome/components/mipi_spi/display.py +474 -0
  247. esphome/components/mipi_spi/mipi_spi.cpp +481 -0
  248. esphome/components/mipi_spi/mipi_spi.h +171 -0
  249. esphome/components/mipi_spi/models/__init__.py +65 -0
  250. esphome/components/mipi_spi/models/amoled.py +72 -0
  251. esphome/components/mipi_spi/models/commands.py +82 -0
  252. esphome/components/mipi_spi/models/cyd.py +10 -0
  253. esphome/components/mipi_spi/models/ili.py +749 -0
  254. esphome/components/mipi_spi/models/jc.py +260 -0
  255. esphome/components/mipi_spi/models/lanbon.py +15 -0
  256. esphome/components/mipi_spi/models/lilygo.py +60 -0
  257. esphome/components/mipi_spi/models/waveshare.py +139 -0
  258. esphome/components/mitsubishi/climate.py +2 -5
  259. esphome/components/mitsubishi/mitsubishi.cpp +9 -9
  260. esphome/components/mixer/speaker/mixer_speaker.cpp +12 -22
  261. esphome/components/mixer/speaker/mixer_speaker.h +1 -3
  262. esphome/components/mlx90393/sensor.py +5 -0
  263. esphome/components/mlx90393/sensor_mlx90393.cpp +195 -13
  264. esphome/components/mlx90393/sensor_mlx90393.h +21 -4
  265. esphome/components/modbus/modbus.cpp +2 -1
  266. esphome/components/mqtt/__init__.py +1 -1
  267. esphome/components/mqtt/mqtt_client.cpp +6 -2
  268. esphome/components/mqtt/mqtt_const.h +4 -0
  269. esphome/components/mqtt/mqtt_fan.cpp +39 -0
  270. esphome/components/mqtt/mqtt_fan.h +2 -0
  271. esphome/components/ms5611/sensor.py +6 -6
  272. esphome/components/ms8607/sensor.py +3 -3
  273. esphome/components/network/__init__.py +1 -1
  274. esphome/components/nextion/base_component.py +17 -16
  275. esphome/components/nextion/display.py +11 -2
  276. esphome/components/nextion/nextion.cpp +39 -1
  277. esphome/components/nextion/nextion.h +50 -0
  278. esphome/components/noblex/climate.py +2 -9
  279. esphome/components/number/__init__.py +12 -9
  280. esphome/components/one_wire/one_wire_bus.cpp +14 -10
  281. esphome/components/one_wire/one_wire_bus.h +14 -8
  282. esphome/components/online_image/bmp_image.cpp +48 -11
  283. esphome/components/online_image/bmp_image.h +2 -0
  284. esphome/components/opentherm/binary_sensor/__init__.py +2 -4
  285. esphome/components/opentherm/number/__init__.py +11 -20
  286. esphome/components/opentherm/sensor/__init__.py +3 -3
  287. esphome/components/opentherm/switch/__init__.py +3 -5
  288. esphome/components/output/lock/__init__.py +11 -9
  289. esphome/components/packages/__init__.py +33 -31
  290. esphome/components/packet_transport/__init__.py +201 -0
  291. esphome/components/packet_transport/binary_sensor.py +19 -0
  292. esphome/components/packet_transport/packet_transport.cpp +534 -0
  293. esphome/components/packet_transport/packet_transport.h +154 -0
  294. esphome/components/packet_transport/sensor.py +19 -0
  295. esphome/components/pca9685/pca9685_output.cpp +2 -1
  296. esphome/components/pid/climate.py +2 -4
  297. esphome/components/pm2005/__init__.py +1 -0
  298. esphome/components/pm2005/pm2005.cpp +123 -0
  299. esphome/components/pm2005/pm2005.h +46 -0
  300. esphome/components/pm2005/sensor.py +86 -0
  301. esphome/components/pmsa003i/pmsa003i.cpp +43 -16
  302. esphome/components/pmsa003i/pmsa003i.h +25 -25
  303. esphome/components/pmsx003/pmsx003.cpp +195 -230
  304. esphome/components/pmsx003/pmsx003.h +51 -33
  305. esphome/components/pmsx003/sensor.py +21 -11
  306. esphome/components/pn7150/pn7150.h +2 -2
  307. esphome/components/pn7160/pn7160.h +2 -2
  308. esphome/components/prometheus/prometheus_handler.cpp +174 -0
  309. esphome/components/prometheus/prometheus_handler.h +17 -0
  310. esphome/components/psram/__init__.py +7 -5
  311. esphome/components/pulse_meter/pulse_meter_sensor.cpp +32 -12
  312. esphome/components/pulse_meter/pulse_meter_sensor.h +5 -5
  313. esphome/components/pzem004t/pzem004t.cpp +2 -1
  314. esphome/components/qspi_dbi/__init__.py +0 -1
  315. esphome/components/qspi_dbi/display.py +2 -1
  316. esphome/components/qspi_dbi/models.py +1 -2
  317. esphome/components/remote_base/__init__.py +91 -0
  318. esphome/components/remote_base/beo4_protocol.cpp +153 -0
  319. esphome/components/remote_base/beo4_protocol.h +43 -0
  320. esphome/components/remote_base/gobox_protocol.cpp +131 -0
  321. esphome/components/remote_base/gobox_protocol.h +54 -0
  322. esphome/components/remote_receiver/remote_receiver_esp32.cpp +16 -9
  323. esphome/components/resampler/speaker/resampler_speaker.cpp +12 -10
  324. esphome/components/resampler/speaker/resampler_speaker.h +1 -1
  325. esphome/components/rf_bridge/rf_bridge.cpp +2 -1
  326. esphome/components/scd30/sensor.py +2 -3
  327. esphome/components/scd4x/sensor.py +4 -5
  328. esphome/components/sdp3x/sensor.py +2 -1
  329. esphome/components/sds011/sds011.cpp +2 -1
  330. esphome/components/select/__init__.py +19 -20
  331. esphome/components/sen5x/sen5x.cpp +55 -36
  332. esphome/components/sen5x/sensor.py +1 -1
  333. esphome/components/senseair/sensor.py +3 -3
  334. esphome/components/sensor/__init__.py +158 -14
  335. esphome/components/sensor/filter.cpp +23 -0
  336. esphome/components/sensor/filter.h +22 -0
  337. esphome/components/sgp30/sensor.py +14 -16
  338. esphome/components/sgp4x/sensor.py +1 -1
  339. esphome/components/sht4x/sht4x.cpp +43 -22
  340. esphome/components/sht4x/sht4x.h +1 -1
  341. esphome/components/shtcx/sensor.py +6 -6
  342. esphome/components/slow_pwm/slow_pwm_output.cpp +2 -1
  343. esphome/components/sml/text_sensor/__init__.py +4 -6
  344. esphome/components/sound_level/__init__.py +0 -0
  345. esphome/components/sound_level/sensor.py +97 -0
  346. esphome/components/sound_level/sound_level.cpp +194 -0
  347. esphome/components/sound_level/sound_level.h +73 -0
  348. esphome/components/speaker/media_player/__init__.py +4 -8
  349. esphome/components/speaker/media_player/speaker_media_player.cpp +0 -18
  350. esphome/components/speaker/media_player/speaker_media_player.h +0 -11
  351. esphome/components/speaker/speaker.h +4 -7
  352. esphome/components/speed/fan/__init__.py +17 -16
  353. esphome/components/spi/spi.h +11 -1
  354. esphome/components/sprinkler/__init__.py +18 -19
  355. esphome/components/sprinkler/sprinkler.cpp +6 -5
  356. esphome/components/switch/__init__.py +32 -42
  357. esphome/components/syslog/__init__.py +41 -0
  358. esphome/components/syslog/esphome_syslog.cpp +49 -0
  359. esphome/components/syslog/esphome_syslog.h +27 -0
  360. esphome/components/t6615/sensor.py +3 -3
  361. esphome/components/t6615/t6615.cpp +2 -1
  362. esphome/components/tca9555/tca9555.cpp +11 -6
  363. esphome/components/tcl112/climate.py +2 -9
  364. esphome/components/template/alarm_control_panel/__init__.py +7 -6
  365. esphome/components/template/alarm_control_panel/template_alarm_control_panel.cpp +21 -17
  366. esphome/components/template/alarm_control_panel/template_alarm_control_panel.h +2 -1
  367. esphome/components/template/cover/__init__.py +27 -21
  368. esphome/components/template/fan/__init__.py +14 -12
  369. esphome/components/template/lock/__init__.py +20 -25
  370. esphome/components/template/lock/automation.h +18 -0
  371. esphome/components/template/text/__init__.py +4 -3
  372. esphome/components/template/valve/__init__.py +32 -21
  373. esphome/components/template/valve/automation.h +24 -0
  374. esphome/components/text/__init__.py +32 -1
  375. esphome/components/text_sensor/__init__.py +24 -29
  376. esphome/components/thermostat/climate.py +5 -5
  377. esphome/components/time_based/cover.py +17 -16
  378. esphome/components/time_based/time_based_cover.cpp +2 -1
  379. esphome/components/tm1638/switch/__init__.py +10 -7
  380. esphome/components/tormatic/cover.py +4 -5
  381. esphome/components/toshiba/climate.py +3 -5
  382. esphome/components/touchscreen/touchscreen.cpp +3 -1
  383. esphome/components/tt21100/touchscreen/tt21100.cpp +1 -1
  384. esphome/components/tuya/climate/__init__.py +5 -6
  385. esphome/components/tuya/cover/__init__.py +6 -11
  386. esphome/components/tuya/select/__init__.py +15 -5
  387. esphome/components/tuya/select/tuya_select.cpp +6 -1
  388. esphome/components/tuya/select/tuya_select.h +5 -1
  389. esphome/components/uart/packet_transport/__init__.py +20 -0
  390. esphome/components/uart/packet_transport/uart_transport.cpp +88 -0
  391. esphome/components/uart/packet_transport/uart_transport.h +41 -0
  392. esphome/components/uart/switch/uart_switch.cpp +2 -1
  393. esphome/components/udp/__init__.py +126 -128
  394. esphome/components/udp/automation.h +40 -0
  395. esphome/components/udp/binary_sensor.py +3 -25
  396. esphome/components/udp/packet_transport/__init__.py +29 -0
  397. esphome/components/udp/packet_transport/udp_transport.cpp +36 -0
  398. esphome/components/udp/packet_transport/udp_transport.h +28 -0
  399. esphome/components/udp/sensor.py +3 -25
  400. esphome/components/udp/udp_component.cpp +26 -470
  401. esphome/components/udp/udp_component.h +21 -128
  402. esphome/components/update/__init__.py +31 -1
  403. esphome/components/uponor_smatrix/climate/__init__.py +4 -9
  404. esphome/components/uponor_smatrix/climate/uponor_smatrix_climate.cpp +2 -1
  405. esphome/components/uponor_smatrix/uponor_smatrix.cpp +2 -1
  406. esphome/components/uptime/text_sensor/__init__.py +47 -7
  407. esphome/components/uptime/text_sensor/uptime_text_sensor.cpp +12 -7
  408. esphome/components/uptime/text_sensor/uptime_text_sensor.h +19 -0
  409. esphome/components/valve/__init__.py +34 -3
  410. esphome/components/valve/automation.h +1 -19
  411. esphome/components/vl53l0x/sensor.py +11 -0
  412. esphome/components/vl53l0x/vl53l0x_sensor.cpp +5 -1
  413. esphome/components/vl53l0x/vl53l0x_sensor.h +2 -1
  414. esphome/components/voice_assistant/__init__.py +36 -10
  415. esphome/components/voice_assistant/voice_assistant.cpp +170 -144
  416. esphome/components/voice_assistant/voice_assistant.h +26 -25
  417. esphome/components/waveshare_epaper/display.py +6 -0
  418. esphome/components/waveshare_epaper/waveshare_epaper.cpp +439 -37
  419. esphome/components/waveshare_epaper/waveshare_epaper.h +60 -11
  420. esphome/components/weikai/weikai.cpp +0 -52
  421. esphome/components/whirlpool/climate.py +3 -5
  422. esphome/components/whynter/climate.py +3 -5
  423. esphome/components/xpt2046/touchscreen/xpt2046.cpp +1 -1
  424. esphome/components/yashima/climate.py +6 -6
  425. esphome/components/zhlt01/climate.py +2 -7
  426. esphome/config.py +13 -13
  427. esphome/config_validation.py +38 -58
  428. esphome/const.py +15 -1
  429. esphome/core/__init__.py +2 -0
  430. esphome/core/application.cpp +27 -10
  431. esphome/core/application.h +9 -1
  432. esphome/core/automation.h +4 -3
  433. esphome/core/component.cpp +28 -7
  434. esphome/core/component.h +10 -1
  435. esphome/core/defines.h +23 -17
  436. esphome/core/doxygen.h +13 -0
  437. esphome/core/macros.h +4 -0
  438. esphome/core/scheduler.cpp +7 -1
  439. esphome/cpp_generator.py +6 -2
  440. esphome/dashboard/web_server.py +3 -3
  441. esphome/helpers.py +39 -0
  442. esphome/loader.py +4 -0
  443. esphome/log.py +15 -19
  444. esphome/mqtt.py +23 -10
  445. esphome/platformio_api.py +1 -1
  446. esphome/schema_extractors.py +0 -1
  447. esphome/voluptuous_schema.py +3 -1
  448. esphome/vscode.py +15 -0
  449. esphome/wizard.py +47 -37
  450. esphome/zeroconf.py +7 -3
  451. {esphome-2025.4.1.dist-info → esphome-2025.5.0.dist-info}/METADATA +10 -11
  452. {esphome-2025.4.1.dist-info → esphome-2025.5.0.dist-info}/RECORD +456 -396
  453. {esphome-2025.4.1.dist-info → esphome-2025.5.0.dist-info}/WHEEL +1 -1
  454. esphome/components/esp32_ble/const_esp32c6.h +0 -74
  455. {esphome-2025.4.1.dist-info → esphome-2025.5.0.dist-info}/entry_points.txt +0 -0
  456. {esphome-2025.4.1.dist-info → esphome-2025.5.0.dist-info}/licenses/LICENSE +0 -0
  457. {esphome-2025.4.1.dist-info → esphome-2025.5.0.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,361 @@
1
+ #pragma once
2
+
3
+ #include "proto.h"
4
+ #include <cstdint>
5
+ #include <string>
6
+
7
+ namespace esphome {
8
+ namespace api {
9
+
10
+ class ProtoSize {
11
+ public:
12
+ /**
13
+ * @brief ProtoSize class for Protocol Buffer serialization size calculation
14
+ *
15
+ * This class provides static methods to calculate the exact byte counts needed
16
+ * for encoding various Protocol Buffer field types. All methods are designed to be
17
+ * efficient for the common case where many fields have default values.
18
+ *
19
+ * Implements Protocol Buffer encoding size calculation according to:
20
+ * https://protobuf.dev/programming-guides/encoding/
21
+ *
22
+ * Key features:
23
+ * - Early-return optimization for zero/default values
24
+ * - Direct total_size updates to avoid unnecessary additions
25
+ * - Specialized handling for different field types according to protobuf spec
26
+ * - Templated helpers for repeated fields and messages
27
+ */
28
+
29
+ /**
30
+ * @brief Calculates the size in bytes needed to encode a uint32_t value as a varint
31
+ *
32
+ * @param value The uint32_t value to calculate size for
33
+ * @return The number of bytes needed to encode the value
34
+ */
35
+ static inline uint32_t varint(uint32_t value) {
36
+ // Optimized varint size calculation using leading zeros
37
+ // Each 7 bits requires one byte in the varint encoding
38
+ if (value < 128)
39
+ return 1; // 7 bits, common case for small values
40
+
41
+ // For larger values, count bytes needed based on the position of the highest bit set
42
+ if (value < 16384) {
43
+ return 2; // 14 bits
44
+ } else if (value < 2097152) {
45
+ return 3; // 21 bits
46
+ } else if (value < 268435456) {
47
+ return 4; // 28 bits
48
+ } else {
49
+ return 5; // 32 bits (maximum for uint32_t)
50
+ }
51
+ }
52
+
53
+ /**
54
+ * @brief Calculates the size in bytes needed to encode a uint64_t value as a varint
55
+ *
56
+ * @param value The uint64_t value to calculate size for
57
+ * @return The number of bytes needed to encode the value
58
+ */
59
+ static inline uint32_t varint(uint64_t value) {
60
+ // Handle common case of values fitting in uint32_t (vast majority of use cases)
61
+ if (value <= UINT32_MAX) {
62
+ return varint(static_cast<uint32_t>(value));
63
+ }
64
+
65
+ // For larger values, determine size based on highest bit position
66
+ if (value < (1ULL << 35)) {
67
+ return 5; // 35 bits
68
+ } else if (value < (1ULL << 42)) {
69
+ return 6; // 42 bits
70
+ } else if (value < (1ULL << 49)) {
71
+ return 7; // 49 bits
72
+ } else if (value < (1ULL << 56)) {
73
+ return 8; // 56 bits
74
+ } else if (value < (1ULL << 63)) {
75
+ return 9; // 63 bits
76
+ } else {
77
+ return 10; // 64 bits (maximum for uint64_t)
78
+ }
79
+ }
80
+
81
+ /**
82
+ * @brief Calculates the size in bytes needed to encode an int32_t value as a varint
83
+ *
84
+ * Special handling is needed for negative values, which are sign-extended to 64 bits
85
+ * in Protocol Buffers, resulting in a 10-byte varint.
86
+ *
87
+ * @param value The int32_t value to calculate size for
88
+ * @return The number of bytes needed to encode the value
89
+ */
90
+ static inline uint32_t varint(int32_t value) {
91
+ // Negative values are sign-extended to 64 bits in protocol buffers,
92
+ // which always results in a 10-byte varint for negative int32
93
+ if (value < 0) {
94
+ return 10; // Negative int32 is always 10 bytes long
95
+ }
96
+ // For non-negative values, use the uint32_t implementation
97
+ return varint(static_cast<uint32_t>(value));
98
+ }
99
+
100
+ /**
101
+ * @brief Calculates the size in bytes needed to encode an int64_t value as a varint
102
+ *
103
+ * @param value The int64_t value to calculate size for
104
+ * @return The number of bytes needed to encode the value
105
+ */
106
+ static inline uint32_t varint(int64_t value) {
107
+ // For int64_t, we convert to uint64_t and calculate the size
108
+ // This works because the bit pattern determines the encoding size,
109
+ // and we've handled negative int32 values as a special case above
110
+ return varint(static_cast<uint64_t>(value));
111
+ }
112
+
113
+ /**
114
+ * @brief Calculates the size in bytes needed to encode a field ID and wire type
115
+ *
116
+ * @param field_id The field identifier
117
+ * @param type The wire type value (from the WireType enum in the protobuf spec)
118
+ * @return The number of bytes needed to encode the field ID and wire type
119
+ */
120
+ static inline uint32_t field(uint32_t field_id, uint32_t type) {
121
+ uint32_t tag = (field_id << 3) | (type & 0b111);
122
+ return varint(tag);
123
+ }
124
+
125
+ /**
126
+ * @brief Common parameters for all add_*_field methods
127
+ *
128
+ * All add_*_field methods follow these common patterns:
129
+ *
130
+ * @param total_size Reference to the total message size to update
131
+ * @param field_id_size Pre-calculated size of the field ID in bytes
132
+ * @param value The value to calculate size for (type varies)
133
+ * @param force Whether to calculate size even if the value is default/zero/empty
134
+ *
135
+ * Each method follows this implementation pattern:
136
+ * 1. Skip calculation if value is default (0, false, empty) and not forced
137
+ * 2. Calculate the size based on the field's encoding rules
138
+ * 3. Add the field_id_size + calculated value size to total_size
139
+ */
140
+
141
+ /**
142
+ * @brief Calculates and adds the size of an int32 field to the total message size
143
+ */
144
+ static inline void add_int32_field(uint32_t &total_size, uint32_t field_id_size, int32_t value, bool force = false) {
145
+ // Skip calculation if value is zero and not forced
146
+ if (value == 0 && !force) {
147
+ return; // No need to update total_size
148
+ }
149
+
150
+ // Calculate and directly add to total_size
151
+ if (value < 0) {
152
+ // Negative values are encoded as 10-byte varints in protobuf
153
+ total_size += field_id_size + 10;
154
+ } else {
155
+ // For non-negative values, use the standard varint size
156
+ total_size += field_id_size + varint(static_cast<uint32_t>(value));
157
+ }
158
+ }
159
+
160
+ /**
161
+ * @brief Calculates and adds the size of a uint32 field to the total message size
162
+ */
163
+ static inline void add_uint32_field(uint32_t &total_size, uint32_t field_id_size, uint32_t value,
164
+ bool force = false) {
165
+ // Skip calculation if value is zero and not forced
166
+ if (value == 0 && !force) {
167
+ return; // No need to update total_size
168
+ }
169
+
170
+ // Calculate and directly add to total_size
171
+ total_size += field_id_size + varint(value);
172
+ }
173
+
174
+ /**
175
+ * @brief Calculates and adds the size of a boolean field to the total message size
176
+ */
177
+ static inline void add_bool_field(uint32_t &total_size, uint32_t field_id_size, bool value, bool force = false) {
178
+ // Skip calculation if value is false and not forced
179
+ if (!value && !force) {
180
+ return; // No need to update total_size
181
+ }
182
+
183
+ // Boolean fields always use 1 byte when true
184
+ total_size += field_id_size + 1;
185
+ }
186
+
187
+ /**
188
+ * @brief Calculates and adds the size of a fixed field to the total message size
189
+ *
190
+ * Fixed fields always take exactly N bytes (4 for fixed32/float, 8 for fixed64/double).
191
+ *
192
+ * @tparam NumBytes The number of bytes for this fixed field (4 or 8)
193
+ * @param is_nonzero Whether the value is non-zero
194
+ */
195
+ template<uint32_t NumBytes>
196
+ static inline void add_fixed_field(uint32_t &total_size, uint32_t field_id_size, bool is_nonzero,
197
+ bool force = false) {
198
+ // Skip calculation if value is zero and not forced
199
+ if (!is_nonzero && !force) {
200
+ return; // No need to update total_size
201
+ }
202
+
203
+ // Fixed fields always take exactly NumBytes
204
+ total_size += field_id_size + NumBytes;
205
+ }
206
+
207
+ /**
208
+ * @brief Calculates and adds the size of an enum field to the total message size
209
+ *
210
+ * Enum fields are encoded as uint32 varints.
211
+ */
212
+ static inline void add_enum_field(uint32_t &total_size, uint32_t field_id_size, uint32_t value, bool force = false) {
213
+ // Skip calculation if value is zero and not forced
214
+ if (value == 0 && !force) {
215
+ return; // No need to update total_size
216
+ }
217
+
218
+ // Enums are encoded as uint32
219
+ total_size += field_id_size + varint(value);
220
+ }
221
+
222
+ /**
223
+ * @brief Calculates and adds the size of a sint32 field to the total message size
224
+ *
225
+ * Sint32 fields use ZigZag encoding, which is more efficient for negative values.
226
+ */
227
+ static inline void add_sint32_field(uint32_t &total_size, uint32_t field_id_size, int32_t value, bool force = false) {
228
+ // Skip calculation if value is zero and not forced
229
+ if (value == 0 && !force) {
230
+ return; // No need to update total_size
231
+ }
232
+
233
+ // ZigZag encoding for sint32: (n << 1) ^ (n >> 31)
234
+ uint32_t zigzag = (static_cast<uint32_t>(value) << 1) ^ (static_cast<uint32_t>(value >> 31));
235
+ total_size += field_id_size + varint(zigzag);
236
+ }
237
+
238
+ /**
239
+ * @brief Calculates and adds the size of an int64 field to the total message size
240
+ */
241
+ static inline void add_int64_field(uint32_t &total_size, uint32_t field_id_size, int64_t value, bool force = false) {
242
+ // Skip calculation if value is zero and not forced
243
+ if (value == 0 && !force) {
244
+ return; // No need to update total_size
245
+ }
246
+
247
+ // Calculate and directly add to total_size
248
+ total_size += field_id_size + varint(value);
249
+ }
250
+
251
+ /**
252
+ * @brief Calculates and adds the size of a uint64 field to the total message size
253
+ */
254
+ static inline void add_uint64_field(uint32_t &total_size, uint32_t field_id_size, uint64_t value,
255
+ bool force = false) {
256
+ // Skip calculation if value is zero and not forced
257
+ if (value == 0 && !force) {
258
+ return; // No need to update total_size
259
+ }
260
+
261
+ // Calculate and directly add to total_size
262
+ total_size += field_id_size + varint(value);
263
+ }
264
+
265
+ /**
266
+ * @brief Calculates and adds the size of a sint64 field to the total message size
267
+ *
268
+ * Sint64 fields use ZigZag encoding, which is more efficient for negative values.
269
+ */
270
+ static inline void add_sint64_field(uint32_t &total_size, uint32_t field_id_size, int64_t value, bool force = false) {
271
+ // Skip calculation if value is zero and not forced
272
+ if (value == 0 && !force) {
273
+ return; // No need to update total_size
274
+ }
275
+
276
+ // ZigZag encoding for sint64: (n << 1) ^ (n >> 63)
277
+ uint64_t zigzag = (static_cast<uint64_t>(value) << 1) ^ (static_cast<uint64_t>(value >> 63));
278
+ total_size += field_id_size + varint(zigzag);
279
+ }
280
+
281
+ /**
282
+ * @brief Calculates and adds the size of a string/bytes field to the total message size
283
+ */
284
+ static inline void add_string_field(uint32_t &total_size, uint32_t field_id_size, const std::string &str,
285
+ bool force = false) {
286
+ // Skip calculation if string is empty and not forced
287
+ if (str.empty() && !force) {
288
+ return; // No need to update total_size
289
+ }
290
+
291
+ // Calculate and directly add to total_size
292
+ const uint32_t str_size = static_cast<uint32_t>(str.size());
293
+ total_size += field_id_size + varint(str_size) + str_size;
294
+ }
295
+
296
+ /**
297
+ * @brief Calculates and adds the size of a nested message field to the total message size
298
+ *
299
+ * This helper function directly updates the total_size reference if the nested size
300
+ * is greater than zero or force is true.
301
+ *
302
+ * @param nested_size The pre-calculated size of the nested message
303
+ */
304
+ static inline void add_message_field(uint32_t &total_size, uint32_t field_id_size, uint32_t nested_size,
305
+ bool force = false) {
306
+ // Skip calculation if nested message is empty and not forced
307
+ if (nested_size == 0 && !force) {
308
+ return; // No need to update total_size
309
+ }
310
+
311
+ // Calculate and directly add to total_size
312
+ // Field ID + length varint + nested message content
313
+ total_size += field_id_size + varint(nested_size) + nested_size;
314
+ }
315
+
316
+ /**
317
+ * @brief Calculates and adds the size of a nested message field to the total message size
318
+ *
319
+ * This templated version directly takes a message object, calculates its size internally,
320
+ * and updates the total_size reference. This eliminates the need for a temporary variable
321
+ * at the call site.
322
+ *
323
+ * @tparam MessageType The type of the nested message (inferred from parameter)
324
+ * @param message The nested message object
325
+ */
326
+ template<typename MessageType>
327
+ static inline void add_message_object(uint32_t &total_size, uint32_t field_id_size, const MessageType &message,
328
+ bool force = false) {
329
+ uint32_t nested_size = 0;
330
+ message.calculate_size(nested_size);
331
+
332
+ // Use the base implementation with the calculated nested_size
333
+ add_message_field(total_size, field_id_size, nested_size, force);
334
+ }
335
+
336
+ /**
337
+ * @brief Calculates and adds the sizes of all messages in a repeated field to the total message size
338
+ *
339
+ * This helper processes a vector of message objects, calculating the size for each message
340
+ * and adding it to the total size.
341
+ *
342
+ * @tparam MessageType The type of the nested messages in the vector
343
+ * @param messages Vector of message objects
344
+ */
345
+ template<typename MessageType>
346
+ static inline void add_repeated_message(uint32_t &total_size, uint32_t field_id_size,
347
+ const std::vector<MessageType> &messages) {
348
+ // Skip if the vector is empty
349
+ if (messages.empty()) {
350
+ return;
351
+ }
352
+
353
+ // For repeated fields, always use force=true
354
+ for (const auto &message : messages) {
355
+ add_message_object(total_size, field_id_size, message, true);
356
+ }
357
+ }
358
+ };
359
+
360
+ } // namespace api
361
+ } // namespace esphome
@@ -22,22 +22,40 @@ namespace api {
22
22
  static const char *const TAG = "api";
23
23
 
24
24
  // APIServer
25
+ APIServer *global_api_server = nullptr; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
26
+
27
+ APIServer::APIServer() { global_api_server = this; }
28
+
25
29
  void APIServer::setup() {
26
30
  ESP_LOGCONFIG(TAG, "Setting up Home Assistant API server...");
27
31
  this->setup_controller();
28
- socket_ = socket::socket_ip(SOCK_STREAM, 0);
29
- if (socket_ == nullptr) {
30
- ESP_LOGW(TAG, "Could not create socket.");
32
+
33
+ #ifdef USE_API_NOISE
34
+ uint32_t hash = 88491486UL;
35
+
36
+ this->noise_pref_ = global_preferences->make_preference<SavedNoisePsk>(hash, true);
37
+
38
+ SavedNoisePsk noise_pref_saved{};
39
+ if (this->noise_pref_.load(&noise_pref_saved)) {
40
+ ESP_LOGD(TAG, "Loaded saved Noise PSK");
41
+
42
+ this->set_noise_psk(noise_pref_saved.psk);
43
+ }
44
+ #endif
45
+
46
+ this->socket_ = socket::socket_ip(SOCK_STREAM, 0);
47
+ if (this->socket_ == nullptr) {
48
+ ESP_LOGW(TAG, "Could not create socket");
31
49
  this->mark_failed();
32
50
  return;
33
51
  }
34
52
  int enable = 1;
35
- int err = socket_->setsockopt(SOL_SOCKET, SO_REUSEADDR, &enable, sizeof(int));
53
+ int err = this->socket_->setsockopt(SOL_SOCKET, SO_REUSEADDR, &enable, sizeof(int));
36
54
  if (err != 0) {
37
55
  ESP_LOGW(TAG, "Socket unable to set reuseaddr: errno %d", err);
38
56
  // we can still continue
39
57
  }
40
- err = socket_->setblocking(false);
58
+ err = this->socket_->setblocking(false);
41
59
  if (err != 0) {
42
60
  ESP_LOGW(TAG, "Socket unable to set nonblocking mode: errno %d", err);
43
61
  this->mark_failed();
@@ -53,14 +71,14 @@ void APIServer::setup() {
53
71
  return;
54
72
  }
55
73
 
56
- err = socket_->bind((struct sockaddr *) &server, sl);
74
+ err = this->socket_->bind((struct sockaddr *) &server, sl);
57
75
  if (err != 0) {
58
76
  ESP_LOGW(TAG, "Socket unable to bind: errno %d", errno);
59
77
  this->mark_failed();
60
78
  return;
61
79
  }
62
80
 
63
- err = socket_->listen(4);
81
+ err = this->socket_->listen(4);
64
82
  if (err != 0) {
65
83
  ESP_LOGW(TAG, "Socket unable to listen: errno %d", errno);
66
84
  this->mark_failed();
@@ -92,34 +110,45 @@ void APIServer::setup() {
92
110
  }
93
111
  #endif
94
112
  }
113
+
95
114
  void APIServer::loop() {
96
115
  // Accept new clients
97
116
  while (true) {
98
117
  struct sockaddr_storage source_addr;
99
118
  socklen_t addr_len = sizeof(source_addr);
100
- auto sock = socket_->accept((struct sockaddr *) &source_addr, &addr_len);
119
+ auto sock = this->socket_->accept((struct sockaddr *) &source_addr, &addr_len);
101
120
  if (!sock)
102
121
  break;
103
122
  ESP_LOGD(TAG, "Accepted %s", sock->getpeername().c_str());
104
123
 
105
124
  auto *conn = new APIConnection(std::move(sock), this);
106
- clients_.emplace_back(conn);
125
+ this->clients_.emplace_back(conn);
107
126
  conn->start();
108
127
  }
109
128
 
110
- // Partition clients into remove and active
111
- auto new_end = std::partition(this->clients_.begin(), this->clients_.end(),
112
- [](const std::unique_ptr<APIConnection> &conn) { return !conn->remove_; });
113
- // print disconnection messages
114
- for (auto it = new_end; it != this->clients_.end(); ++it) {
115
- this->client_disconnected_trigger_->trigger((*it)->client_info_, (*it)->client_peername_);
116
- ESP_LOGV(TAG, "Removing connection to %s", (*it)->client_info_.c_str());
117
- }
118
- // resize vector
119
- this->clients_.erase(new_end, this->clients_.end());
120
-
121
- for (auto &client : this->clients_) {
122
- client->loop();
129
+ // Process clients and remove disconnected ones in a single pass
130
+ if (!this->clients_.empty()) {
131
+ size_t client_index = 0;
132
+ while (client_index < this->clients_.size()) {
133
+ auto &client = this->clients_[client_index];
134
+
135
+ if (client->remove_) {
136
+ // Handle disconnection
137
+ this->client_disconnected_trigger_->trigger(client->client_info_, client->client_peername_);
138
+ ESP_LOGV(TAG, "Removing connection to %s", client->client_info_.c_str());
139
+
140
+ // Swap with the last element and pop (avoids expensive vector shifts)
141
+ if (client_index < this->clients_.size() - 1) {
142
+ std::swap(this->clients_[client_index], this->clients_.back());
143
+ }
144
+ this->clients_.pop_back();
145
+ // Don't increment client_index since we need to process the swapped element
146
+ } else {
147
+ // Process active client
148
+ client->loop();
149
+ client_index++; // Move to next client
150
+ }
151
+ }
123
152
  }
124
153
 
125
154
  if (this->reboot_timeout_ != 0) {
@@ -136,16 +165,22 @@ void APIServer::loop() {
136
165
  }
137
166
  }
138
167
  }
168
+
139
169
  void APIServer::dump_config() {
140
170
  ESP_LOGCONFIG(TAG, "API Server:");
141
171
  ESP_LOGCONFIG(TAG, " Address: %s:%u", network::get_use_address().c_str(), this->port_);
142
172
  #ifdef USE_API_NOISE
143
- ESP_LOGCONFIG(TAG, " Using noise encryption: YES");
173
+ ESP_LOGCONFIG(TAG, " Using noise encryption: %s", YESNO(this->noise_ctx_->has_psk()));
174
+ if (!this->noise_ctx_->has_psk()) {
175
+ ESP_LOGCONFIG(TAG, " Supports noise encryption: YES");
176
+ }
144
177
  #else
145
178
  ESP_LOGCONFIG(TAG, " Using noise encryption: NO");
146
179
  #endif
147
180
  }
181
+
148
182
  bool APIServer::uses_password() const { return !this->password_.empty(); }
183
+
149
184
  bool APIServer::check_password(const std::string &password) const {
150
185
  // depend only on input password length
151
186
  const char *a = this->password_.c_str();
@@ -174,7 +209,9 @@ bool APIServer::check_password(const std::string &password) const {
174
209
 
175
210
  return result == 0;
176
211
  }
212
+
177
213
  void APIServer::handle_disconnect(APIConnection *conn) {}
214
+
178
215
  #ifdef USE_BINARY_SENSOR
179
216
  void APIServer::on_binary_sensor_update(binary_sensor::BinarySensor *obj, bool state) {
180
217
  if (obj->is_internal())
@@ -342,18 +379,27 @@ void APIServer::on_update(update::UpdateEntity *obj) {
342
379
  }
343
380
  #endif
344
381
 
382
+ #ifdef USE_ALARM_CONTROL_PANEL
383
+ void APIServer::on_alarm_control_panel_update(alarm_control_panel::AlarmControlPanel *obj) {
384
+ if (obj->is_internal())
385
+ return;
386
+ for (auto &c : this->clients_)
387
+ c->send_alarm_control_panel_state(obj);
388
+ }
389
+ #endif
390
+
345
391
  float APIServer::get_setup_priority() const { return setup_priority::AFTER_WIFI; }
392
+
346
393
  void APIServer::set_port(uint16_t port) { this->port_ = port; }
347
- APIServer *global_api_server = nullptr; // NOLINT(cppcoreguidelines-avoid-non-const-global-variables)
348
394
 
349
395
  void APIServer::set_password(const std::string &password) { this->password_ = password; }
396
+
350
397
  void APIServer::send_homeassistant_service_call(const HomeassistantServiceResponse &call) {
351
398
  for (auto &client : this->clients_) {
352
399
  client->send_homeassistant_service_call(call);
353
400
  }
354
401
  }
355
402
 
356
- APIServer::APIServer() { global_api_server = this; }
357
403
  void APIServer::subscribe_home_assistant_state(std::string entity_id, optional<std::string> attribute,
358
404
  std::function<void(std::string)> f) {
359
405
  this->state_subs_.push_back(HomeAssistantStateSubscription{
@@ -363,6 +409,7 @@ void APIServer::subscribe_home_assistant_state(std::string entity_id, optional<s
363
409
  .once = false,
364
410
  });
365
411
  }
412
+
366
413
  void APIServer::get_home_assistant_state(std::string entity_id, optional<std::string> attribute,
367
414
  std::function<void(std::string)> f) {
368
415
  this->state_subs_.push_back(HomeAssistantStateSubscription{
@@ -372,11 +419,47 @@ void APIServer::get_home_assistant_state(std::string entity_id, optional<std::st
372
419
  .once = true,
373
420
  });
374
421
  };
422
+
375
423
  const std::vector<APIServer::HomeAssistantStateSubscription> &APIServer::get_state_subs() const {
376
424
  return this->state_subs_;
377
425
  }
426
+
378
427
  uint16_t APIServer::get_port() const { return this->port_; }
428
+
379
429
  void APIServer::set_reboot_timeout(uint32_t reboot_timeout) { this->reboot_timeout_ = reboot_timeout; }
430
+
431
+ #ifdef USE_API_NOISE
432
+ bool APIServer::save_noise_psk(psk_t psk, bool make_active) {
433
+ auto &old_psk = this->noise_ctx_->get_psk();
434
+ if (std::equal(old_psk.begin(), old_psk.end(), psk.begin())) {
435
+ ESP_LOGW(TAG, "New PSK matches old");
436
+ return true;
437
+ }
438
+
439
+ SavedNoisePsk new_saved_psk{psk};
440
+ if (!this->noise_pref_.save(&new_saved_psk)) {
441
+ ESP_LOGW(TAG, "Failed to save Noise PSK");
442
+ return false;
443
+ }
444
+ // ensure it's written immediately
445
+ if (!global_preferences->sync()) {
446
+ ESP_LOGW(TAG, "Failed to sync preferences");
447
+ return false;
448
+ }
449
+ ESP_LOGD(TAG, "Noise PSK saved");
450
+ if (make_active) {
451
+ this->set_timeout(100, [this, psk]() {
452
+ ESP_LOGW(TAG, "Disconnecting all clients to reset connections");
453
+ this->set_noise_psk(psk);
454
+ for (auto &c : this->clients_) {
455
+ c->send_disconnect_request(DisconnectRequest());
456
+ }
457
+ });
458
+ }
459
+ return true;
460
+ }
461
+ #endif
462
+
380
463
  #ifdef USE_HOMEASSISTANT_TIME
381
464
  void APIServer::request_time() {
382
465
  for (auto &client : this->clients_) {
@@ -385,7 +468,9 @@ void APIServer::request_time() {
385
468
  }
386
469
  }
387
470
  #endif
471
+
388
472
  bool APIServer::is_connected() const { return !this->clients_.empty(); }
473
+
389
474
  void APIServer::on_shutdown() {
390
475
  for (auto &c : this->clients_) {
391
476
  c->send_disconnect_request(DisconnectRequest());
@@ -393,15 +478,6 @@ void APIServer::on_shutdown() {
393
478
  delay(10);
394
479
  }
395
480
 
396
- #ifdef USE_ALARM_CONTROL_PANEL
397
- void APIServer::on_alarm_control_panel_update(alarm_control_panel::AlarmControlPanel *obj) {
398
- if (obj->is_internal())
399
- return;
400
- for (auto &c : this->clients_)
401
- c->send_alarm_control_panel_state(obj);
402
- }
403
- #endif
404
-
405
481
  } // namespace api
406
482
  } // namespace esphome
407
483
  #endif
@@ -19,6 +19,12 @@
19
19
  namespace esphome {
20
20
  namespace api {
21
21
 
22
+ #ifdef USE_API_NOISE
23
+ struct SavedNoisePsk {
24
+ psk_t psk;
25
+ } PACKED; // NOLINT
26
+ #endif
27
+
22
28
  class APIServer : public Component, public Controller {
23
29
  public:
24
30
  APIServer();
@@ -35,6 +41,7 @@ class APIServer : public Component, public Controller {
35
41
  void set_reboot_timeout(uint32_t reboot_timeout);
36
42
 
37
43
  #ifdef USE_API_NOISE
44
+ bool save_noise_psk(psk_t psk, bool make_active = true);
38
45
  void set_noise_psk(psk_t psk) { noise_ctx_->set_psk(psk); }
39
46
  std::shared_ptr<APINoiseContext> get_noise_ctx() { return noise_ctx_; }
40
47
  #endif // USE_API_NOISE
@@ -142,6 +149,7 @@ class APIServer : public Component, public Controller {
142
149
 
143
150
  #ifdef USE_API_NOISE
144
151
  std::shared_ptr<APINoiseContext> noise_ctx_ = std::make_shared<APINoiseContext>();
152
+ ESPPreferenceObject noise_pref_;
145
153
  #endif // USE_API_NOISE
146
154
  };
147
155