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.
- esphome/__main__.py +16 -14
- esphome/components/ac_dimmer/ac_dimmer.cpp +3 -2
- esphome/components/adc/__init__.py +51 -34
- esphome/components/airthings_wave_base/__init__.py +1 -1
- esphome/components/alarm_control_panel/__init__.py +37 -2
- esphome/components/am43/cover/__init__.py +4 -5
- esphome/components/analog_threshold/analog_threshold_binary_sensor.cpp +6 -4
- esphome/components/analog_threshold/analog_threshold_binary_sensor.h +4 -5
- esphome/components/analog_threshold/binary_sensor.py +10 -8
- esphome/components/anova/climate.py +4 -5
- esphome/components/api/__init__.py +25 -8
- esphome/components/api/api_connection.cpp +416 -662
- esphome/components/api/api_connection.h +256 -57
- esphome/components/api/api_frame_helper.cpp +232 -177
- esphome/components/api/api_frame_helper.h +61 -8
- esphome/components/api/api_noise_context.h +13 -4
- esphome/components/api/api_pb2.cpp +1422 -1
- esphome/components/api/api_pb2.h +255 -1
- esphome/components/api/api_pb2_service.cpp +162 -49
- esphome/components/api/api_pb2_service.h +90 -51
- esphome/components/api/api_pb2_size.h +361 -0
- esphome/components/api/api_server.cpp +110 -34
- esphome/components/api/api_server.h +8 -0
- esphome/components/api/proto.h +86 -17
- esphome/components/as3935_i2c/as3935_i2c.h +0 -3
- esphome/components/as7341/as7341.h +1 -1
- esphome/components/at581x/at581x.h +4 -4
- esphome/components/atm90e32/__init__.py +1 -0
- esphome/components/atm90e32/atm90e32.cpp +576 -199
- esphome/components/atm90e32/atm90e32.h +128 -31
- esphome/components/atm90e32/atm90e32_reg.h +4 -2
- esphome/components/atm90e32/button/__init__.py +62 -10
- esphome/components/atm90e32/button/atm90e32_button.cpp +63 -4
- esphome/components/atm90e32/button/atm90e32_button.h +36 -4
- esphome/components/atm90e32/number/__init__.py +130 -0
- esphome/components/atm90e32/number/atm90e32_number.h +16 -0
- esphome/components/atm90e32/sensor.py +21 -4
- esphome/components/atm90e32/text_sensor/__init__.py +48 -0
- esphome/components/audio/__init__.py +96 -49
- esphome/components/audio/audio.h +48 -0
- esphome/components/audio/audio_decoder.cpp +1 -1
- esphome/components/audio/audio_resampler.cpp +2 -0
- esphome/components/audio/audio_resampler.h +1 -0
- esphome/components/ballu/climate.py +2 -9
- esphome/components/bang_bang/climate.py +5 -6
- esphome/components/bedjet/bedjet_hub.cpp +1 -0
- esphome/components/bedjet/climate/__init__.py +3 -8
- esphome/components/bedjet/fan/__init__.py +2 -11
- esphome/components/binary/fan/__init__.py +13 -16
- esphome/components/binary_sensor/__init__.py +13 -10
- esphome/components/bl0906/constants.h +16 -16
- esphome/components/ble_client/text_sensor/__init__.py +3 -5
- esphome/components/bluetooth_proxy/bluetooth_connection.cpp +4 -6
- esphome/components/bluetooth_proxy/bluetooth_proxy.cpp +136 -21
- esphome/components/bluetooth_proxy/bluetooth_proxy.h +7 -0
- esphome/components/button/__init__.py +11 -8
- esphome/components/canbus/canbus.cpp +3 -0
- esphome/components/canbus/canbus.h +16 -0
- esphome/components/ccs811/sensor.py +9 -6
- esphome/components/climate/__init__.py +35 -2
- esphome/components/climate/climate_mode.h +1 -1
- esphome/components/climate/climate_traits.h +63 -57
- esphome/components/climate_ir/__init__.py +57 -17
- esphome/components/climate_ir_lg/climate.py +2 -5
- esphome/components/climate_ir_lg/climate_ir_lg.cpp +7 -7
- esphome/components/climate_ir_lg/climate_ir_lg.h +1 -1
- esphome/components/color/__init__.py +2 -0
- esphome/components/const/__init__.py +5 -0
- esphome/components/coolix/climate.py +2 -9
- esphome/components/copy/cover/__init__.py +10 -9
- esphome/components/copy/fan/__init__.py +11 -9
- esphome/components/copy/lock/__init__.py +11 -9
- esphome/components/copy/text/__init__.py +9 -6
- esphome/components/cover/__init__.py +37 -2
- esphome/components/cse7766/cse7766.cpp +2 -1
- esphome/components/cst226/binary_sensor/__init__.py +28 -0
- esphome/components/cst226/binary_sensor/cs226_button.h +22 -0
- esphome/components/cst226/binary_sensor/cstt6_button.cpp +19 -0
- esphome/components/cst226/touchscreen/cst226_touchscreen.cpp +27 -5
- esphome/components/cst226/touchscreen/cst226_touchscreen.h +10 -10
- esphome/components/current_based/cover.py +37 -36
- esphome/components/current_based/current_based_cover.cpp +2 -1
- esphome/components/daikin/climate.py +2 -9
- esphome/components/daikin/daikin.cpp +15 -9
- esphome/components/daikin/daikin.h +5 -5
- esphome/components/daikin_arc/climate.py +2 -7
- esphome/components/daikin_brc/climate.py +3 -5
- esphome/components/dallas_temp/dallas_temp.cpp +17 -24
- esphome/components/dallas_temp/dallas_temp.h +0 -1
- esphome/components/daly_bms/daly_bms.cpp +2 -1
- esphome/components/debug/debug_component.cpp +6 -1
- esphome/components/debug/debug_component.h +8 -0
- esphome/components/debug/debug_esp32.cpp +109 -254
- esphome/components/debug/sensor.py +14 -0
- esphome/components/deep_sleep/deep_sleep_esp32.cpp +13 -1
- esphome/components/delonghi/climate.py +2 -9
- esphome/components/demo/__init__.py +18 -20
- esphome/components/dfrobot_sen0395/switch/__init__.py +21 -22
- esphome/components/display/rect.cpp +4 -9
- esphome/components/display/rect.h +1 -1
- esphome/components/dps310/sensor.py +6 -6
- esphome/components/ee895/sensor.py +9 -9
- esphome/components/emmeti/climate.py +2 -9
- esphome/components/endstop/cover.py +17 -16
- esphome/components/endstop/endstop_cover.cpp +2 -1
- esphome/components/ens160_base/__init__.py +12 -9
- esphome/components/esp32/__init__.py +60 -3
- esphome/components/esp32/core.cpp +11 -5
- esphome/components/esp32/gpio.cpp +86 -24
- esphome/components/esp32/gpio.py +15 -16
- esphome/components/esp32/gpio_esp32.py +1 -2
- esphome/components/esp32/gpio_esp32_c2.py +1 -1
- esphome/components/esp32/gpio_esp32_c3.py +1 -1
- esphome/components/esp32/gpio_esp32_c6.py +1 -1
- esphome/components/esp32/gpio_esp32_h2.py +1 -1
- esphome/components/esp32_ble/ble.cpp +1 -8
- esphome/components/esp32_ble/ble.h +5 -3
- esphome/components/esp32_ble/ble_advertising.cpp +2 -1
- esphome/components/esp32_ble/ble_advertising.h +1 -0
- esphome/components/esp32_ble_server/__init__.py +3 -0
- esphome/components/esp32_ble_tracker/__init__.py +7 -1
- esphome/components/esp32_ble_tracker/esp32_ble_tracker.cpp +192 -118
- esphome/components/esp32_ble_tracker/esp32_ble_tracker.h +29 -3
- esphome/components/esp32_camera/__init__.py +1 -1
- esphome/components/esp32_camera/esp32_camera.cpp +2 -10
- esphome/components/esp32_camera/esp32_camera.h +1 -1
- esphome/components/esp32_can/esp32_can.cpp +1 -1
- esphome/components/esp32_improv/esp32_improv_component.cpp +1 -1
- esphome/components/esp32_rmt_led_strip/led_strip.cpp +1 -1
- esphome/components/esp32_rmt_led_strip/led_strip.h +7 -5
- esphome/components/esp32_rmt_led_strip/light.py +9 -1
- esphome/components/esp32_touch/esp32_touch.cpp +1 -1
- esphome/components/esp8266/gpio.cpp +69 -8
- esphome/components/ethernet/ethernet_component.cpp +1 -1
- esphome/components/event/__init__.py +13 -10
- esphome/components/factory_reset/switch/__init__.py +7 -21
- esphome/components/fan/__init__.py +52 -5
- esphome/components/fastled_base/__init__.py +1 -4
- esphome/components/fastled_base/fastled_light.cpp +1 -1
- esphome/components/feedback/cover.py +38 -33
- esphome/components/feedback/feedback_cover.cpp +2 -1
- esphome/components/fujitsu_general/climate.py +2 -9
- esphome/components/gcja5/gcja5.cpp +2 -1
- esphome/components/gpio/one_wire/gpio_one_wire.cpp +45 -43
- esphome/components/gpio/one_wire/gpio_one_wire.h +2 -1
- esphome/components/gpio_expander/cached_gpio.h +22 -7
- esphome/components/gps/__init__.py +47 -17
- esphome/components/gps/gps.cpp +42 -23
- esphome/components/gps/gps.h +17 -13
- esphome/components/graph/__init__.py +1 -2
- esphome/components/gree/climate.py +4 -6
- esphome/components/gree/gree.cpp +16 -2
- esphome/components/gree/gree.h +2 -2
- esphome/components/growatt_solar/growatt_solar.cpp +2 -1
- esphome/components/haier/climate.py +37 -34
- esphome/components/hbridge/fan/__init__.py +19 -17
- esphome/components/he60r/cover.py +4 -5
- esphome/components/heatpumpir/climate.py +3 -6
- esphome/components/hitachi_ac344/climate.py +2 -9
- esphome/components/hitachi_ac424/climate.py +2 -9
- esphome/components/hlw8012/hlw8012.cpp +1 -1
- esphome/components/hm3301/hm3301.h +1 -1
- esphome/components/hte501/sensor.py +6 -6
- esphome/components/http_request/__init__.py +39 -6
- esphome/components/http_request/http_request.cpp +20 -0
- esphome/components/http_request/http_request.h +57 -15
- esphome/components/http_request/http_request_arduino.cpp +22 -6
- esphome/components/http_request/http_request_arduino.h +4 -3
- esphome/components/http_request/http_request_host.cpp +141 -0
- esphome/components/http_request/http_request_host.h +37 -0
- esphome/components/http_request/http_request_idf.cpp +35 -3
- esphome/components/http_request/http_request_idf.h +10 -3
- esphome/components/http_request/httplib.h +9691 -0
- esphome/components/http_request/update/__init__.py +11 -8
- esphome/components/hyt271/sensor.py +6 -6
- esphome/components/i2c/i2c.h +4 -0
- esphome/components/i2c/i2c_bus_esp_idf.cpp +1 -1
- esphome/components/i2s_audio/__init__.py +131 -22
- esphome/components/i2s_audio/i2s_audio.h +44 -4
- esphome/components/i2s_audio/media_player/__init__.py +19 -9
- esphome/components/i2s_audio/microphone/__init__.py +63 -5
- esphome/components/i2s_audio/microphone/i2s_audio_microphone.cpp +351 -61
- esphome/components/i2s_audio/microphone/i2s_audio_microphone.h +40 -6
- esphome/components/i2s_audio/speaker/__init__.py +31 -5
- esphome/components/i2s_audio/speaker/i2s_audio_speaker.cpp +155 -19
- esphome/components/i2s_audio/speaker/i2s_audio_speaker.h +17 -4
- esphome/components/ili9xxx/ili9xxx_init.h +1 -1
- esphome/components/image/__init__.py +37 -17
- esphome/components/image/image.cpp +25 -8
- esphome/components/internal_temperature/internal_temperature.cpp +6 -4
- esphome/components/key_collector/__init__.py +35 -0
- esphome/components/key_collector/key_collector.cpp +8 -0
- esphome/components/key_collector/key_collector.h +10 -0
- esphome/components/kuntze/kuntze.cpp +2 -1
- esphome/components/ld2410/ld2410.h +1 -1
- esphome/components/ld2450/ld2450.h +1 -1
- esphome/components/light/__init__.py +57 -0
- esphome/components/lock/__init__.py +51 -4
- esphome/components/lock/automation.h +2 -13
- esphome/components/logger/__init__.py +22 -0
- esphome/components/logger/logger.cpp +154 -103
- esphome/components/logger/logger.h +211 -36
- esphome/components/logger/task_log_buffer.cpp +138 -0
- esphome/components/logger/task_log_buffer.h +69 -0
- esphome/components/lvgl/__init__.py +13 -5
- esphome/components/lvgl/automation.py +50 -1
- esphome/components/lvgl/defines.py +0 -1
- esphome/components/lvgl/lv_validation.py +10 -1
- esphome/components/lvgl/lvgl_esphome.cpp +5 -1
- esphome/components/lvgl/schemas.py +14 -14
- esphome/components/lvgl/text/__init__.py +1 -2
- esphome/components/lvgl/widgets/arc.py +7 -6
- esphome/components/lvgl/widgets/buttonmatrix.py +3 -3
- esphome/components/lvgl/widgets/checkbox.py +2 -2
- esphome/components/lvgl/widgets/dropdown.py +2 -1
- esphome/components/lvgl/widgets/img.py +15 -12
- esphome/components/mapping/__init__.py +134 -0
- esphome/components/matrix_keypad/matrix_keypad.cpp +2 -1
- esphome/components/max7219digit/max7219digit.cpp +28 -27
- esphome/components/mdns/__init__.py +11 -5
- esphome/components/mdns/mdns_component.cpp +11 -5
- esphome/components/mdns/mdns_component.h +3 -2
- esphome/components/mdns/mdns_esp32.cpp +4 -3
- esphome/components/mdns/mdns_esp8266.cpp +4 -2
- esphome/components/mdns/mdns_libretiny.cpp +4 -2
- esphome/components/mdns/mdns_rp2040.cpp +4 -2
- esphome/components/media_player/__init__.py +40 -6
- esphome/components/mhz19/sensor.py +11 -7
- esphome/components/micro_wake_word/__init__.py +99 -31
- esphome/components/micro_wake_word/automation.h +54 -0
- esphome/components/micro_wake_word/micro_wake_word.cpp +331 -319
- esphome/components/micro_wake_word/micro_wake_word.h +58 -105
- esphome/components/micro_wake_word/preprocessor_settings.h +19 -2
- esphome/components/micro_wake_word/streaming_model.cpp +158 -41
- esphome/components/micro_wake_word/streaming_model.h +85 -13
- esphome/components/microphone/__init__.py +139 -9
- esphome/components/microphone/automation.h +14 -2
- esphome/components/microphone/microphone.cpp +21 -0
- esphome/components/microphone/microphone.h +14 -5
- esphome/components/microphone/microphone_source.cpp +95 -0
- esphome/components/microphone/microphone_source.h +80 -0
- esphome/components/mics_4514/sensor.py +25 -14
- esphome/components/midea/climate.py +3 -4
- esphome/components/midea_ir/climate.py +3 -5
- esphome/components/mipi_spi/__init__.py +15 -0
- esphome/components/mipi_spi/display.py +474 -0
- esphome/components/mipi_spi/mipi_spi.cpp +481 -0
- esphome/components/mipi_spi/mipi_spi.h +171 -0
- esphome/components/mipi_spi/models/__init__.py +65 -0
- esphome/components/mipi_spi/models/amoled.py +72 -0
- esphome/components/mipi_spi/models/commands.py +82 -0
- esphome/components/mipi_spi/models/cyd.py +10 -0
- esphome/components/mipi_spi/models/ili.py +749 -0
- esphome/components/mipi_spi/models/jc.py +260 -0
- esphome/components/mipi_spi/models/lanbon.py +15 -0
- esphome/components/mipi_spi/models/lilygo.py +60 -0
- esphome/components/mipi_spi/models/waveshare.py +139 -0
- esphome/components/mitsubishi/climate.py +2 -5
- esphome/components/mitsubishi/mitsubishi.cpp +9 -9
- esphome/components/mixer/speaker/mixer_speaker.cpp +12 -22
- esphome/components/mixer/speaker/mixer_speaker.h +1 -3
- esphome/components/mlx90393/sensor.py +5 -0
- esphome/components/mlx90393/sensor_mlx90393.cpp +195 -13
- esphome/components/mlx90393/sensor_mlx90393.h +21 -4
- esphome/components/modbus/modbus.cpp +2 -1
- esphome/components/mqtt/__init__.py +1 -1
- esphome/components/mqtt/mqtt_client.cpp +6 -2
- esphome/components/mqtt/mqtt_const.h +4 -0
- esphome/components/mqtt/mqtt_fan.cpp +39 -0
- esphome/components/mqtt/mqtt_fan.h +2 -0
- esphome/components/ms5611/sensor.py +6 -6
- esphome/components/ms8607/sensor.py +3 -3
- esphome/components/network/__init__.py +1 -1
- esphome/components/nextion/base_component.py +17 -16
- esphome/components/nextion/display.py +11 -2
- esphome/components/nextion/nextion.cpp +39 -1
- esphome/components/nextion/nextion.h +50 -0
- esphome/components/noblex/climate.py +2 -9
- esphome/components/number/__init__.py +12 -9
- esphome/components/one_wire/one_wire_bus.cpp +14 -10
- esphome/components/one_wire/one_wire_bus.h +14 -8
- esphome/components/online_image/bmp_image.cpp +48 -11
- esphome/components/online_image/bmp_image.h +2 -0
- esphome/components/opentherm/binary_sensor/__init__.py +2 -4
- esphome/components/opentherm/number/__init__.py +11 -20
- esphome/components/opentherm/sensor/__init__.py +3 -3
- esphome/components/opentherm/switch/__init__.py +3 -5
- esphome/components/output/lock/__init__.py +11 -9
- esphome/components/packages/__init__.py +33 -31
- esphome/components/packet_transport/__init__.py +201 -0
- esphome/components/packet_transport/binary_sensor.py +19 -0
- esphome/components/packet_transport/packet_transport.cpp +534 -0
- esphome/components/packet_transport/packet_transport.h +154 -0
- esphome/components/packet_transport/sensor.py +19 -0
- esphome/components/pca9685/pca9685_output.cpp +2 -1
- esphome/components/pid/climate.py +2 -4
- esphome/components/pm2005/__init__.py +1 -0
- esphome/components/pm2005/pm2005.cpp +123 -0
- esphome/components/pm2005/pm2005.h +46 -0
- esphome/components/pm2005/sensor.py +86 -0
- esphome/components/pmsa003i/pmsa003i.cpp +43 -16
- esphome/components/pmsa003i/pmsa003i.h +25 -25
- esphome/components/pmsx003/pmsx003.cpp +195 -230
- esphome/components/pmsx003/pmsx003.h +51 -33
- esphome/components/pmsx003/sensor.py +21 -11
- esphome/components/pn7150/pn7150.h +2 -2
- esphome/components/pn7160/pn7160.h +2 -2
- esphome/components/prometheus/prometheus_handler.cpp +174 -0
- esphome/components/prometheus/prometheus_handler.h +17 -0
- esphome/components/psram/__init__.py +7 -5
- esphome/components/pulse_meter/pulse_meter_sensor.cpp +32 -12
- esphome/components/pulse_meter/pulse_meter_sensor.h +5 -5
- esphome/components/pzem004t/pzem004t.cpp +2 -1
- esphome/components/qspi_dbi/__init__.py +0 -1
- esphome/components/qspi_dbi/display.py +2 -1
- esphome/components/qspi_dbi/models.py +1 -2
- esphome/components/remote_base/__init__.py +91 -0
- esphome/components/remote_base/beo4_protocol.cpp +153 -0
- esphome/components/remote_base/beo4_protocol.h +43 -0
- esphome/components/remote_base/gobox_protocol.cpp +131 -0
- esphome/components/remote_base/gobox_protocol.h +54 -0
- esphome/components/remote_receiver/remote_receiver_esp32.cpp +16 -9
- esphome/components/resampler/speaker/resampler_speaker.cpp +12 -10
- esphome/components/resampler/speaker/resampler_speaker.h +1 -1
- esphome/components/rf_bridge/rf_bridge.cpp +2 -1
- esphome/components/scd30/sensor.py +2 -3
- esphome/components/scd4x/sensor.py +4 -5
- esphome/components/sdp3x/sensor.py +2 -1
- esphome/components/sds011/sds011.cpp +2 -1
- esphome/components/select/__init__.py +19 -20
- esphome/components/sen5x/sen5x.cpp +55 -36
- esphome/components/sen5x/sensor.py +1 -1
- esphome/components/senseair/sensor.py +3 -3
- esphome/components/sensor/__init__.py +158 -14
- esphome/components/sensor/filter.cpp +23 -0
- esphome/components/sensor/filter.h +22 -0
- esphome/components/sgp30/sensor.py +14 -16
- esphome/components/sgp4x/sensor.py +1 -1
- esphome/components/sht4x/sht4x.cpp +43 -22
- esphome/components/sht4x/sht4x.h +1 -1
- esphome/components/shtcx/sensor.py +6 -6
- esphome/components/slow_pwm/slow_pwm_output.cpp +2 -1
- esphome/components/sml/text_sensor/__init__.py +4 -6
- esphome/components/sound_level/__init__.py +0 -0
- esphome/components/sound_level/sensor.py +97 -0
- esphome/components/sound_level/sound_level.cpp +194 -0
- esphome/components/sound_level/sound_level.h +73 -0
- esphome/components/speaker/media_player/__init__.py +4 -8
- esphome/components/speaker/media_player/speaker_media_player.cpp +0 -18
- esphome/components/speaker/media_player/speaker_media_player.h +0 -11
- esphome/components/speaker/speaker.h +4 -7
- esphome/components/speed/fan/__init__.py +17 -16
- esphome/components/spi/spi.h +11 -1
- esphome/components/sprinkler/__init__.py +18 -19
- esphome/components/sprinkler/sprinkler.cpp +6 -5
- esphome/components/switch/__init__.py +32 -42
- esphome/components/syslog/__init__.py +41 -0
- esphome/components/syslog/esphome_syslog.cpp +49 -0
- esphome/components/syslog/esphome_syslog.h +27 -0
- esphome/components/t6615/sensor.py +3 -3
- esphome/components/t6615/t6615.cpp +2 -1
- esphome/components/tca9555/tca9555.cpp +11 -6
- esphome/components/tcl112/climate.py +2 -9
- esphome/components/template/alarm_control_panel/__init__.py +7 -6
- esphome/components/template/alarm_control_panel/template_alarm_control_panel.cpp +21 -17
- esphome/components/template/alarm_control_panel/template_alarm_control_panel.h +2 -1
- esphome/components/template/cover/__init__.py +27 -21
- esphome/components/template/fan/__init__.py +14 -12
- esphome/components/template/lock/__init__.py +20 -25
- esphome/components/template/lock/automation.h +18 -0
- esphome/components/template/text/__init__.py +4 -3
- esphome/components/template/valve/__init__.py +32 -21
- esphome/components/template/valve/automation.h +24 -0
- esphome/components/text/__init__.py +32 -1
- esphome/components/text_sensor/__init__.py +24 -29
- esphome/components/thermostat/climate.py +5 -5
- esphome/components/time_based/cover.py +17 -16
- esphome/components/time_based/time_based_cover.cpp +2 -1
- esphome/components/tm1638/switch/__init__.py +10 -7
- esphome/components/tormatic/cover.py +4 -5
- esphome/components/toshiba/climate.py +3 -5
- esphome/components/touchscreen/touchscreen.cpp +3 -1
- esphome/components/tt21100/touchscreen/tt21100.cpp +1 -1
- esphome/components/tuya/climate/__init__.py +5 -6
- esphome/components/tuya/cover/__init__.py +6 -11
- esphome/components/tuya/select/__init__.py +15 -5
- esphome/components/tuya/select/tuya_select.cpp +6 -1
- esphome/components/tuya/select/tuya_select.h +5 -1
- esphome/components/uart/packet_transport/__init__.py +20 -0
- esphome/components/uart/packet_transport/uart_transport.cpp +88 -0
- esphome/components/uart/packet_transport/uart_transport.h +41 -0
- esphome/components/uart/switch/uart_switch.cpp +2 -1
- esphome/components/udp/__init__.py +126 -128
- esphome/components/udp/automation.h +40 -0
- esphome/components/udp/binary_sensor.py +3 -25
- esphome/components/udp/packet_transport/__init__.py +29 -0
- esphome/components/udp/packet_transport/udp_transport.cpp +36 -0
- esphome/components/udp/packet_transport/udp_transport.h +28 -0
- esphome/components/udp/sensor.py +3 -25
- esphome/components/udp/udp_component.cpp +26 -470
- esphome/components/udp/udp_component.h +21 -128
- esphome/components/update/__init__.py +31 -1
- esphome/components/uponor_smatrix/climate/__init__.py +4 -9
- esphome/components/uponor_smatrix/climate/uponor_smatrix_climate.cpp +2 -1
- esphome/components/uponor_smatrix/uponor_smatrix.cpp +2 -1
- esphome/components/uptime/text_sensor/__init__.py +47 -7
- esphome/components/uptime/text_sensor/uptime_text_sensor.cpp +12 -7
- esphome/components/uptime/text_sensor/uptime_text_sensor.h +19 -0
- esphome/components/valve/__init__.py +34 -3
- esphome/components/valve/automation.h +1 -19
- esphome/components/vl53l0x/sensor.py +11 -0
- esphome/components/vl53l0x/vl53l0x_sensor.cpp +5 -1
- esphome/components/vl53l0x/vl53l0x_sensor.h +2 -1
- esphome/components/voice_assistant/__init__.py +36 -10
- esphome/components/voice_assistant/voice_assistant.cpp +170 -144
- esphome/components/voice_assistant/voice_assistant.h +26 -25
- esphome/components/waveshare_epaper/display.py +6 -0
- esphome/components/waveshare_epaper/waveshare_epaper.cpp +439 -37
- esphome/components/waveshare_epaper/waveshare_epaper.h +60 -11
- esphome/components/weikai/weikai.cpp +0 -52
- esphome/components/whirlpool/climate.py +3 -5
- esphome/components/whynter/climate.py +3 -5
- esphome/components/xpt2046/touchscreen/xpt2046.cpp +1 -1
- esphome/components/yashima/climate.py +6 -6
- esphome/components/zhlt01/climate.py +2 -7
- esphome/config.py +13 -13
- esphome/config_validation.py +38 -58
- esphome/const.py +15 -1
- esphome/core/__init__.py +2 -0
- esphome/core/application.cpp +27 -10
- esphome/core/application.h +9 -1
- esphome/core/automation.h +4 -3
- esphome/core/component.cpp +28 -7
- esphome/core/component.h +10 -1
- esphome/core/defines.h +23 -17
- esphome/core/doxygen.h +13 -0
- esphome/core/macros.h +4 -0
- esphome/core/scheduler.cpp +7 -1
- esphome/cpp_generator.py +6 -2
- esphome/dashboard/web_server.py +3 -3
- esphome/helpers.py +39 -0
- esphome/loader.py +4 -0
- esphome/log.py +15 -19
- esphome/mqtt.py +23 -10
- esphome/platformio_api.py +1 -1
- esphome/schema_extractors.py +0 -1
- esphome/voluptuous_schema.py +3 -1
- esphome/vscode.py +15 -0
- esphome/wizard.py +47 -37
- esphome/zeroconf.py +7 -3
- {esphome-2025.4.1.dist-info → esphome-2025.5.0.dist-info}/METADATA +10 -11
- {esphome-2025.4.1.dist-info → esphome-2025.5.0.dist-info}/RECORD +456 -396
- {esphome-2025.4.1.dist-info → esphome-2025.5.0.dist-info}/WHEEL +1 -1
- esphome/components/esp32_ble/const_esp32c6.h +0 -74
- {esphome-2025.4.1.dist-info → esphome-2025.5.0.dist-info}/entry_points.txt +0 -0
- {esphome-2025.4.1.dist-info → esphome-2025.5.0.dist-info}/licenses/LICENSE +0 -0
- {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
|
-
|
29
|
-
|
30
|
-
|
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
|
-
//
|
111
|
-
|
112
|
-
|
113
|
-
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
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:
|
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
|
|