9#if ESP_IDF_VERSION >= ESP_IDF_VERSION_VAL(5, 1, 0)
10#include <esp_eap_client.h>
20#include <user_interface.h>
36#ifdef USE_CAPTIVE_PORTAL
44#ifdef USE_IMPROV_SERIAL
50static const char *
const TAG =
"wifi";
69 char *heap_data =
new char[
len + 1];
70 std::memcpy(heap_data, str,
len);
71 heap_data[
len] =
'\0';
88 std::memcpy(this->storage_, other.storage_, INLINE_CAPACITY + 1);
91 other.storage_[0] =
'\0';
109 return this->
size() == other.
size() && std::memcmp(this->
data(), other.
data(), this->size()) == 0;
112 return this->
size() == other.
size() && std::memcmp(this->
data(), other.
c_str(), this->size()) == 0;
311#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_INFO
313static const LogString *retry_phase_to_log_string(
WiFiRetryPhase phase) {
315 return LOG_STR(
"INITIAL_CONNECT");
316#ifdef USE_WIFI_FAST_CONNECT
318 return LOG_STR(
"FAST_CONNECT_CYCLING");
321 return LOG_STR(
"EXPLICIT_HIDDEN");
323 return LOG_STR(
"SCAN_CONNECTING");
325 return LOG_STR(
"RETRY_HIDDEN");
327 return LOG_STR(
"RESTARTING");
328 return LOG_STR(
"UNKNOWN");
335 return !this->
sta_.empty() && this->
sta_[0].get_hidden();
341 for (
size_t i = 0; i < this->
sta_.size(); i++) {
342 if (!this->
sta_[i].get_hidden()) {
343 return static_cast<int8_t
>(i);
354static constexpr uint8_t WIFI_RETRY_COUNT_PER_BSSID = 2;
358static constexpr uint8_t WIFI_RETRY_COUNT_PER_SSID = 1;
362static constexpr uint8_t WIFI_RETRY_COUNT_PER_AP = 1;
366static constexpr uint32_t WIFI_COOLDOWN_DURATION_MS = 500;
371static constexpr uint32_t WIFI_COOLDOWN_WITH_AP_ACTIVE_MS = 30000;
376static constexpr uint32_t WIFI_SCAN_TIMEOUT_MS = 31000;
386static constexpr uint32_t WIFI_CONNECT_TIMEOUT_MS = 46000;
388static constexpr uint8_t get_max_retries_for_phase(
WiFiRetryPhase phase) {
391#ifdef USE_WIFI_FAST_CONNECT
395 return WIFI_RETRY_COUNT_PER_AP;
398 return WIFI_RETRY_COUNT_PER_SSID;
401 return WIFI_RETRY_COUNT_PER_BSSID;
404 return WIFI_RETRY_COUNT_PER_SSID;
406 return WIFI_RETRY_COUNT_PER_BSSID;
410static void apply_scan_result_to_params(WiFiAP ¶ms,
const WiFiScanResult &scan) {
411 params.set_hidden(
false);
412 params.set_ssid(scan.get_ssid());
413 params.set_bssid(scan.get_bssid());
414 params.set_channel(scan.get_channel());
429 for (
const auto &conf : this->
sta_) {
430 if (conf.ssid_ == ssid && conf.get_hidden()) {
437 if (scan.ssid_ == ssid) {
451#ifdef USE_CAPTIVE_PORTAL
458#ifdef USE_IMPROV_SERIAL
477 if (ssid[0] ==
'\0') {
480 for (
const auto &sta : this->
sta_) {
482 if (sta.get_hidden()) {
486 if (sta.ssid_.empty()) {
487 if (sta.has_bssid() && std::memcmp(sta.get_bssid().data(), bssid, 6) == 0) {
493 if (sta.ssid_ == ssid) {
502 if (it.bssid == bssid) {
514#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
520 char bssid_s[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
522 ESP_LOGV(TAG,
"- " LOG_SECRET(
"'%s'")
" " LOG_SECRET(
"(%s)")
" %ddB Ch:%u", ssid, bssid_s, rssi, channel);
546 for (
size_t i = start_index + 1; i < this->
sta_.size(); i++) {
547 const auto &sta = this->
sta_[i];
552 if (!include_explicit_hidden && sta.get_hidden()) {
554 if (first_non_hidden_idx < 0 ||
static_cast<int8_t
>(i) < first_non_hidden_idx) {
555 ESP_LOGD(TAG,
"Skipping " LOG_SECRET(
"'%s'")
" (explicit hidden, already tried)", sta.ssid_.c_str());
563 ESP_LOGD(TAG,
"Hidden candidate " LOG_SECRET(
"'%s'")
" at index %d", sta.ssid_.c_str(),
static_cast<int>(i));
564 return static_cast<int8_t
>(i);
566 ESP_LOGD(TAG,
"Skipping hidden retry for visible network " LOG_SECRET(
"'%s'"), sta.ssid_.c_str());
575 if (!this->
sta_.empty() && this->sta_[0].get_hidden()) {
576 ESP_LOGI(TAG,
"Starting with explicit hidden network (highest priority)");
586#if defined(USE_ESP32) && defined(USE_WIFI_WPA2_EAP) && ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
587static const char *eap_phase2_to_str(esp_eap_ttls_phase2_types
type) {
589 case ESP_EAP_TTLS_PHASE2_PAP:
591 case ESP_EAP_TTLS_PHASE2_CHAP:
593 case ESP_EAP_TTLS_PHASE2_MSCHAP:
595 case ESP_EAP_TTLS_PHASE2_MSCHAPV2:
597 case ESP_EAP_TTLS_PHASE2_EAP:
610#if defined(USE_ESP32) && defined(USE_WIFI_RUNTIME_POWER_SAVE)
615 ESP_LOGE(TAG,
"Failed semaphore");
633 ESP_LOGCONFIG(TAG,
"Starting");
639#ifdef USE_WIFI_FAST_CONNECT
645 ESP_LOGD(TAG,
"Loaded settings: %s", save.ssid);
649 sta.set_password(save.password);
656 ESP_LOGV(TAG,
"Setting Output Power Option failed");
659#if defined(USE_ESP32) && defined(USE_WIFI_RUNTIME_POWER_SAVE)
663 if (semaphore_count > 0) {
673 ESP_LOGV(TAG,
"Setting Power Save Option failed");
677#ifdef USE_WIFI_FAST_CONNECT
682 if (loaded_fast_connect) {
683 ESP_LOGI(TAG,
"Starting fast_connect (saved) " LOG_SECRET(
"'%s'"), params.
ssid_.
c_str());
685 }
else if (!this->
sta_.empty() && !this->sta_[0].get_hidden()) {
687 ESP_LOGI(TAG,
"Starting fast_connect (config) " LOG_SECRET(
"'%s'"), this->
sta_[0].ssid_.c_str());
700 }
else if (this->
has_ap()) {
703 ESP_LOGV(TAG,
"Setting Output Power Option failed");
705#ifdef USE_CAPTIVE_PORTAL
724 ESP_LOGW(TAG,
"Restarting adapter");
740#if defined(USE_WIFI_CONNECT_TRIGGER) || defined(USE_WIFI_DISCONNECT_TRIGGER)
742#ifdef USE_WIFI_DISCONNECT_TRIGGER
747#ifdef USE_WIFI_CONNECT_TRIGGER
767 uint32_t cooldown_duration = portal_active ? WIFI_COOLDOWN_WITH_AP_ACTIVE_MS : WIFI_COOLDOWN_DURATION_MS;
791 ESP_LOGW(TAG,
"Connection lost; reconnecting");
823 ESP_LOGI(TAG,
"Starting fallback AP");
825#ifdef USE_CAPTIVE_PORTAL
850 ESP_LOGE(TAG,
"Can't connect; rebooting");
856#if defined(USE_ESP32) && defined(USE_WIFI_RUNTIME_POWER_SAVE)
864 ESP_LOGV(TAG,
"Switching to high-performance mode (%" PRIu32
" active %s)", (
uint32_t) semaphore_count,
865 semaphore_count == 1 ?
"request" :
"requests");
872 ESP_LOGV(TAG,
"Restoring power save mode to configured setting");
884#ifdef USE_WIFI_11KV_SUPPORT
915 static constexpr size_t AP_SSID_MAX_LEN = 32;
916 static constexpr size_t AP_SSID_PREFIX_LEN = 25;
917 static constexpr size_t AP_SSID_SUFFIX_LEN = 7;
920 const char *name_ptr = app_name.
c_str();
921 size_t name_len = app_name.length();
923 if (name_len <= AP_SSID_MAX_LEN) {
928 char ssid_buf[AP_SSID_MAX_LEN + 1];
931 memcpy(ssid_buf, name_ptr, AP_SSID_PREFIX_LEN);
932 memcpy(ssid_buf + AP_SSID_PREFIX_LEN, name_ptr + name_len - AP_SSID_SUFFIX_LEN, AP_SSID_SUFFIX_LEN);
934 memcpy(ssid_buf, name_ptr, AP_SSID_MAX_LEN);
936 ssid_buf[AP_SSID_MAX_LEN] =
'\0';
942 char ip_buf[network::IP_ADDRESS_BUFFER_SIZE];
946 " AP Password: '%s'\n"
948 this->
ap_.
ssid_.
c_str(), this->ap_.password_.c_str(), this->wifi_soft_ap_ip().str_to(ip_buf));
950#ifdef USE_WIFI_MANUAL_IP
952 if (manual_ip.has_value()) {
953 char static_ip_buf[network::IP_ADDRESS_BUFFER_SIZE];
954 char gateway_buf[network::IP_ADDRESS_BUFFER_SIZE];
955 char subnet_buf[network::IP_ADDRESS_BUFFER_SIZE];
957 " AP Static IP: '%s'\n"
958 " AP Gateway: '%s'\n"
960 manual_ip->static_ip.str_to(static_ip_buf), manual_ip->gateway.str_to(gateway_buf),
961 manual_ip->subnet.str_to(subnet_buf));
995 if (config ==
nullptr) {
996 ESP_LOGE(TAG,
"No valid network config (selected_sta_index_=%d, sta_.size()=%zu)",
1006#ifdef USE_WIFI_FAST_CONNECT
1024 apply_scan_result_to_params(params, this->
scan_result_[0]);
1038 return config ? *config :
WiFiAP{};
1045 strncpy(save.ssid, ssid,
sizeof(save.ssid) - 1);
1046 strncpy(save.password, password,
sizeof(save.password) - 1);
1053 sta.set_password(password);
1063 ESP_LOGD(TAG,
"Exiting cooldown early due to new WiFi credentials");
1079 "Connecting to " LOG_SECRET(
"'%s'")
" " LOG_SECRET(
"(%s)")
" (priority %d, attempt %u/%u in phase %s)...",
1083#ifdef ESPHOME_LOG_HAS_VERBOSE
1085 "Connection Params:\n"
1089 ESP_LOGV(TAG,
" BSSID: %s", bssid_s);
1091 ESP_LOGV(TAG,
" BSSID: Not Set");
1094#ifdef USE_WIFI_WPA2_EAP
1096 if (eap_opt.has_value()) {
1097 EAPAuth eap_config = *eap_opt;
1101 " WPA2 Enterprise authentication configured:\n"
1102 " Identity: " LOG_SECRET(
"'%s'")
"\n"
1103 " Username: " LOG_SECRET(
"'%s'")
"\n"
1104 " Password: " LOG_SECRET(
"'%s'"),
1107#if defined(USE_ESP32) && defined(USE_WIFI_WPA2_EAP) && ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
1108 ESP_LOGV(TAG,
" TTLS Phase 2: " LOG_SECRET(
"'%s'"), eap_phase2_to_str(eap_config.
ttls_phase_2));
1110 bool ca_cert_present = eap_config.
ca_cert !=
nullptr && strlen(eap_config.
ca_cert);
1115 " Client Cert: %s\n"
1117 ca_cert_present ?
"present" :
"not present", client_cert_present ?
"present" :
"not present",
1118 client_key_present ?
"present" :
"not present");
1121 ESP_LOGV(TAG,
" Password: " LOG_SECRET(
"'%s'"), ap.
password_.
c_str());
1122#ifdef USE_WIFI_WPA2_EAP
1128 ESP_LOGV(TAG,
" Channel not set");
1130#ifdef USE_WIFI_MANUAL_IP
1132 if (manual_ip.has_value()) {
1134 char static_ip_buf[network::IP_ADDRESS_BUFFER_SIZE];
1135 char gateway_buf[network::IP_ADDRESS_BUFFER_SIZE];
1136 char subnet_buf[network::IP_ADDRESS_BUFFER_SIZE];
1137 char dns1_buf[network::IP_ADDRESS_BUFFER_SIZE];
1138 char dns2_buf[network::IP_ADDRESS_BUFFER_SIZE];
1139 ESP_LOGV(TAG,
" Manual IP: Static IP=%s Gateway=%s Subnet=%s DNS1=%s DNS2=%s",
m.static_ip.str_to(static_ip_buf),
1140 m.gateway.str_to(gateway_buf),
m.subnet.str_to(subnet_buf),
m.dns1.str_to(dns1_buf),
1141 m.dns2.str_to(dns2_buf));
1145 ESP_LOGV(TAG,
" Using DHCP IP");
1147 ESP_LOGV(TAG,
" Hidden: %s", YESNO(ap.
get_hidden()));
1157 ESP_LOGE(TAG,
"wifi_sta_connect_ failed");
1177 return LOG_STR(
"\033[0;32m"
1183 }
else if (rssi >= -65) {
1184 return LOG_STR(
"\033[0;33m"
1191 }
else if (rssi >= -85) {
1192 return LOG_STR(
"\033[0;33m"
1200 return LOG_STR(
"\033[0;31m"
1212 char bssid_s[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
1215 char ip_buf[network::IP_ADDRESS_BUFFER_SIZE];
1218 ESP_LOGCONFIG(TAG,
" IP Address: %s", ip.str_to(ip_buf));
1223 char ssid_buf[SSID_BUFFER_SIZE];
1224 char subnet_buf[network::IP_ADDRESS_BUFFER_SIZE];
1225 char gateway_buf[network::IP_ADDRESS_BUFFER_SIZE];
1226 char dns1_buf[network::IP_ADDRESS_BUFFER_SIZE];
1227 char dns2_buf[network::IP_ADDRESS_BUFFER_SIZE];
1230 " SSID: " LOG_SECRET(
"'%s'")
"\n"
1231 " BSSID: " LOG_SECRET(
"%s")
"\n"
1233 " Signal strength: %d dB %s\n"
1234 " Channel: %" PRId32
"\n"
1243#ifdef ESPHOME_LOG_HAS_VERBOSE
1245 ESP_LOGV(TAG,
" Priority: %d", this->
get_sta_priority(config->get_bssid()));
1248#ifdef USE_WIFI_11KV_SUPPORT
1252 this->
btm_ ?
"enabled" :
"disabled", this->
rrm_ ?
"enabled" :
"disabled");
1260 ESP_LOGD(TAG,
"Enabling");
1269 ESP_LOGD(TAG,
"Disabling");
1279 ESP_LOGD(TAG,
"Starting scan");
1331 return a.
get_rssi() > b.get_rssi();
1357template<
typename VectorType>
static void insertion_sort_scan_results(VectorType &results) {
1363 static_assert(!std::is_polymorphic<WiFiScanResult>::value,
"WiFiScanResult must not have vtable");
1364 static_assert(!std::is_polymorphic<CompactString>::value,
"CompactString must not have vtable");
1367 static_assert(std::is_standard_layout<WiFiScanResult>::value,
"WiFiScanResult must be standard layout");
1368 static_assert(std::is_standard_layout<CompactString>::value,
"CompactString must be standard layout");
1370 static_assert(
sizeof(WiFiScanResult) == 32,
"WiFiScanResult size changed - verify memcpy sort is still safe");
1371 static_assert(
sizeof(CompactString) == 20,
"CompactString size changed - verify memcpy sort is still safe");
1373 static_assert(
alignof(WiFiScanResult) <=
alignof(std::max_align_t),
"WiFiScanResult alignment exceeds max_align_t");
1374 const size_t size = results.size();
1375 constexpr size_t elem_size =
sizeof(WiFiScanResult);
1379 auto *memcpy_fn = &memcpy;
1380 for (
size_t i = 1; i <
size; i++) {
1381 alignas(WiFiScanResult) uint8_t key_buf[
elem_size];
1382 memcpy_fn(key_buf, &results[i],
elem_size);
1383 const auto &key = *
reinterpret_cast<const WiFiScanResult *
>(key_buf);
1388 while (j >= 0 && wifi_scan_result_is_better(key, results[j])) {
1389 memcpy_fn(&results[j + 1], &results[j],
elem_size);
1392 memcpy_fn(&results[j + 1], key_buf,
elem_size);
1404 char bssid_s[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
1408#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_DEBUG
1410 ESP_LOGI(TAG,
"- '%s' %s" LOG_SECRET(
"(%s) ")
"%s Ch:%2u %3ddB P:%d", res.
get_ssid().
c_str(),
1411 res.
get_is_hidden() ? LOG_STR_LITERAL(
"(HIDDEN) ") : LOG_STR_LITERAL(
""), bssid_s,
1414 ESP_LOGI(TAG,
"- '%s' %s" LOG_SECRET(
"(%s) ")
"%s", res.
get_ssid().
c_str(),
1415 res.
get_is_hidden() ? LOG_STR_LITERAL(
"(HIDDEN) ") : LOG_STR_LITERAL(
""), bssid_s,
1423 ESP_LOGE(TAG,
"Scan timeout");
1434 ESP_LOGW(TAG,
"No networks found");
1439 ESP_LOGD(TAG,
"Found networks:");
1441 for (
auto &ap : this->
sta_) {
1442 if (res.matches(ap)) {
1443 res.set_matches(
true);
1445 const bssid_t &bssid = res.get_bssid();
1456 insertion_sort_scan_results(this->scan_result_);
1459 for (
auto &res : this->scan_result_) {
1460 if (res.get_matches()) {
1461 log_scan_result(res);
1470 bool found_match =
false;
1472 for (
size_t i = 0; i < this->
sta_.size(); i++) {
1473 if (scan_res.
matches(this->sta_[i])) {
1484 ESP_LOGW(TAG,
"No matching network found");
1506 char mac_s[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
1513 ESP_LOGCONFIG(TAG,
" Disabled");
1516#if defined(USE_ESP32) && defined(SOC_WIFI_SUPPORT_5G)
1517 const char *band_mode_s;
1519 case WIFI_BAND_MODE_2G_ONLY:
1520 band_mode_s =
"2.4GHz";
1522 case WIFI_BAND_MODE_5G_ONLY:
1523 band_mode_s =
"5GHz";
1525 case WIFI_BAND_MODE_AUTO:
1527 band_mode_s =
"Auto";
1530 ESP_LOGCONFIG(TAG,
" Band Mode: %s", band_mode_s);
1541 char ssid_buf[SSID_BUFFER_SIZE];
1543 ESP_LOGW(TAG,
"Connection incomplete");
1548 ESP_LOGI(TAG,
"Connected");
1554 ESP_LOGW(TAG, LOG_SECRET(
"'%s'")
" should be marked hidden", config->ssid_.c_str());
1560#ifdef USE_CAPTIVE_PORTAL
1565 ESP_LOGD(TAG,
"Disabling AP");
1584 ESP_LOGD(TAG,
"Roam successful");
1591 char bssid_buf[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
1615#ifdef USE_WIFI_FAST_CONNECT
1621#ifdef USE_WIFI_CONNECT_STATE_LISTENERS
1627#if defined(USE_ESP8266) && defined(USE_WIFI_IP_STATE_LISTENERS) && defined(USE_WIFI_MANUAL_IP)
1639 ESP_LOGW(TAG,
"Connection timeout, aborting connection attempt");
1649 ESP_LOGW(TAG,
"Connecting to network failed (callback)");
1660 ESP_LOGW(TAG,
"Network no longer found");
1666 ESP_LOGW(TAG,
"Connecting to network failed");
1671 ESP_LOGW(TAG,
"Unknown connection status %d", (
int)
status);
1684#ifdef USE_WIFI_FAST_CONNECT
1693 if (!this->
sta_.empty() && this->sta_[0].get_hidden()) {
1701 if (this->
num_retried_ + 1 < WIFI_RETRY_COUNT_PER_SSID) {
1708 if (next_index < this->
sta_.size() && this->sta_[next_index].get_hidden()) {
1724 if (this->
scan_result_.empty() || !this->scan_result_[0].get_matches()) {
1728 if (this->
num_retried_ + 1 < WIFI_RETRY_COUNT_PER_BSSID) {
1747 if (this->
num_retried_ + 1 < WIFI_RETRY_COUNT_PER_SSID) {
1824 if (old_phase == new_phase) {
1828 ESP_LOGD(TAG,
"Retry phase: %s → %s", LOG_STR_ARG(retry_phase_to_log_string(old_phase)),
1829 LOG_STR_ARG(retry_phase_to_log_string(new_phase)));
1835 switch (new_phase) {
1836#ifdef USE_WIFI_FAST_CONNECT
1851#ifdef USE_WIFI_FAST_CONNECT
1853 ESP_LOGI(TAG,
"Fast connect exhausted, falling back to scan");
1881 ESP_LOGD(TAG,
"All SSIDs visible or already tried, skipping hidden mode");
1934 if (first_priority != std::numeric_limits<int8_t>::min()) {
1939 if (pri.priority != first_priority) {
1945 ESP_LOGD(TAG,
"Clearing BSSID priorities (all at minimum)");
1969 optional<bssid_t> failed_bssid;
1976 failed_bssid = config->get_bssid();
1979 if (!failed_bssid.has_value()) {
1984 const char *ssid =
nullptr;
1988 ssid = config->ssid_.c_str();
1993 uint8_t max_retries = get_max_retries_for_phase(this->
retry_phase_);
1994 bool is_last_attempt = (this->
num_retried_ + 1 >= max_retries);
1998 int8_t new_priority = old_priority;
2000 if (is_last_attempt) {
2003 (old_priority > std::numeric_limits<int8_t>::min()) ? (old_priority - 1) : std::numeric_limits<int8_t>::min();
2008 ESP_LOGD(TAG,
"Failed " LOG_SECRET(
"'%s'")
" " LOG_SECRET(
"(%s)")
", priority %d → %d", ssid !=
nullptr ? ssid :
"",
2009 bssid_s, old_priority, new_priority);
2030#ifdef USE_WIFI_FAST_CONNECT
2035 ESP_LOGD(TAG,
"Next AP in %s", LOG_STR_ARG(retry_phase_to_log_string(this->
retry_phase_)));
2044 if (next_index < this->
sta_.size() && this->sta_[next_index].get_hidden()) {
2047 ESP_LOGD(TAG,
"Next explicit hidden network at index %d",
static_cast<int>(next_index));
2059 if (next_index != -1) {
2079 ESP_LOGD(TAG,
"Retry attempt %u/%u in phase %s", this->
num_retried_ + 1,
2120 if (next_phase == current_phase) {
2137#if defined(USE_ESP32) && defined(USE_WIFI_RUNTIME_POWER_SAVE)
2145#ifdef USE_CAPTIVE_PORTAL
2159#if defined(USE_ESP32) && defined(USE_WIFI_RUNTIME_POWER_SAVE)
2191#ifdef USE_WIFI_FAST_CONNECT
2197 if (fast_connect_save.ap_index < 0 ||
static_cast<size_t>(fast_connect_save.ap_index) >= this->sta_.size()) {
2198 ESP_LOGW(TAG,
"AP index out of bounds");
2206 params = this->
sta_[fast_connect_save.ap_index];
2210 std::copy(fast_connect_save.bssid, fast_connect_save.bssid + 6, bssid.begin());
2216 ESP_LOGD(TAG,
"Loaded fast_connect settings");
2217#if defined(USE_ESP32) && defined(SOC_WIFI_SUPPORT_5G)
2219 (this->band_mode_ == WIFI_BAND_MODE_2G_ONLY && fast_connect_save.channel >=
FIRST_5GHZ_CHANNEL)) {
2220 ESP_LOGW(TAG,
"Saved channel %u not allowed by band mode, ignoring fast_connect", fast_connect_save.channel);
2240 if (this->
fast_connect_pref_.
load(&previous_save) && memcmp(previous_save.bssid, bssid.data(), 6) == 0 &&
2241 previous_save.channel == channel && previous_save.ap_index == ap_index) {
2246 memcpy(fast_connect_save.bssid, bssid.data(), 6);
2247 fast_connect_save.channel = channel;
2248 fast_connect_save.ap_index = ap_index;
2252 ESP_LOGD(TAG,
"Saved fast_connect settings");
2264#ifdef USE_WIFI_WPA2_EAP
2269#ifdef USE_WIFI_MANUAL_IP
2275#ifdef USE_WIFI_WPA2_EAP
2278#ifdef USE_WIFI_MANUAL_IP
2284 bool with_auth,
bool is_hidden)
2288 ssid_(ssid, ssid_len),
2289 with_auth_(with_auth),
2290 is_hidden_(is_hidden) {}
2308#ifdef USE_WIFI_WPA2_EAP
2348#if defined(USE_RP2040) || defined(USE_ESP32)
2358#ifdef USE_WIFI_CONNECT_STATE_LISTENERS
2362 this->
pending_.connect_state =
false;
2364 char ssid_buf[SSID_BUFFER_SIZE];
2368 listener->on_wifi_connect_state(
StringRef(ssid, strlen(ssid)), bssid);
2373 constexpr uint8_t empty_bssid[6] = {};
2375 listener->on_wifi_connect_state(
StringRef(), empty_bssid);
2380#ifdef USE_WIFI_IP_STATE_LISTENERS
2388#ifdef USE_WIFI_SCAN_RESULTS_LISTENERS
2399 if (selected ==
nullptr || selected->
get_hidden()) {
2410 ESP_LOGD(TAG,
"Roam check skipped, signal good (%d dBm, attempt %u/%u)", rssi, this->
roaming_attempts_,
2429 int8_t current_rssi = this->
wifi_rssi();
2431 if (current_rssi == WIFI_RSSI_DISCONNECTED) {
2436 char ssid_buf[SSID_BUFFER_SIZE];
2442 char bssid_buf[MAC_ADDRESS_PRETTY_BUFFER_SIZE];
2446 if (result.ssid_ != current_ssid || result.get_bssid() == current_bssid)
2449#if ESPHOME_LOG_LEVEL >= ESPHOME_LOG_LEVEL_VERBOSE
2451 ESP_LOGV(TAG,
"Roam candidate %s %d dBm", bssid_buf, result.get_rssi());
2462 int8_t improvement = (best ==
nullptr) ? 0 : best->
get_rssi() - current_rssi;
2471 ESP_LOGI(TAG,
"Roaming to %s (%+d dB)", bssid_buf, improvement);
2473 WiFiAP roam_params = *selected;
2474 apply_scan_result_to_params(roam_params, *best);
const StringRef & get_name() const
Get the name of this Application set by pre_setup().
bool is_name_add_mac_suffix_enabled() const
uint32_t get_config_version_hash()
Get the config hash extended with ESPHome version.
uint32_t IRAM_ATTR HOT get_loop_component_start_time() const
Get the cached time in milliseconds from when the current component started its loop execution.
void status_set_warning()
void status_clear_warning()
StringRef is a reference to a string owned by something else.
constexpr const char * c_str() const
constexpr size_type size() const
void trigger(const Ts &...x) ESPHOME_ALWAYS_INLINE
Inform the parent automation that the event has triggered.
20-byte string: 18 chars inline + null, heap for longer.
const char * data() const
CompactString & operator=(const CompactString &other)
bool operator==(const CompactString &other) const
static constexpr uint8_t INLINE_CAPACITY
const char * c_str() const
char storage_[INLINE_CAPACITY+1]
static constexpr uint8_t MAX_LENGTH
void set_heap_ptr_(char *ptr)
char * get_heap_ptr_() const
uint8_t get_channel() const
void set_ssid(const std::string &ssid)
const optional< EAPAuth > & get_eap() const
void set_bssid(const bssid_t &bssid)
void set_channel(uint8_t channel)
optional< ManualIP > manual_ip_
void set_eap(optional< EAPAuth > eap_auth)
void set_password(const std::string &password)
void set_manual_ip(optional< ManualIP > manual_ip)
const optional< ManualIP > & get_manual_ip() const
void set_hidden(bool hidden)
const bssid_t & get_bssid() const
This component is responsible for managing the ESP WiFi interface.
void notify_scan_results_listeners_()
Notify scan results listeners with current scan results.
bool is_captive_portal_active_()
bool error_from_callback_
void add_sta(const WiFiAP &ap)
bool load_fast_connect_settings_(WiFiAP ¶ms)
bool wifi_apply_power_save_()
void set_ap(const WiFiAP &ap)
Setup an Access Point that should be created if no connection to a station can be made.
bool request_high_performance()
Request high-performance mode (no power saving) for improved WiFi latency.
ESPPreferenceObject pref_
void set_sta(const WiFiAP &ap)
WiFiPowerSaveMode power_save_
bool has_sta_priority(const bssid_t &bssid)
const WiFiAP * get_selected_sta_() const
WiFiSTAConnectStatus wifi_sta_connect_status_() const
uint32_t roaming_last_check_
int8_t get_sta_priority(const bssid_t bssid)
void log_and_adjust_priority_for_failed_connect_()
Log failed connection and decrease BSSID priority to avoid repeated attempts.
void init_sta(size_t count)
void save_wifi_sta(const std::string &ssid, const std::string &password)
void notify_connect_state_listeners_()
Notify connect state listeners (called after state machine reaches STA_CONNECTED)
WiFiComponentState state_
wifi_scan_vector_t< WiFiScanResult > scan_result_
bool skip_cooldown_next_cycle_
WiFiPowerSaveMode configured_power_save_
struct esphome::wifi::WiFiComponent::@190 pending_
void set_sta_priority(bssid_t bssid, int8_t priority)
StaticVector< WiFiScanResultsListener *, ESPHOME_WIFI_SCAN_RESULTS_LISTENERS > scan_results_listeners_
void print_connect_params_()
void loop() override
Reconnect WiFi if required.
void notify_ip_state_listeners_()
Notify IP state listeners with current addresses.
void start_connecting(const WiFiAP &ap)
void advance_to_next_target_or_increment_retry_()
Advance to next target (AP/SSID) within current phase, or increment retry counter Called when staying...
void check_roaming_(uint32_t now)
static constexpr uint32_t ROAMING_CHECK_INTERVAL
RoamingState roaming_state_
void check_scanning_finished()
void process_roaming_scan_()
SemaphoreHandle_t high_performance_semaphore_
network::IPAddress get_dns_address(int num)
WiFiComponent()
Construct a WiFiComponent.
wifi_band_mode_t band_mode_
std::vector< WiFiSTAPriority > sta_priorities_
bssid_t roaming_target_bssid_
int8_t selected_sta_index_
static constexpr int8_t ROAMING_GOOD_RSSI
void save_fast_connect_settings_()
void notify_disconnect_state_listeners_()
Notify connect state listeners of disconnection.
bool wifi_start_ap_(const WiFiAP &ap)
StaticVector< WiFiConnectStateListener *, ESPHOME_WIFI_CONNECT_STATE_LISTENERS > connect_state_listeners_
void log_discarded_scan_result_(const char *ssid, const uint8_t *bssid, int8_t rssi, uint8_t channel)
Log a discarded scan result at VERBOSE level (skipped during roaming scans to avoid log overflow)
int32_t get_wifi_channel()
ESPDEPRECATED("Use wifi_ssid_to() instead. Removed in 2026.9.0", "2026.3.0") std const char * wifi_ssid_to(std::span< char, SSID_BUFFER_SIZE > buffer)
Write SSID to buffer without heap allocation.
network::IPAddress wifi_subnet_mask_()
static constexpr uint8_t ROAMING_MAX_ATTEMPTS
network::IPAddress wifi_soft_ap_ip()
void set_passive_scan(bool passive)
uint8_t roaming_attempts_
network::IPAddress wifi_gateway_ip_()
void set_power_save_mode(WiFiPowerSaveMode power_save)
int8_t find_next_hidden_sta_(int8_t start_index)
Find next SSID that wasn't in scan results (might be hidden) Returns index of next potentially hidden...
ESPPreferenceObject fast_connect_pref_
void clear_priorities_if_all_min_()
Clear BSSID priority tracking if all priorities are at minimum (saves memory)
static constexpr uint32_t ROAMING_SCAN_GRACE_PERIOD
WiFiRetryPhase determine_next_phase_()
Determine next retry phase based on current state and failure conditions.
network::IPAddress wifi_dns_ip_(int num)
bool wifi_apply_hostname_()
bool wifi_sta_pre_setup_()
network::IPAddresses get_ip_addresses()
static constexpr int8_t ROAMING_MIN_IMPROVEMENT
bool matches_configured_network_(const char *ssid, const uint8_t *bssid) const
Check if network matches any configured network (for scan result filtering) Matches by SSID when conf...
float get_setup_priority() const override
WIFI setup_priority.
bool is_esp32_improv_active_()
StaticVector< WiFiIPStateListener *, ESPHOME_WIFI_IP_STATE_LISTENERS > ip_state_listeners_
FixedVector< WiFiAP > sta_
int8_t find_first_non_hidden_index_() const
Find the index of the first non-hidden network Returns where EXPLICIT_HIDDEN phase would have stopped...
void release_scan_results_()
Free scan results memory unless a component needs them.
bool needs_scan_results_() const
Check if we need valid scan results for the current phase but don't have any Returns true if the phas...
void update_connected_state_()
WiFiRetryPhase retry_phase_
bool transition_to_phase_(WiFiRetryPhase new_phase)
Transition to a new retry phase with logging Returns true if a scan was started (caller should wait),...
bool needs_full_scan_results_() const
Check if full scan results are needed (captive portal active, improv, listeners)
WiFiAP build_params_for_current_phase_()
bool is_high_performance_mode_
void dump_config() override
static constexpr uint8_t FIRST_5GHZ_CHANNEL
void clear_roaming_state_()
bool release_high_performance()
Release a high-performance mode request.
RetryHiddenMode retry_hidden_mode_
bool wifi_apply_output_power_(float output_power)
bool has_completed_scan_after_captive_portal_start_
bool is_connected() const
bool wifi_sta_connect_(const WiFiAP &ap)
bool went_through_explicit_hidden_phase_() const
Check if we went through EXPLICIT_HIDDEN phase (first network is marked hidden) Used in RETRY_HIDDEN ...
bool wifi_mode_(optional< bool > sta, optional< bool > ap)
uint32_t roaming_scan_end_
void set_reboot_timeout(uint32_t reboot_timeout)
network::IPAddresses wifi_sta_ip_addresses()
void check_connecting_finished(uint32_t now)
void start_initial_connection_()
Start initial connection - either scan or connect directly to hidden networks.
bool wifi_scan_start_(bool passive)
bool ssid_was_seen_in_scan_(const CompactString &ssid) const
Check if an SSID was seen in the most recent scan results Used to skip hidden mode for SSIDs we know ...
bool handled_connected_state_
void setup() override
Setup WiFi interface.
void clear_all_bssid_priorities_()
Clear all BSSID priority penalties after successful connection (stale after disconnect)
bool post_connect_roaming_
Trigger disconnect_trigger_
int8_t get_priority() const
uint8_t get_channel() const
bool get_with_auth() const
WiFiScanResult(const bssid_t &bssid, const char *ssid, size_t ssid_len, uint8_t channel, int8_t rssi, bool with_auth, bool is_hidden)
const bssid_t & get_bssid() const
StringRef get_ssid() const
bool matches(const WiFiAP &config) const
void set_matches(bool matches)
bool get_is_hidden() const
bool operator==(const WiFiScanResult &rhs) const
struct @65::@66 __attribute__
Wake the main loop task from an ISR. ISR-safe.
CaptivePortal * global_captive_portal
ESP32ImprovComponent * global_improv_component
ImprovSerialComponent * global_improv_serial_component
std::array< IPAddress, 5 > IPAddresses
std::array< uint8_t, 6 > bssid_t
const LogString * get_signal_bars(int8_t rssi)
@ BLIND_RETRY
Blind retry mode: scanning disabled (captive portal/improv active), try ALL configured networks seque...
@ SCAN_BASED
Normal mode: scan completed, only try networks NOT visible in scan results (truly hidden networks tha...
@ ERROR_NETWORK_NOT_FOUND
WiFiRetryPhase
Tracks the current retry strategy/phase for WiFi connection attempts.
@ RETRY_HIDDEN
Retry networks not found in scan (might be hidden)
@ RESTARTING_ADAPTER
Restarting WiFi adapter to clear stuck state.
@ INITIAL_CONNECT
Initial connection attempt (varies based on fast_connect setting)
@ EXPLICIT_HIDDEN
Explicitly hidden networks (user marked as hidden, try before scanning)
@ FAST_CONNECT_CYCLING_APS
Fast connect mode: cycling through configured APs (config-only, no scan)
@ SCAN_CONNECTING
Scan-based: connecting to best AP from scan results.
WiFiComponent * global_wifi_component
@ SCANNING
Scanning for better AP.
@ CONNECTING
Attempting to connect to better AP found in scan.
@ IDLE
Not roaming, waiting for next check interval.
@ RECONNECTING
Roam connection failed, reconnecting to any available AP.
@ WIFI_COMPONENT_STATE_DISABLED
WiFi is disabled.
@ WIFI_COMPONENT_STATE_AP
WiFi is in AP-only mode and internal AP is already enabled.
@ WIFI_COMPONENT_STATE_STA_CONNECTING
WiFi is in STA(+AP) mode and currently connecting to an AP.
@ WIFI_COMPONENT_STATE_OFF
Nothing has been initialized yet.
@ WIFI_COMPONENT_STATE_STA_SCANNING
WiFi is in STA-only mode and currently scanning for APs.
@ WIFI_COMPONENT_STATE_COOLDOWN
WiFi is in cooldown mode because something went wrong, scanning will begin after a short period of ti...
@ WIFI_COMPONENT_STATE_STA_CONNECTED
WiFi is in STA(+AP) mode and successfully connected.
uint16_t uint16_t size_t elem_size
ESPPreferences * global_preferences
const char * get_mac_address_pretty_into_buffer(std::span< char, MAC_ADDRESS_PRETTY_BUFFER_SIZE > buf)
Get the device MAC address into the given buffer, in colon-separated uppercase hex notation.
uint32_t IRAM_ATTR HOT millis()
Application App
Global storage of Application pointer - only one Application can exist.
char * format_mac_addr_upper(const uint8_t *mac, char *output)
Format MAC address as XX:XX:XX:XX:XX:XX (uppercase, colon separators)
ESPPreferenceObject make_preference(size_t, uint32_t, bool)
bool sync()
Commit pending writes to flash.
esp_eap_ttls_phase2_types ttls_phase_2
Struct for setting static IPs in WiFiComponent.