mirror of
				https://github.com/esphome/esphome.git
				synced 2025-10-25 21:23:53 +01:00 
			
		
		
		
	Merge branch 'dev' into idf_webserver_ota
This commit is contained in:
		| @@ -136,23 +136,26 @@ async def to_code(config): | ||||
|     cg.add(var.set_reboot_timeout(config[CONF_REBOOT_TIMEOUT])) | ||||
|     cg.add(var.set_batch_delay(config[CONF_BATCH_DELAY])) | ||||
|  | ||||
|     for conf in config.get(CONF_ACTIONS, []): | ||||
|         template_args = [] | ||||
|         func_args = [] | ||||
|         service_arg_names = [] | ||||
|         for name, var_ in conf[CONF_VARIABLES].items(): | ||||
|             native = SERVICE_ARG_NATIVE_TYPES[var_] | ||||
|             template_args.append(native) | ||||
|             func_args.append((native, name)) | ||||
|             service_arg_names.append(name) | ||||
|         templ = cg.TemplateArguments(*template_args) | ||||
|         trigger = cg.new_Pvariable( | ||||
|             conf[CONF_TRIGGER_ID], templ, conf[CONF_ACTION], service_arg_names | ||||
|         ) | ||||
|         cg.add(var.register_user_service(trigger)) | ||||
|         await automation.build_automation(trigger, func_args, conf) | ||||
|     if actions := config.get(CONF_ACTIONS, []): | ||||
|         cg.add_define("USE_API_YAML_SERVICES") | ||||
|         for conf in actions: | ||||
|             template_args = [] | ||||
|             func_args = [] | ||||
|             service_arg_names = [] | ||||
|             for name, var_ in conf[CONF_VARIABLES].items(): | ||||
|                 native = SERVICE_ARG_NATIVE_TYPES[var_] | ||||
|                 template_args.append(native) | ||||
|                 func_args.append((native, name)) | ||||
|                 service_arg_names.append(name) | ||||
|             templ = cg.TemplateArguments(*template_args) | ||||
|             trigger = cg.new_Pvariable( | ||||
|                 conf[CONF_TRIGGER_ID], templ, conf[CONF_ACTION], service_arg_names | ||||
|             ) | ||||
|             cg.add(var.register_user_service(trigger)) | ||||
|             await automation.build_automation(trigger, func_args, conf) | ||||
|  | ||||
|     if CONF_ON_CLIENT_CONNECTED in config: | ||||
|         cg.add_define("USE_API_CLIENT_CONNECTED_TRIGGER") | ||||
|         await automation.build_automation( | ||||
|             var.get_client_connected_trigger(), | ||||
|             [(cg.std_string, "client_info"), (cg.std_string, "client_address")], | ||||
| @@ -160,6 +163,7 @@ async def to_code(config): | ||||
|         ) | ||||
|  | ||||
|     if CONF_ON_CLIENT_DISCONNECTED in config: | ||||
|         cg.add_define("USE_API_CLIENT_DISCONNECTED_TRIGGER") | ||||
|         await automation.build_automation( | ||||
|             var.get_client_disconnected_trigger(), | ||||
|             [(cg.std_string, "client_info"), (cg.std_string, "client_address")], | ||||
|   | ||||
| @@ -1511,7 +1511,9 @@ ConnectResponse APIConnection::connect(const ConnectRequest &msg) { | ||||
|   if (correct) { | ||||
|     ESP_LOGD(TAG, "%s connected", this->get_client_combined_info().c_str()); | ||||
|     this->flags_.connection_state = static_cast<uint8_t>(ConnectionState::AUTHENTICATED); | ||||
| #ifdef USE_API_CLIENT_CONNECTED_TRIGGER | ||||
|     this->parent_->get_client_connected_trigger()->trigger(this->client_info_, this->client_peername_); | ||||
| #endif | ||||
| #ifdef USE_HOMEASSISTANT_TIME | ||||
|     if (homeassistant::global_homeassistant_time != nullptr) { | ||||
|       this->send_time_request(); | ||||
|   | ||||
| @@ -184,7 +184,9 @@ void APIServer::loop() { | ||||
|     } | ||||
|  | ||||
|     // Rare case: handle disconnection | ||||
| #ifdef USE_API_CLIENT_DISCONNECTED_TRIGGER | ||||
|     this->client_disconnected_trigger_->trigger(client->client_info_, client->client_peername_); | ||||
| #endif | ||||
|     ESP_LOGV(TAG, "Remove connection %s", client->client_info_.c_str()); | ||||
|  | ||||
|     // Swap with the last element and pop (avoids expensive vector shifts) | ||||
|   | ||||
| @@ -105,7 +105,18 @@ class APIServer : public Component, public Controller { | ||||
|   void on_media_player_update(media_player::MediaPlayer *obj) override; | ||||
| #endif | ||||
|   void send_homeassistant_service_call(const HomeassistantServiceResponse &call); | ||||
|   void register_user_service(UserServiceDescriptor *descriptor) { this->user_services_.push_back(descriptor); } | ||||
|   void register_user_service(UserServiceDescriptor *descriptor) { | ||||
| #ifdef USE_API_YAML_SERVICES | ||||
|     // Vector is pre-allocated when services are defined in YAML | ||||
|     this->user_services_.push_back(descriptor); | ||||
| #else | ||||
|     // Lazy allocate vector on first use for CustomAPIDevice | ||||
|     if (!this->user_services_) { | ||||
|       this->user_services_ = std::make_unique<std::vector<UserServiceDescriptor *>>(); | ||||
|     } | ||||
|     this->user_services_->push_back(descriptor); | ||||
| #endif | ||||
|   } | ||||
| #ifdef USE_HOMEASSISTANT_TIME | ||||
|   void request_time(); | ||||
| #endif | ||||
| @@ -134,19 +145,34 @@ class APIServer : public Component, public Controller { | ||||
|   void get_home_assistant_state(std::string entity_id, optional<std::string> attribute, | ||||
|                                 std::function<void(std::string)> f); | ||||
|   const std::vector<HomeAssistantStateSubscription> &get_state_subs() const; | ||||
|   const std::vector<UserServiceDescriptor *> &get_user_services() const { return this->user_services_; } | ||||
|   const std::vector<UserServiceDescriptor *> &get_user_services() const { | ||||
| #ifdef USE_API_YAML_SERVICES | ||||
|     return this->user_services_; | ||||
| #else | ||||
|     static const std::vector<UserServiceDescriptor *> EMPTY; | ||||
|     return this->user_services_ ? *this->user_services_ : EMPTY; | ||||
| #endif | ||||
|   } | ||||
|  | ||||
| #ifdef USE_API_CLIENT_CONNECTED_TRIGGER | ||||
|   Trigger<std::string, std::string> *get_client_connected_trigger() const { return this->client_connected_trigger_; } | ||||
| #endif | ||||
| #ifdef USE_API_CLIENT_DISCONNECTED_TRIGGER | ||||
|   Trigger<std::string, std::string> *get_client_disconnected_trigger() const { | ||||
|     return this->client_disconnected_trigger_; | ||||
|   } | ||||
| #endif | ||||
|  | ||||
|  protected: | ||||
|   void schedule_reboot_timeout_(); | ||||
|   // Pointers and pointer-like types first (4 bytes each) | ||||
|   std::unique_ptr<socket::Socket> socket_ = nullptr; | ||||
| #ifdef USE_API_CLIENT_CONNECTED_TRIGGER | ||||
|   Trigger<std::string, std::string> *client_connected_trigger_ = new Trigger<std::string, std::string>(); | ||||
| #endif | ||||
| #ifdef USE_API_CLIENT_DISCONNECTED_TRIGGER | ||||
|   Trigger<std::string, std::string> *client_disconnected_trigger_ = new Trigger<std::string, std::string>(); | ||||
| #endif | ||||
|  | ||||
|   // 4-byte aligned types | ||||
|   uint32_t reboot_timeout_{300000}; | ||||
| @@ -156,7 +182,15 @@ class APIServer : public Component, public Controller { | ||||
|   std::string password_; | ||||
|   std::vector<uint8_t> shared_write_buffer_;  // Shared proto write buffer for all connections | ||||
|   std::vector<HomeAssistantStateSubscription> state_subs_; | ||||
| #ifdef USE_API_YAML_SERVICES | ||||
|   // When services are defined in YAML, we know at compile time that services will be registered | ||||
|   std::vector<UserServiceDescriptor *> user_services_; | ||||
| #else | ||||
|   // Services can still be registered at runtime by CustomAPIDevice components even when not | ||||
|   // defined in YAML. Using unique_ptr allows lazy allocation, saving 12 bytes in the common | ||||
|   // case where no services (YAML or custom) are used. | ||||
|   std::unique_ptr<std::vector<UserServiceDescriptor *>> user_services_; | ||||
| #endif | ||||
|  | ||||
|   // Group smaller types together | ||||
|   uint16_t port_{6053}; | ||||
|   | ||||
| @@ -60,10 +60,18 @@ void Component::set_interval(const std::string &name, uint32_t interval, std::fu | ||||
|   App.scheduler.set_interval(this, name, interval, std::move(f)); | ||||
| } | ||||
|  | ||||
| void Component::set_interval(const char *name, uint32_t interval, std::function<void()> &&f) {  // NOLINT | ||||
|   App.scheduler.set_interval(this, name, interval, std::move(f)); | ||||
| } | ||||
|  | ||||
| bool Component::cancel_interval(const std::string &name) {  // NOLINT | ||||
|   return App.scheduler.cancel_interval(this, name); | ||||
| } | ||||
|  | ||||
| bool Component::cancel_interval(const char *name) {  // NOLINT | ||||
|   return App.scheduler.cancel_interval(this, name); | ||||
| } | ||||
|  | ||||
| void Component::set_retry(const std::string &name, uint32_t initial_wait_time, uint8_t max_attempts, | ||||
|                           std::function<RetryResult(uint8_t)> &&f, float backoff_increase_factor) {  // NOLINT | ||||
|   App.scheduler.set_retry(this, name, initial_wait_time, max_attempts, std::move(f), backoff_increase_factor); | ||||
| @@ -77,10 +85,18 @@ void Component::set_timeout(const std::string &name, uint32_t timeout, std::func | ||||
|   App.scheduler.set_timeout(this, name, timeout, std::move(f)); | ||||
| } | ||||
|  | ||||
| void Component::set_timeout(const char *name, uint32_t timeout, std::function<void()> &&f) {  // NOLINT | ||||
|   App.scheduler.set_timeout(this, name, timeout, std::move(f)); | ||||
| } | ||||
|  | ||||
| bool Component::cancel_timeout(const std::string &name) {  // NOLINT | ||||
|   return App.scheduler.cancel_timeout(this, name); | ||||
| } | ||||
|  | ||||
| bool Component::cancel_timeout(const char *name) {  // NOLINT | ||||
|   return App.scheduler.cancel_timeout(this, name); | ||||
| } | ||||
|  | ||||
| void Component::call_loop() { this->loop(); } | ||||
| void Component::call_setup() { this->setup(); } | ||||
| void Component::call_dump_config() { | ||||
| @@ -189,7 +205,7 @@ bool Component::is_in_loop_state() const { | ||||
|   return (this->component_state_ & COMPONENT_STATE_MASK) == COMPONENT_STATE_LOOP; | ||||
| } | ||||
| void Component::defer(std::function<void()> &&f) {  // NOLINT | ||||
|   App.scheduler.set_timeout(this, "", 0, std::move(f)); | ||||
|   App.scheduler.set_timeout(this, static_cast<const char *>(nullptr), 0, std::move(f)); | ||||
| } | ||||
| bool Component::cancel_defer(const std::string &name) {  // NOLINT | ||||
|   return App.scheduler.cancel_timeout(this, name); | ||||
|   | ||||
| @@ -260,6 +260,22 @@ class Component { | ||||
|    */ | ||||
|   void set_interval(const std::string &name, uint32_t interval, std::function<void()> &&f);  // NOLINT | ||||
|  | ||||
|   /** Set an interval function with a const char* name. | ||||
|    * | ||||
|    * IMPORTANT: The provided name pointer must remain valid for the lifetime of the scheduler item. | ||||
|    * This means the name should be: | ||||
|    *   - A string literal (e.g., "update") | ||||
|    *   - A static const char* variable | ||||
|    *   - A pointer with lifetime >= the scheduled task | ||||
|    * | ||||
|    * For dynamic strings, use the std::string overload instead. | ||||
|    * | ||||
|    * @param name The identifier for this interval function (must have static lifetime) | ||||
|    * @param interval The interval in ms | ||||
|    * @param f The function to call | ||||
|    */ | ||||
|   void set_interval(const char *name, uint32_t interval, std::function<void()> &&f);  // NOLINT | ||||
|  | ||||
|   void set_interval(uint32_t interval, std::function<void()> &&f);  // NOLINT | ||||
|  | ||||
|   /** Cancel an interval function. | ||||
| @@ -268,6 +284,7 @@ class Component { | ||||
|    * @return Whether an interval functions was deleted. | ||||
|    */ | ||||
|   bool cancel_interval(const std::string &name);  // NOLINT | ||||
|   bool cancel_interval(const char *name);         // NOLINT | ||||
|  | ||||
|   /** Set an retry function with a unique name. Empty name means no cancelling possible. | ||||
|    * | ||||
| @@ -328,6 +345,22 @@ class Component { | ||||
|    */ | ||||
|   void set_timeout(const std::string &name, uint32_t timeout, std::function<void()> &&f);  // NOLINT | ||||
|  | ||||
|   /** Set a timeout function with a const char* name. | ||||
|    * | ||||
|    * IMPORTANT: The provided name pointer must remain valid for the lifetime of the scheduler item. | ||||
|    * This means the name should be: | ||||
|    *   - A string literal (e.g., "init") | ||||
|    *   - A static const char* variable | ||||
|    *   - A pointer with lifetime >= the timeout duration | ||||
|    * | ||||
|    * For dynamic strings, use the std::string overload instead. | ||||
|    * | ||||
|    * @param name The identifier for this timeout function (must have static lifetime) | ||||
|    * @param timeout The timeout in ms | ||||
|    * @param f The function to call | ||||
|    */ | ||||
|   void set_timeout(const char *name, uint32_t timeout, std::function<void()> &&f);  // NOLINT | ||||
|  | ||||
|   void set_timeout(uint32_t timeout, std::function<void()> &&f);  // NOLINT | ||||
|  | ||||
|   /** Cancel a timeout function. | ||||
| @@ -336,6 +369,7 @@ class Component { | ||||
|    * @return Whether a timeout functions was deleted. | ||||
|    */ | ||||
|   bool cancel_timeout(const std::string &name);  // NOLINT | ||||
|   bool cancel_timeout(const char *name);         // NOLINT | ||||
|  | ||||
|   /** Defer a callback to the next loop() call. | ||||
|    * | ||||
|   | ||||
| @@ -101,8 +101,11 @@ | ||||
| #define USE_AUDIO_FLAC_SUPPORT | ||||
| #define USE_AUDIO_MP3_SUPPORT | ||||
| #define USE_API | ||||
| #define USE_API_CLIENT_CONNECTED_TRIGGER | ||||
| #define USE_API_CLIENT_DISCONNECTED_TRIGGER | ||||
| #define USE_API_NOISE | ||||
| #define USE_API_PLAINTEXT | ||||
| #define USE_API_YAML_SERVICES | ||||
| #define USE_MD5 | ||||
| #define USE_MQTT | ||||
| #define USE_NETWORK | ||||
|   | ||||
| @@ -7,6 +7,7 @@ | ||||
| #include "esphome/core/log.h" | ||||
| #include <algorithm> | ||||
| #include <cinttypes> | ||||
| #include <cstring> | ||||
|  | ||||
| namespace esphome { | ||||
|  | ||||
| @@ -17,75 +18,138 @@ static const uint32_t MAX_LOGICALLY_DELETED_ITEMS = 10; | ||||
| // Uncomment to debug scheduler | ||||
| // #define ESPHOME_DEBUG_SCHEDULER | ||||
|  | ||||
| #ifdef ESPHOME_DEBUG_SCHEDULER | ||||
| // Helper to validate that a pointer looks like it's in static memory | ||||
| static void validate_static_string(const char *name) { | ||||
|   if (name == nullptr) | ||||
|     return; | ||||
|  | ||||
|   // This is a heuristic check - stack and heap pointers are typically | ||||
|   // much higher in memory than static data | ||||
|   uintptr_t addr = reinterpret_cast<uintptr_t>(name); | ||||
|  | ||||
|   // Create a stack variable to compare against | ||||
|   int stack_var; | ||||
|   uintptr_t stack_addr = reinterpret_cast<uintptr_t>(&stack_var); | ||||
|  | ||||
|   // If the string pointer is near our stack variable, it's likely on the stack | ||||
|   // Using 8KB range as ESP32 main task stack is typically 8192 bytes | ||||
|   if (addr > (stack_addr - 0x2000) && addr < (stack_addr + 0x2000)) { | ||||
|     ESP_LOGW(TAG, | ||||
|              "WARNING: Scheduler name '%s' at %p appears to be on the stack - this is unsafe!\n" | ||||
|              "         Stack reference at %p", | ||||
|              name, name, &stack_var); | ||||
|   } | ||||
|  | ||||
|   // Also check if it might be on the heap by seeing if it's in a very different range | ||||
|   // This is platform-specific but generally heap is allocated far from static memory | ||||
|   static const char *static_str = "test"; | ||||
|   uintptr_t static_addr = reinterpret_cast<uintptr_t>(static_str); | ||||
|  | ||||
|   // If the address is very far from known static memory, it might be heap | ||||
|   if (addr > static_addr + 0x100000 || (static_addr > 0x100000 && addr < static_addr - 0x100000)) { | ||||
|     ESP_LOGW(TAG, "WARNING: Scheduler name '%s' at %p might be on heap (static ref at %p)", name, name, static_str); | ||||
|   } | ||||
| } | ||||
| #endif | ||||
|  | ||||
| // A note on locking: the `lock_` lock protects the `items_` and `to_add_` containers. It must be taken when writing to | ||||
| // them (i.e. when adding/removing items, but not when changing items). As items are only deleted from the loop task, | ||||
| // iterating over them from the loop task is fine; but iterating from any other context requires the lock to be held to | ||||
| // avoid the main thread modifying the list while it is being accessed. | ||||
|  | ||||
| void HOT Scheduler::set_timeout(Component *component, const std::string &name, uint32_t timeout, | ||||
|                                 std::function<void()> func) { | ||||
|   const auto now = this->millis_(); | ||||
| // Common implementation for both timeout and interval | ||||
| void HOT Scheduler::set_timer_common_(Component *component, SchedulerItem::Type type, bool is_static_string, | ||||
|                                       const void *name_ptr, uint32_t delay, std::function<void()> func) { | ||||
|   // Get the name as const char* | ||||
|   const char *name_cstr = | ||||
|       is_static_string ? static_cast<const char *>(name_ptr) : static_cast<const std::string *>(name_ptr)->c_str(); | ||||
|  | ||||
|   if (!name.empty()) | ||||
|     this->cancel_timeout(component, name); | ||||
|   // Cancel existing timer if name is not empty | ||||
|   if (name_cstr != nullptr && name_cstr[0] != '\0') { | ||||
|     this->cancel_item_(component, name_cstr, type); | ||||
|   } | ||||
|  | ||||
|   if (timeout == SCHEDULER_DONT_RUN) | ||||
|   if (delay == SCHEDULER_DONT_RUN) | ||||
|     return; | ||||
|  | ||||
|   const auto now = this->millis_(); | ||||
|  | ||||
|   // Create and populate the scheduler item | ||||
|   auto item = make_unique<SchedulerItem>(); | ||||
|   item->component = component; | ||||
|   item->name = name; | ||||
|   item->type = SchedulerItem::TIMEOUT; | ||||
|   item->next_execution_ = now + timeout; | ||||
|   item->set_name(name_cstr, !is_static_string); | ||||
|   item->type = type; | ||||
|   item->callback = std::move(func); | ||||
|   item->remove = false; | ||||
|  | ||||
|   // Type-specific setup | ||||
|   if (type == SchedulerItem::INTERVAL) { | ||||
|     item->interval = delay; | ||||
|     // Calculate random offset (0 to interval/2) | ||||
|     uint32_t offset = (delay != 0) ? (random_uint32() % delay) / 2 : 0; | ||||
|     item->next_execution_ = now + offset; | ||||
|   } else { | ||||
|     item->interval = 0; | ||||
|     item->next_execution_ = now + delay; | ||||
|   } | ||||
|  | ||||
| #ifdef ESPHOME_DEBUG_SCHEDULER | ||||
|   ESP_LOGD(TAG, "set_timeout(name='%s/%s', timeout=%" PRIu32 ")", item->get_source(), name.c_str(), timeout); | ||||
|   // Validate static strings in debug mode | ||||
|   if (is_static_string && name_cstr != nullptr) { | ||||
|     validate_static_string(name_cstr); | ||||
|   } | ||||
|  | ||||
|   // Debug logging | ||||
|   const char *type_str = (type == SchedulerItem::TIMEOUT) ? "timeout" : "interval"; | ||||
|   if (type == SchedulerItem::TIMEOUT) { | ||||
|     ESP_LOGD(TAG, "set_%s(name='%s/%s', %s=%" PRIu32 ")", type_str, item->get_source(), | ||||
|              name_cstr ? name_cstr : "(null)", type_str, delay); | ||||
|   } else { | ||||
|     ESP_LOGD(TAG, "set_%s(name='%s/%s', %s=%" PRIu32 ", offset=%" PRIu32 ")", type_str, item->get_source(), | ||||
|              name_cstr ? name_cstr : "(null)", type_str, delay, static_cast<uint32_t>(item->next_execution_ - now)); | ||||
|   } | ||||
| #endif | ||||
|  | ||||
|   this->push_(std::move(item)); | ||||
| } | ||||
|  | ||||
| void HOT Scheduler::set_timeout(Component *component, const char *name, uint32_t timeout, std::function<void()> func) { | ||||
|   this->set_timer_common_(component, SchedulerItem::TIMEOUT, true, name, timeout, std::move(func)); | ||||
| } | ||||
|  | ||||
| void HOT Scheduler::set_timeout(Component *component, const std::string &name, uint32_t timeout, | ||||
|                                 std::function<void()> func) { | ||||
|   this->set_timer_common_(component, SchedulerItem::TIMEOUT, false, &name, timeout, std::move(func)); | ||||
| } | ||||
| bool HOT Scheduler::cancel_timeout(Component *component, const std::string &name) { | ||||
|   return this->cancel_item_(component, name, SchedulerItem::TIMEOUT); | ||||
| } | ||||
| bool HOT Scheduler::cancel_timeout(Component *component, const char *name) { | ||||
|   return this->cancel_item_(component, name, SchedulerItem::TIMEOUT); | ||||
| } | ||||
| void HOT Scheduler::set_interval(Component *component, const std::string &name, uint32_t interval, | ||||
|                                  std::function<void()> func) { | ||||
|   const auto now = this->millis_(); | ||||
|   this->set_timer_common_(component, SchedulerItem::INTERVAL, false, &name, interval, std::move(func)); | ||||
| } | ||||
|  | ||||
|   if (!name.empty()) | ||||
|     this->cancel_interval(component, name); | ||||
|  | ||||
|   if (interval == SCHEDULER_DONT_RUN) | ||||
|     return; | ||||
|  | ||||
|   // only put offset in lower half | ||||
|   uint32_t offset = 0; | ||||
|   if (interval != 0) | ||||
|     offset = (random_uint32() % interval) / 2; | ||||
|  | ||||
|   auto item = make_unique<SchedulerItem>(); | ||||
|   item->component = component; | ||||
|   item->name = name; | ||||
|   item->type = SchedulerItem::INTERVAL; | ||||
|   item->interval = interval; | ||||
|   item->next_execution_ = now + offset; | ||||
|   item->callback = std::move(func); | ||||
|   item->remove = false; | ||||
| #ifdef ESPHOME_DEBUG_SCHEDULER | ||||
|   ESP_LOGD(TAG, "set_interval(name='%s/%s', interval=%" PRIu32 ", offset=%" PRIu32 ")", item->get_source(), | ||||
|            name.c_str(), interval, offset); | ||||
| #endif | ||||
|   this->push_(std::move(item)); | ||||
| void HOT Scheduler::set_interval(Component *component, const char *name, uint32_t interval, | ||||
|                                  std::function<void()> func) { | ||||
|   this->set_timer_common_(component, SchedulerItem::INTERVAL, true, name, interval, std::move(func)); | ||||
| } | ||||
| bool HOT Scheduler::cancel_interval(Component *component, const std::string &name) { | ||||
|   return this->cancel_item_(component, name, SchedulerItem::INTERVAL); | ||||
| } | ||||
| bool HOT Scheduler::cancel_interval(Component *component, const char *name) { | ||||
|   return this->cancel_item_(component, name, SchedulerItem::INTERVAL); | ||||
| } | ||||
|  | ||||
| struct RetryArgs { | ||||
|   std::function<RetryResult(uint8_t)> func; | ||||
|   uint8_t retry_countdown; | ||||
|   uint32_t current_interval; | ||||
|   Component *component; | ||||
|   std::string name; | ||||
|   std::string name;  // Keep as std::string since retry uses it dynamically | ||||
|   float backoff_increase_factor; | ||||
|   Scheduler *scheduler; | ||||
| }; | ||||
| @@ -154,7 +218,7 @@ void HOT Scheduler::call() { | ||||
|   if (now - last_print > 2000) { | ||||
|     last_print = now; | ||||
|     std::vector<std::unique_ptr<SchedulerItem>> old_items; | ||||
|     ESP_LOGD(TAG, "Items: count=%u, now=%" PRIu64 " (%u, %" PRIu32 ")", this->items_.size(), now, this->millis_major_, | ||||
|     ESP_LOGD(TAG, "Items: count=%zu, now=%" PRIu64 " (%u, %" PRIu32 ")", this->items_.size(), now, this->millis_major_, | ||||
|              this->last_millis_); | ||||
|     while (!this->empty_()) { | ||||
|       this->lock_.lock(); | ||||
| @@ -162,8 +226,9 @@ void HOT Scheduler::call() { | ||||
|       this->pop_raw_(); | ||||
|       this->lock_.unlock(); | ||||
|  | ||||
|       const char *name = item->get_name(); | ||||
|       ESP_LOGD(TAG, "  %s '%s/%s' interval=%" PRIu32 " next_execution in %" PRIu64 "ms at %" PRIu64, | ||||
|                item->get_type_str(), item->get_source(), item->name.c_str(), item->interval, | ||||
|                item->get_type_str(), item->get_source(), name ? name : "(null)", item->interval, | ||||
|                item->next_execution_ - now, item->next_execution_); | ||||
|  | ||||
|       old_items.push_back(std::move(item)); | ||||
| @@ -220,9 +285,10 @@ void HOT Scheduler::call() { | ||||
|       App.set_current_component(item->component); | ||||
|  | ||||
| #ifdef ESPHOME_DEBUG_SCHEDULER | ||||
|       const char *item_name = item->get_name(); | ||||
|       ESP_LOGV(TAG, "Running %s '%s/%s' with interval=%" PRIu32 " next_execution=%" PRIu64 " (now=%" PRIu64 ")", | ||||
|                item->get_type_str(), item->get_source(), item->name.c_str(), item->interval, item->next_execution_, | ||||
|                now); | ||||
|                item->get_type_str(), item->get_source(), item_name ? item_name : "(null)", item->interval, | ||||
|                item->next_execution_, now); | ||||
| #endif | ||||
|  | ||||
|       // Warning: During callback(), a lot of stuff can happen, including: | ||||
| @@ -298,19 +364,33 @@ void HOT Scheduler::push_(std::unique_ptr<Scheduler::SchedulerItem> item) { | ||||
|   LockGuard guard{this->lock_}; | ||||
|   this->to_add_.push_back(std::move(item)); | ||||
| } | ||||
| bool HOT Scheduler::cancel_item_(Component *component, const std::string &name, Scheduler::SchedulerItem::Type type) { | ||||
| // Common implementation for cancel operations | ||||
| bool HOT Scheduler::cancel_item_common_(Component *component, bool is_static_string, const void *name_ptr, | ||||
|                                         SchedulerItem::Type type) { | ||||
|   // Get the name as const char* | ||||
|   const char *name_cstr = | ||||
|       is_static_string ? static_cast<const char *>(name_ptr) : static_cast<const std::string *>(name_ptr)->c_str(); | ||||
|  | ||||
|   // Handle null or empty names | ||||
|   if (name_cstr == nullptr) | ||||
|     return false; | ||||
|  | ||||
|   // obtain lock because this function iterates and can be called from non-loop task context | ||||
|   LockGuard guard{this->lock_}; | ||||
|   bool ret = false; | ||||
|  | ||||
|   for (auto &it : this->items_) { | ||||
|     if (it->component == component && it->name == name && it->type == type && !it->remove) { | ||||
|     const char *item_name = it->get_name(); | ||||
|     if (it->component == component && item_name != nullptr && strcmp(name_cstr, item_name) == 0 && it->type == type && | ||||
|         !it->remove) { | ||||
|       to_remove_++; | ||||
|       it->remove = true; | ||||
|       ret = true; | ||||
|     } | ||||
|   } | ||||
|   for (auto &it : this->to_add_) { | ||||
|     if (it->component == component && it->name == name && it->type == type) { | ||||
|     const char *item_name = it->get_name(); | ||||
|     if (it->component == component && item_name != nullptr && strcmp(name_cstr, item_name) == 0 && it->type == type) { | ||||
|       it->remove = true; | ||||
|       ret = true; | ||||
|     } | ||||
| @@ -318,6 +398,15 @@ bool HOT Scheduler::cancel_item_(Component *component, const std::string &name, | ||||
|  | ||||
|   return ret; | ||||
| } | ||||
|  | ||||
| bool HOT Scheduler::cancel_item_(Component *component, const std::string &name, Scheduler::SchedulerItem::Type type) { | ||||
|   return this->cancel_item_common_(component, false, &name, type); | ||||
| } | ||||
|  | ||||
| bool HOT Scheduler::cancel_item_(Component *component, const char *name, SchedulerItem::Type type) { | ||||
|   return this->cancel_item_common_(component, true, name, type); | ||||
| } | ||||
|  | ||||
| uint64_t Scheduler::millis_() { | ||||
|   // Get the current 32-bit millis value | ||||
|   const uint32_t now = millis(); | ||||
|   | ||||
| @@ -12,11 +12,40 @@ class Component; | ||||
|  | ||||
| class Scheduler { | ||||
|  public: | ||||
|   // Public API - accepts std::string for backward compatibility | ||||
|   void set_timeout(Component *component, const std::string &name, uint32_t timeout, std::function<void()> func); | ||||
|   bool cancel_timeout(Component *component, const std::string &name); | ||||
|   void set_interval(Component *component, const std::string &name, uint32_t interval, std::function<void()> func); | ||||
|   bool cancel_interval(Component *component, const std::string &name); | ||||
|  | ||||
|   /** Set a timeout with a const char* name. | ||||
|    * | ||||
|    * IMPORTANT: The provided name pointer must remain valid for the lifetime of the scheduler item. | ||||
|    * This means the name should be: | ||||
|    *   - A string literal (e.g., "update") | ||||
|    *   - A static const char* variable | ||||
|    *   - A pointer with lifetime >= the scheduled task | ||||
|    * | ||||
|    * For dynamic strings, use the std::string overload instead. | ||||
|    */ | ||||
|   void set_timeout(Component *component, const char *name, uint32_t timeout, std::function<void()> func); | ||||
|  | ||||
|   bool cancel_timeout(Component *component, const std::string &name); | ||||
|   bool cancel_timeout(Component *component, const char *name); | ||||
|  | ||||
|   void set_interval(Component *component, const std::string &name, uint32_t interval, std::function<void()> func); | ||||
|  | ||||
|   /** Set an interval with a const char* name. | ||||
|    * | ||||
|    * IMPORTANT: The provided name pointer must remain valid for the lifetime of the scheduler item. | ||||
|    * This means the name should be: | ||||
|    *   - A string literal (e.g., "update") | ||||
|    *   - A static const char* variable | ||||
|    *   - A pointer with lifetime >= the scheduled task | ||||
|    * | ||||
|    * For dynamic strings, use the std::string overload instead. | ||||
|    */ | ||||
|   void set_interval(Component *component, const char *name, uint32_t interval, std::function<void()> func); | ||||
|  | ||||
|   bool cancel_interval(Component *component, const std::string &name); | ||||
|   bool cancel_interval(Component *component, const char *name); | ||||
|   void set_retry(Component *component, const std::string &name, uint32_t initial_wait_time, uint8_t max_attempts, | ||||
|                  std::function<RetryResult(uint8_t)> func, float backoff_increase_factor = 1.0f); | ||||
|   bool cancel_retry(Component *component, const std::string &name); | ||||
| @@ -36,32 +65,86 @@ class Scheduler { | ||||
|     // with a 16-bit rollover counter to create a 64-bit time that won't roll over for | ||||
|     // billions of years. This ensures correct scheduling even when devices run for months. | ||||
|     uint64_t next_execution_; | ||||
|     std::string name; | ||||
|     std::function<void()> callback; | ||||
|     enum Type : uint8_t { TIMEOUT, INTERVAL } type; | ||||
|     bool remove; | ||||
|  | ||||
|     static bool cmp(const std::unique_ptr<SchedulerItem> &a, const std::unique_ptr<SchedulerItem> &b); | ||||
|     const char *get_type_str() { | ||||
|       switch (this->type) { | ||||
|         case SchedulerItem::INTERVAL: | ||||
|           return "interval"; | ||||
|         case SchedulerItem::TIMEOUT: | ||||
|           return "timeout"; | ||||
|         default: | ||||
|           return ""; | ||||
|     // Optimized name storage using tagged union | ||||
|     union { | ||||
|       const char *static_name;  // For string literals (no allocation) | ||||
|       char *dynamic_name;       // For allocated strings | ||||
|     } name_; | ||||
|  | ||||
|     std::function<void()> callback; | ||||
|  | ||||
|     // Bit-packed fields to minimize padding | ||||
|     enum Type : uint8_t { TIMEOUT, INTERVAL } type : 1; | ||||
|     bool remove : 1; | ||||
|     bool name_is_dynamic : 1;  // True if name was dynamically allocated (needs delete[]) | ||||
|     // 5 bits padding | ||||
|  | ||||
|     // Constructor | ||||
|     SchedulerItem() | ||||
|         : component(nullptr), interval(0), next_execution_(0), type(TIMEOUT), remove(false), name_is_dynamic(false) { | ||||
|       name_.static_name = nullptr; | ||||
|     } | ||||
|  | ||||
|     // Destructor to clean up dynamic names | ||||
|     ~SchedulerItem() { | ||||
|       if (name_is_dynamic) { | ||||
|         delete[] name_.dynamic_name; | ||||
|       } | ||||
|     } | ||||
|     const char *get_source() { | ||||
|       return this->component != nullptr ? this->component->get_component_source() : "unknown"; | ||||
|  | ||||
|     // Delete copy operations to prevent accidental copies | ||||
|     SchedulerItem(const SchedulerItem &) = delete; | ||||
|     SchedulerItem &operator=(const SchedulerItem &) = delete; | ||||
|  | ||||
|     // Default move operations | ||||
|     SchedulerItem(SchedulerItem &&) = default; | ||||
|     SchedulerItem &operator=(SchedulerItem &&) = default; | ||||
|  | ||||
|     // Helper to get the name regardless of storage type | ||||
|     const char *get_name() const { return name_is_dynamic ? name_.dynamic_name : name_.static_name; } | ||||
|  | ||||
|     // Helper to set name with proper ownership | ||||
|     void set_name(const char *name, bool make_copy = false) { | ||||
|       // Clean up old dynamic name if any | ||||
|       if (name_is_dynamic && name_.dynamic_name) { | ||||
|         delete[] name_.dynamic_name; | ||||
|         name_is_dynamic = false; | ||||
|       } | ||||
|  | ||||
|       if (!name || !name[0]) { | ||||
|         name_.static_name = nullptr; | ||||
|       } else if (make_copy) { | ||||
|         // Make a copy for dynamic strings | ||||
|         size_t len = strlen(name); | ||||
|         name_.dynamic_name = new char[len + 1]; | ||||
|         memcpy(name_.dynamic_name, name, len + 1); | ||||
|         name_is_dynamic = true; | ||||
|       } else { | ||||
|         // Use static string directly | ||||
|         name_.static_name = name; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     static bool cmp(const std::unique_ptr<SchedulerItem> &a, const std::unique_ptr<SchedulerItem> &b); | ||||
|     const char *get_type_str() const { return (type == TIMEOUT) ? "timeout" : "interval"; } | ||||
|     const char *get_source() const { return component ? component->get_component_source() : "unknown"; } | ||||
|   }; | ||||
|  | ||||
|   // Common implementation for both timeout and interval | ||||
|   void set_timer_common_(Component *component, SchedulerItem::Type type, bool is_static_string, const void *name_ptr, | ||||
|                          uint32_t delay, std::function<void()> func); | ||||
|  | ||||
|   uint64_t millis_(); | ||||
|   void cleanup_(); | ||||
|   void pop_raw_(); | ||||
|   void push_(std::unique_ptr<SchedulerItem> item); | ||||
|   // Common implementation for cancel operations | ||||
|   bool cancel_item_common_(Component *component, bool is_static_string, const void *name_ptr, SchedulerItem::Type type); | ||||
|  | ||||
|   bool cancel_item_(Component *component, const std::string &name, SchedulerItem::Type type); | ||||
|   bool cancel_item_(Component *component, const char *name, SchedulerItem::Type type); | ||||
|  | ||||
|   bool empty_() { | ||||
|     this->cleanup_(); | ||||
|     return this->items_.empty(); | ||||
|   | ||||
							
								
								
									
										71
									
								
								tests/integration/fixtures/api_conditional_memory.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										71
									
								
								tests/integration/fixtures/api_conditional_memory.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,71 @@ | ||||
| esphome: | ||||
|   name: api-conditional-memory-test | ||||
| host: | ||||
| api: | ||||
|   actions: | ||||
|     - action: test_simple_service | ||||
|       then: | ||||
|         - logger.log: "Simple service called" | ||||
|         - binary_sensor.template.publish: | ||||
|             id: service_called_sensor | ||||
|             state: ON | ||||
|     - action: test_service_with_args | ||||
|       variables: | ||||
|         arg_string: string | ||||
|         arg_int: int | ||||
|         arg_bool: bool | ||||
|         arg_float: float | ||||
|       then: | ||||
|         - logger.log: | ||||
|             format: "Service called with: %s, %d, %d, %.2f" | ||||
|             args: [arg_string.c_str(), arg_int, arg_bool, arg_float] | ||||
|         - sensor.template.publish: | ||||
|             id: service_arg_sensor | ||||
|             state: !lambda 'return arg_float;' | ||||
|   on_client_connected: | ||||
|     - logger.log: | ||||
|         format: "Client %s connected from %s" | ||||
|         args: [client_info.c_str(), client_address.c_str()] | ||||
|     - binary_sensor.template.publish: | ||||
|         id: client_connected | ||||
|         state: ON | ||||
|     - text_sensor.template.publish: | ||||
|         id: last_client_info | ||||
|         state: !lambda 'return client_info;' | ||||
|   on_client_disconnected: | ||||
|     - logger.log: | ||||
|         format: "Client %s disconnected from %s" | ||||
|         args: [client_info.c_str(), client_address.c_str()] | ||||
|     - binary_sensor.template.publish: | ||||
|         id: client_connected | ||||
|         state: OFF | ||||
|     - binary_sensor.template.publish: | ||||
|         id: client_disconnected_event | ||||
|         state: ON | ||||
|  | ||||
| logger: | ||||
|   level: DEBUG | ||||
|  | ||||
| binary_sensor: | ||||
|   - platform: template | ||||
|     name: "Client Connected" | ||||
|     id: client_connected | ||||
|     device_class: connectivity | ||||
|   - platform: template | ||||
|     name: "Client Disconnected Event" | ||||
|     id: client_disconnected_event | ||||
|   - platform: template | ||||
|     name: "Service Called" | ||||
|     id: service_called_sensor | ||||
|  | ||||
| sensor: | ||||
|   - platform: template | ||||
|     name: "Service Argument Value" | ||||
|     id: service_arg_sensor | ||||
|     unit_of_measurement: "" | ||||
|     accuracy_decimals: 2 | ||||
|  | ||||
| text_sensor: | ||||
|   - platform: template | ||||
|     name: "Last Client Info" | ||||
|     id: last_client_info | ||||
							
								
								
									
										164
									
								
								tests/integration/fixtures/scheduler_string_test.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										164
									
								
								tests/integration/fixtures/scheduler_string_test.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,164 @@ | ||||
| esphome: | ||||
|   name: scheduler-string-test | ||||
|   on_boot: | ||||
|     priority: -100 | ||||
|     then: | ||||
|       - logger.log: "Starting scheduler string tests" | ||||
|   platformio_options: | ||||
|     build_flags: | ||||
|       - "-DESPHOME_DEBUG_SCHEDULER"  # Enable scheduler debug logging | ||||
|  | ||||
| host: | ||||
| api: | ||||
| logger: | ||||
|   level: VERBOSE | ||||
|  | ||||
| globals: | ||||
|   - id: timeout_counter | ||||
|     type: int | ||||
|     initial_value: '0' | ||||
|   - id: interval_counter | ||||
|     type: int | ||||
|     initial_value: '0' | ||||
|   - id: dynamic_counter | ||||
|     type: int | ||||
|     initial_value: '0' | ||||
|   - id: static_tests_done | ||||
|     type: bool | ||||
|     initial_value: 'false' | ||||
|   - id: dynamic_tests_done | ||||
|     type: bool | ||||
|     initial_value: 'false' | ||||
|   - id: results_reported | ||||
|     type: bool | ||||
|     initial_value: 'false' | ||||
|  | ||||
| script: | ||||
|   - id: test_static_strings | ||||
|     then: | ||||
|       - logger.log: "Testing static string timeouts and intervals" | ||||
|       - lambda: |- | ||||
|           auto *component1 = id(test_sensor1); | ||||
|           // Test 1: Static string literals with set_timeout | ||||
|           App.scheduler.set_timeout(component1, "static_timeout_1", 50, []() { | ||||
|             ESP_LOGI("test", "Static timeout 1 fired"); | ||||
|             id(timeout_counter) += 1; | ||||
|           }); | ||||
|  | ||||
|           // Test 2: Static const char* with set_timeout | ||||
|           static const char* TIMEOUT_NAME = "static_timeout_2"; | ||||
|           App.scheduler.set_timeout(component1, TIMEOUT_NAME, 100, []() { | ||||
|             ESP_LOGI("test", "Static timeout 2 fired"); | ||||
|             id(timeout_counter) += 1; | ||||
|           }); | ||||
|  | ||||
|           // Test 3: Static string literal with set_interval | ||||
|           App.scheduler.set_interval(component1, "static_interval_1", 200, []() { | ||||
|             ESP_LOGI("test", "Static interval 1 fired, count: %d", id(interval_counter)); | ||||
|             id(interval_counter) += 1; | ||||
|             if (id(interval_counter) >= 3) { | ||||
|               App.scheduler.cancel_interval(id(test_sensor1), "static_interval_1"); | ||||
|               ESP_LOGI("test", "Cancelled static interval 1"); | ||||
|             } | ||||
|           }); | ||||
|  | ||||
|           // Test 4: Empty string (should be handled safely) | ||||
|           App.scheduler.set_timeout(component1, "", 150, []() { | ||||
|             ESP_LOGI("test", "Empty string timeout fired"); | ||||
|           }); | ||||
|  | ||||
|           // Test 5: Cancel timeout with const char* literal | ||||
|           App.scheduler.set_timeout(component1, "cancel_static_timeout", 5000, []() { | ||||
|             ESP_LOGI("test", "This static timeout should be cancelled"); | ||||
|           }); | ||||
|           // Cancel using const char* directly | ||||
|           App.scheduler.cancel_timeout(component1, "cancel_static_timeout"); | ||||
|           ESP_LOGI("test", "Cancelled static timeout using const char*"); | ||||
|  | ||||
|   - id: test_dynamic_strings | ||||
|     then: | ||||
|       - logger.log: "Testing dynamic string timeouts and intervals" | ||||
|       - lambda: |- | ||||
|           auto *component2 = id(test_sensor2); | ||||
|  | ||||
|           // Test 6: Dynamic string with set_timeout (std::string) | ||||
|           std::string dynamic_name = "dynamic_timeout_" + std::to_string(id(dynamic_counter)++); | ||||
|           App.scheduler.set_timeout(component2, dynamic_name, 100, []() { | ||||
|             ESP_LOGI("test", "Dynamic timeout fired"); | ||||
|             id(timeout_counter) += 1; | ||||
|           }); | ||||
|  | ||||
|           // Test 7: Dynamic string with set_interval | ||||
|           std::string interval_name = "dynamic_interval_" + std::to_string(id(dynamic_counter)++); | ||||
|           App.scheduler.set_interval(component2, interval_name, 250, [interval_name]() { | ||||
|             ESP_LOGI("test", "Dynamic interval fired: %s", interval_name.c_str()); | ||||
|             id(interval_counter) += 1; | ||||
|             if (id(interval_counter) >= 6) { | ||||
|               App.scheduler.cancel_interval(id(test_sensor2), interval_name); | ||||
|               ESP_LOGI("test", "Cancelled dynamic interval"); | ||||
|             } | ||||
|           }); | ||||
|  | ||||
|           // Test 8: Cancel with different string object but same content | ||||
|           std::string cancel_name = "cancel_test"; | ||||
|           App.scheduler.set_timeout(component2, cancel_name, 2000, []() { | ||||
|             ESP_LOGI("test", "This should be cancelled"); | ||||
|           }); | ||||
|  | ||||
|           // Cancel using a different string object | ||||
|           std::string cancel_name_2 = "cancel_test"; | ||||
|           App.scheduler.cancel_timeout(component2, cancel_name_2); | ||||
|           ESP_LOGI("test", "Cancelled timeout using different string object"); | ||||
|  | ||||
|   - id: report_results | ||||
|     then: | ||||
|       - lambda: |- | ||||
|           ESP_LOGI("test", "Final results - Timeouts: %d, Intervals: %d", | ||||
|                    id(timeout_counter), id(interval_counter)); | ||||
|  | ||||
| sensor: | ||||
|   - platform: template | ||||
|     name: Test Sensor 1 | ||||
|     id: test_sensor1 | ||||
|     lambda: return 1.0; | ||||
|     update_interval: never | ||||
|  | ||||
|   - platform: template | ||||
|     name: Test Sensor 2 | ||||
|     id: test_sensor2 | ||||
|     lambda: return 2.0; | ||||
|     update_interval: never | ||||
|  | ||||
| interval: | ||||
|   # Run static string tests after boot - using script to run once | ||||
|   - interval: 0.1s | ||||
|     then: | ||||
|       - if: | ||||
|           condition: | ||||
|             lambda: 'return id(static_tests_done) == false;' | ||||
|           then: | ||||
|             - lambda: 'id(static_tests_done) = true;' | ||||
|             - script.execute: test_static_strings | ||||
|             - logger.log: "Started static string tests" | ||||
|  | ||||
|   # Run dynamic string tests after static tests | ||||
|   - interval: 0.2s | ||||
|     then: | ||||
|       - if: | ||||
|           condition: | ||||
|             lambda: 'return id(static_tests_done) && !id(dynamic_tests_done);' | ||||
|           then: | ||||
|             - lambda: 'id(dynamic_tests_done) = true;' | ||||
|             - delay: 0.2s | ||||
|             - script.execute: test_dynamic_strings | ||||
|  | ||||
|   # Report results after all tests | ||||
|   - interval: 0.2s | ||||
|     then: | ||||
|       - if: | ||||
|           condition: | ||||
|             lambda: 'return id(dynamic_tests_done) && !id(results_reported);' | ||||
|           then: | ||||
|             - lambda: 'id(results_reported) = true;' | ||||
|             - delay: 1s | ||||
|             - script.execute: report_results | ||||
							
								
								
									
										205
									
								
								tests/integration/test_api_conditional_memory.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										205
									
								
								tests/integration/test_api_conditional_memory.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,205 @@ | ||||
| """Integration test for API conditional memory optimization with triggers and services.""" | ||||
|  | ||||
| from __future__ import annotations | ||||
|  | ||||
| import asyncio | ||||
|  | ||||
| from aioesphomeapi import ( | ||||
|     BinarySensorInfo, | ||||
|     EntityState, | ||||
|     SensorInfo, | ||||
|     TextSensorInfo, | ||||
|     UserService, | ||||
|     UserServiceArgType, | ||||
| ) | ||||
| import pytest | ||||
|  | ||||
| from .types import APIClientConnectedFactory, RunCompiledFunction | ||||
|  | ||||
|  | ||||
| @pytest.mark.asyncio | ||||
| async def test_api_conditional_memory( | ||||
|     yaml_config: str, | ||||
|     run_compiled: RunCompiledFunction, | ||||
|     api_client_connected: APIClientConnectedFactory, | ||||
| ) -> None: | ||||
|     """Test API triggers and services work correctly with conditional compilation.""" | ||||
|     loop = asyncio.get_running_loop() | ||||
|     # Keep ESPHome process running throughout the test | ||||
|     async with run_compiled(yaml_config): | ||||
|         # First connection | ||||
|         async with api_client_connected() as client: | ||||
|             # Verify device info | ||||
|             device_info = await client.device_info() | ||||
|             assert device_info is not None | ||||
|             assert device_info.name == "api-conditional-memory-test" | ||||
|  | ||||
|             # List entities and services | ||||
|             entity_info, services = await asyncio.wait_for( | ||||
|                 client.list_entities_services(), timeout=5.0 | ||||
|             ) | ||||
|  | ||||
|             # Find our entities | ||||
|             client_connected: BinarySensorInfo | None = None | ||||
|             client_disconnected_event: BinarySensorInfo | None = None | ||||
|             service_called_sensor: BinarySensorInfo | None = None | ||||
|             service_arg_sensor: SensorInfo | None = None | ||||
|             last_client_info: TextSensorInfo | None = None | ||||
|  | ||||
|             for entity in entity_info: | ||||
|                 if isinstance(entity, BinarySensorInfo): | ||||
|                     if entity.object_id == "client_connected": | ||||
|                         client_connected = entity | ||||
|                     elif entity.object_id == "client_disconnected_event": | ||||
|                         client_disconnected_event = entity | ||||
|                     elif entity.object_id == "service_called": | ||||
|                         service_called_sensor = entity | ||||
|                 elif isinstance(entity, SensorInfo): | ||||
|                     if entity.object_id == "service_argument_value": | ||||
|                         service_arg_sensor = entity | ||||
|                 elif isinstance(entity, TextSensorInfo): | ||||
|                     if entity.object_id == "last_client_info": | ||||
|                         last_client_info = entity | ||||
|  | ||||
|             # Verify all entities exist | ||||
|             assert client_connected is not None, "client_connected sensor not found" | ||||
|             assert client_disconnected_event is not None, ( | ||||
|                 "client_disconnected_event sensor not found" | ||||
|             ) | ||||
|             assert service_called_sensor is not None, "service_called sensor not found" | ||||
|             assert service_arg_sensor is not None, "service_arg_sensor not found" | ||||
|             assert last_client_info is not None, "last_client_info sensor not found" | ||||
|  | ||||
|             # Verify services exist | ||||
|             assert len(services) == 2, f"Expected 2 services, found {len(services)}" | ||||
|  | ||||
|             # Find our services | ||||
|             simple_service: UserService | None = None | ||||
|             service_with_args: UserService | None = None | ||||
|  | ||||
|             for service in services: | ||||
|                 if service.name == "test_simple_service": | ||||
|                     simple_service = service | ||||
|                 elif service.name == "test_service_with_args": | ||||
|                     service_with_args = service | ||||
|  | ||||
|             assert simple_service is not None, "test_simple_service not found" | ||||
|             assert service_with_args is not None, "test_service_with_args not found" | ||||
|  | ||||
|             # Verify service arguments | ||||
|             assert len(service_with_args.args) == 4, ( | ||||
|                 f"Expected 4 args, found {len(service_with_args.args)}" | ||||
|             ) | ||||
|  | ||||
|             # Check arg types | ||||
|             arg_types = {arg.name: arg.type for arg in service_with_args.args} | ||||
|             assert arg_types["arg_string"] == UserServiceArgType.STRING | ||||
|             assert arg_types["arg_int"] == UserServiceArgType.INT | ||||
|             assert arg_types["arg_bool"] == UserServiceArgType.BOOL | ||||
|             assert arg_types["arg_float"] == UserServiceArgType.FLOAT | ||||
|  | ||||
|             # Track state changes | ||||
|             states: dict[int, EntityState] = {} | ||||
|             states_future: asyncio.Future[None] = loop.create_future() | ||||
|  | ||||
|             def on_state(state: EntityState) -> None: | ||||
|                 states[state.key] = state | ||||
|                 # Check if we have initial states for connection sensors | ||||
|                 if ( | ||||
|                     client_connected.key in states | ||||
|                     and last_client_info.key in states | ||||
|                     and not states_future.done() | ||||
|                 ): | ||||
|                     states_future.set_result(None) | ||||
|  | ||||
|             client.subscribe_states(on_state) | ||||
|  | ||||
|             # Wait for initial states | ||||
|             await asyncio.wait_for(states_future, timeout=5.0) | ||||
|  | ||||
|             # Verify on_client_connected trigger fired | ||||
|             connected_state = states.get(client_connected.key) | ||||
|             assert connected_state is not None | ||||
|             assert connected_state.state is True, "Client should be connected" | ||||
|  | ||||
|             # Verify client info was captured | ||||
|             client_info_state = states.get(last_client_info.key) | ||||
|             assert client_info_state is not None | ||||
|             assert isinstance(client_info_state.state, str) | ||||
|             assert len(client_info_state.state) > 0, "Client info should not be empty" | ||||
|  | ||||
|             # Test simple service | ||||
|             service_future: asyncio.Future[None] = loop.create_future() | ||||
|  | ||||
|             def check_service_called(state: EntityState) -> None: | ||||
|                 if state.key == service_called_sensor.key and state.state is True: | ||||
|                     if not service_future.done(): | ||||
|                         service_future.set_result(None) | ||||
|  | ||||
|             # Update callback to check for service execution | ||||
|             client.subscribe_states(check_service_called) | ||||
|  | ||||
|             # Call simple service | ||||
|             client.execute_service(simple_service, {}) | ||||
|  | ||||
|             # Wait for service to execute | ||||
|             await asyncio.wait_for(service_future, timeout=5.0) | ||||
|  | ||||
|             # Test service with arguments | ||||
|             arg_future: asyncio.Future[None] = loop.create_future() | ||||
|             expected_float = 42.5 | ||||
|  | ||||
|             def check_arg_sensor(state: EntityState) -> None: | ||||
|                 if ( | ||||
|                     state.key == service_arg_sensor.key | ||||
|                     and abs(state.state - expected_float) < 0.01 | ||||
|                 ): | ||||
|                     if not arg_future.done(): | ||||
|                         arg_future.set_result(None) | ||||
|  | ||||
|             client.subscribe_states(check_arg_sensor) | ||||
|  | ||||
|             # Call service with arguments | ||||
|             client.execute_service( | ||||
|                 service_with_args, | ||||
|                 { | ||||
|                     "arg_string": "test_string", | ||||
|                     "arg_int": 123, | ||||
|                     "arg_bool": True, | ||||
|                     "arg_float": expected_float, | ||||
|                 }, | ||||
|             ) | ||||
|  | ||||
|             # Wait for service with args to execute | ||||
|             await asyncio.wait_for(arg_future, timeout=5.0) | ||||
|  | ||||
|         # After disconnecting first client, reconnect and verify triggers work | ||||
|         async with api_client_connected() as client2: | ||||
|             # Subscribe to states with new client | ||||
|             states2: dict[int, EntityState] = {} | ||||
|             connected_future: asyncio.Future[None] = loop.create_future() | ||||
|  | ||||
|             def on_state2(state: EntityState) -> None: | ||||
|                 states2[state.key] = state | ||||
|                 # Check for reconnection | ||||
|                 if state.key == client_connected.key and state.state is True: | ||||
|                     if not connected_future.done(): | ||||
|                         connected_future.set_result(None) | ||||
|  | ||||
|             client2.subscribe_states(on_state2) | ||||
|  | ||||
|             # Wait for connected state | ||||
|             await asyncio.wait_for(connected_future, timeout=5.0) | ||||
|  | ||||
|             # Verify client is connected again (on_client_connected fired) | ||||
|             assert states2[client_connected.key].state is True, ( | ||||
|                 "Client should be reconnected" | ||||
|             ) | ||||
|  | ||||
|             # The client_disconnected_event should be ON from when we disconnected | ||||
|             # (it was set ON by on_client_disconnected trigger) | ||||
|             disconnected_state = states2.get(client_disconnected_event.key) | ||||
|             assert disconnected_state is not None | ||||
|             assert disconnected_state.state is True, ( | ||||
|                 "Disconnect event should be ON from previous disconnect" | ||||
|             ) | ||||
							
								
								
									
										166
									
								
								tests/integration/test_scheduler_string_test.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										166
									
								
								tests/integration/test_scheduler_string_test.py
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,166 @@ | ||||
| """Test scheduler string optimization with static and dynamic strings.""" | ||||
|  | ||||
| import asyncio | ||||
| import re | ||||
|  | ||||
| import pytest | ||||
|  | ||||
| from .types import APIClientConnectedFactory, RunCompiledFunction | ||||
|  | ||||
|  | ||||
| @pytest.mark.asyncio | ||||
| async def test_scheduler_string_test( | ||||
|     yaml_config: str, | ||||
|     run_compiled: RunCompiledFunction, | ||||
|     api_client_connected: APIClientConnectedFactory, | ||||
| ) -> None: | ||||
|     """Test that scheduler handles both static and dynamic strings correctly.""" | ||||
|     # Track counts | ||||
|     timeout_count = 0 | ||||
|     interval_count = 0 | ||||
|  | ||||
|     # Events for each test completion | ||||
|     static_timeout_1_fired = asyncio.Event() | ||||
|     static_timeout_2_fired = asyncio.Event() | ||||
|     static_interval_fired = asyncio.Event() | ||||
|     static_interval_cancelled = asyncio.Event() | ||||
|     empty_string_timeout_fired = asyncio.Event() | ||||
|     static_timeout_cancelled = asyncio.Event() | ||||
|     dynamic_timeout_fired = asyncio.Event() | ||||
|     dynamic_interval_fired = asyncio.Event() | ||||
|     cancel_test_done = asyncio.Event() | ||||
|     final_results_logged = asyncio.Event() | ||||
|  | ||||
|     # Track interval counts | ||||
|     static_interval_count = 0 | ||||
|     dynamic_interval_count = 0 | ||||
|  | ||||
|     def on_log_line(line: str) -> None: | ||||
|         nonlocal \ | ||||
|             timeout_count, \ | ||||
|             interval_count, \ | ||||
|             static_interval_count, \ | ||||
|             dynamic_interval_count | ||||
|  | ||||
|         # Strip ANSI color codes | ||||
|         clean_line = re.sub(r"\x1b\[[0-9;]*m", "", line) | ||||
|  | ||||
|         # Check for static timeout completions | ||||
|         if "Static timeout 1 fired" in clean_line: | ||||
|             static_timeout_1_fired.set() | ||||
|             timeout_count += 1 | ||||
|  | ||||
|         elif "Static timeout 2 fired" in clean_line: | ||||
|             static_timeout_2_fired.set() | ||||
|             timeout_count += 1 | ||||
|  | ||||
|         # Check for static interval | ||||
|         elif "Static interval 1 fired" in clean_line: | ||||
|             match = re.search(r"count: (\d+)", clean_line) | ||||
|             if match: | ||||
|                 static_interval_count = int(match.group(1)) | ||||
|                 static_interval_fired.set() | ||||
|  | ||||
|         elif "Cancelled static interval 1" in clean_line: | ||||
|             static_interval_cancelled.set() | ||||
|  | ||||
|         # Check for empty string timeout | ||||
|         elif "Empty string timeout fired" in clean_line: | ||||
|             empty_string_timeout_fired.set() | ||||
|  | ||||
|         # Check for static timeout cancellation | ||||
|         elif "Cancelled static timeout using const char*" in clean_line: | ||||
|             static_timeout_cancelled.set() | ||||
|  | ||||
|         # Check for dynamic string tests | ||||
|         elif "Dynamic timeout fired" in clean_line: | ||||
|             dynamic_timeout_fired.set() | ||||
|             timeout_count += 1 | ||||
|  | ||||
|         elif "Dynamic interval fired" in clean_line: | ||||
|             dynamic_interval_count += 1 | ||||
|             dynamic_interval_fired.set() | ||||
|  | ||||
|         # Check for cancel test | ||||
|         elif "Cancelled timeout using different string object" in clean_line: | ||||
|             cancel_test_done.set() | ||||
|  | ||||
|         # Check for final results | ||||
|         elif "Final results" in clean_line: | ||||
|             match = re.search(r"Timeouts: (\d+), Intervals: (\d+)", clean_line) | ||||
|             if match: | ||||
|                 timeout_count = int(match.group(1)) | ||||
|                 interval_count = int(match.group(2)) | ||||
|                 final_results_logged.set() | ||||
|  | ||||
|     async with ( | ||||
|         run_compiled(yaml_config, line_callback=on_log_line), | ||||
|         api_client_connected() as client, | ||||
|     ): | ||||
|         # Verify we can connect | ||||
|         device_info = await client.device_info() | ||||
|         assert device_info is not None | ||||
|         assert device_info.name == "scheduler-string-test" | ||||
|  | ||||
|         # Wait for static string tests | ||||
|         try: | ||||
|             await asyncio.wait_for(static_timeout_1_fired.wait(), timeout=0.5) | ||||
|         except asyncio.TimeoutError: | ||||
|             pytest.fail("Static timeout 1 did not fire within 0.5 seconds") | ||||
|  | ||||
|         try: | ||||
|             await asyncio.wait_for(static_timeout_2_fired.wait(), timeout=0.5) | ||||
|         except asyncio.TimeoutError: | ||||
|             pytest.fail("Static timeout 2 did not fire within 0.5 seconds") | ||||
|  | ||||
|         try: | ||||
|             await asyncio.wait_for(static_interval_fired.wait(), timeout=1.0) | ||||
|         except asyncio.TimeoutError: | ||||
|             pytest.fail("Static interval did not fire within 1 second") | ||||
|  | ||||
|         try: | ||||
|             await asyncio.wait_for(static_interval_cancelled.wait(), timeout=2.0) | ||||
|         except asyncio.TimeoutError: | ||||
|             pytest.fail("Static interval was not cancelled within 2 seconds") | ||||
|  | ||||
|         # Verify static interval ran at least 3 times | ||||
|         assert static_interval_count >= 2, ( | ||||
|             f"Expected static interval to run at least 3 times, got {static_interval_count + 1}" | ||||
|         ) | ||||
|  | ||||
|         # Verify static timeout was cancelled | ||||
|         assert static_timeout_cancelled.is_set(), ( | ||||
|             "Static timeout should have been cancelled" | ||||
|         ) | ||||
|  | ||||
|         # Wait for dynamic string tests | ||||
|         try: | ||||
|             await asyncio.wait_for(dynamic_timeout_fired.wait(), timeout=1.0) | ||||
|         except asyncio.TimeoutError: | ||||
|             pytest.fail("Dynamic timeout did not fire within 1 second") | ||||
|  | ||||
|         try: | ||||
|             await asyncio.wait_for(dynamic_interval_fired.wait(), timeout=1.5) | ||||
|         except asyncio.TimeoutError: | ||||
|             pytest.fail("Dynamic interval did not fire within 1.5 seconds") | ||||
|  | ||||
|         # Wait for cancel test | ||||
|         try: | ||||
|             await asyncio.wait_for(cancel_test_done.wait(), timeout=1.0) | ||||
|         except asyncio.TimeoutError: | ||||
|             pytest.fail("Cancel test did not complete within 1 second") | ||||
|  | ||||
|         # Wait for final results | ||||
|         try: | ||||
|             await asyncio.wait_for(final_results_logged.wait(), timeout=4.0) | ||||
|         except asyncio.TimeoutError: | ||||
|             pytest.fail("Final results were not logged within 4 seconds") | ||||
|  | ||||
|         # Verify results | ||||
|         assert timeout_count >= 3, f"Expected at least 3 timeouts, got {timeout_count}" | ||||
|         assert interval_count >= 3, ( | ||||
|             f"Expected at least 3 interval fires, got {interval_count}" | ||||
|         ) | ||||
|  | ||||
|         # Empty string timeout DOES fire (scheduler accepts empty names) | ||||
|         assert empty_string_timeout_fired.is_set(), "Empty string timeout should fire" | ||||
		Reference in New Issue
	
	Block a user