1
0
mirror of https://github.com/esphome/esphome.git synced 2025-10-25 13:13:48 +01:00

[esp32_improv, improv_base] Reduce flash usage by 352 bytes (#11406)

This commit is contained in:
J. Nick Koston
2025-10-20 11:07:39 -10:00
committed by GitHub
parent 1706a69fad
commit 0f4b54aa82
2 changed files with 29 additions and 21 deletions

View File

@@ -384,26 +384,32 @@ void ESP32ImprovComponent::check_wifi_connection_() {
this->connecting_sta_ = {}; this->connecting_sta_ = {};
this->cancel_timeout("wifi-connect-timeout"); this->cancel_timeout("wifi-connect-timeout");
std::vector<std::string> urls; // Build URL list with minimal allocations
// Maximum 3 URLs: custom next_url + ESPHOME_MY_LINK + webserver URL
std::string url_strings[3];
size_t url_count = 0;
// Add next_url if configured (should be first per Improv BLE spec) // Add next_url if configured (should be first per Improv BLE spec)
std::string next_url = this->get_formatted_next_url_(); std::string next_url = this->get_formatted_next_url_();
if (!next_url.empty()) { if (!next_url.empty()) {
urls.push_back(next_url); url_strings[url_count++] = std::move(next_url);
} }
// Add default URLs for backward compatibility // Add default URLs for backward compatibility
urls.emplace_back(ESPHOME_MY_LINK); url_strings[url_count++] = ESPHOME_MY_LINK;
#ifdef USE_WEBSERVER #ifdef USE_WEBSERVER
for (auto &ip : wifi::global_wifi_component->wifi_sta_ip_addresses()) { for (auto &ip : wifi::global_wifi_component->wifi_sta_ip_addresses()) {
if (ip.is_ip4()) { if (ip.is_ip4()) {
std::string webserver_url = "http://" + ip.str() + ":" + to_string(USE_WEBSERVER_PORT); char url_buffer[64];
urls.push_back(webserver_url); snprintf(url_buffer, sizeof(url_buffer), "http://%s:%d", ip.str().c_str(), USE_WEBSERVER_PORT);
url_strings[url_count++] = url_buffer;
break; break;
} }
} }
#endif #endif
std::vector<uint8_t> data = improv::build_rpc_response(improv::WIFI_SETTINGS, urls); // Pass to build_rpc_response using vector constructor from iterators to avoid extra copies
std::vector<uint8_t> data = improv::build_rpc_response(
improv::WIFI_SETTINGS, std::vector<std::string>(url_strings, url_strings + url_count));
this->send_response_(data); this->send_response_(data);
} else if (this->is_active() && this->state_ != improv::STATE_PROVISIONED) { } else if (this->is_active() && this->state_ != improv::STATE_PROVISIONED) {
ESP_LOGD(TAG, "WiFi provisioned externally"); ESP_LOGD(TAG, "WiFi provisioned externally");

View File

@@ -6,6 +6,21 @@
namespace esphome { namespace esphome {
namespace improv_base { namespace improv_base {
static constexpr const char DEVICE_NAME_PLACEHOLDER[] = "{{device_name}}";
static constexpr size_t DEVICE_NAME_PLACEHOLDER_LEN = sizeof(DEVICE_NAME_PLACEHOLDER) - 1;
static constexpr const char IP_ADDRESS_PLACEHOLDER[] = "{{ip_address}}";
static constexpr size_t IP_ADDRESS_PLACEHOLDER_LEN = sizeof(IP_ADDRESS_PLACEHOLDER) - 1;
static void replace_all_in_place(std::string &str, const char *placeholder, size_t placeholder_len,
const std::string &replacement) {
size_t pos = 0;
const size_t replacement_len = replacement.length();
while ((pos = str.find(placeholder, pos)) != std::string::npos) {
str.replace(pos, placeholder_len, replacement);
pos += replacement_len;
}
}
std::string ImprovBase::get_formatted_next_url_() { std::string ImprovBase::get_formatted_next_url_() {
if (this->next_url_.empty()) { if (this->next_url_.empty()) {
return ""; return "";
@@ -14,28 +29,15 @@ std::string ImprovBase::get_formatted_next_url_() {
std::string formatted_url = this->next_url_; std::string formatted_url = this->next_url_;
// Replace all occurrences of {{device_name}} // Replace all occurrences of {{device_name}}
const std::string device_name_placeholder = "{{device_name}}"; replace_all_in_place(formatted_url, DEVICE_NAME_PLACEHOLDER, DEVICE_NAME_PLACEHOLDER_LEN, App.get_name());
const std::string &device_name = App.get_name();
size_t pos = 0;
while ((pos = formatted_url.find(device_name_placeholder, pos)) != std::string::npos) {
formatted_url.replace(pos, device_name_placeholder.length(), device_name);
pos += device_name.length();
}
// Replace all occurrences of {{ip_address}} // Replace all occurrences of {{ip_address}}
const std::string ip_address_placeholder = "{{ip_address}}";
std::string ip_address_str;
for (auto &ip : network::get_ip_addresses()) { for (auto &ip : network::get_ip_addresses()) {
if (ip.is_ip4()) { if (ip.is_ip4()) {
ip_address_str = ip.str(); replace_all_in_place(formatted_url, IP_ADDRESS_PLACEHOLDER, IP_ADDRESS_PLACEHOLDER_LEN, ip.str());
break; break;
} }
} }
pos = 0;
while ((pos = formatted_url.find(ip_address_placeholder, pos)) != std::string::npos) {
formatted_url.replace(pos, ip_address_placeholder.length(), ip_address_str);
pos += ip_address_str.length();
}
// Note: {{esphome_version}} is replaced at code generation time in Python // Note: {{esphome_version}} is replaced at code generation time in Python