WifiList.cpp 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845
  1. #define LOG_LOCAL_LEVEL ESP_LOG_DEBUG
  2. #include "WifiList.h"
  3. #include "Config.h"
  4. #include "esp_check.h"
  5. #include "esp_log.h"
  6. #include "esp_system.h"
  7. #include <memory>
  8. static const char* TAG_CRED_MANAGER = "credentials_manager";
  9. bool sys_status_wifi_callback(pb_istream_t* istream, pb_ostream_t* ostream, const pb_field_iter_t* field) {
  10. return sys_net_config_callback(istream, ostream, field);
  11. }
  12. bool sys_net_config_callback(pb_istream_t* istream, pb_ostream_t* ostream, const pb_field_iter_t* field) {
  13. WifiList** managerPtr = static_cast<WifiList**>(field->pData);
  14. WifiList* manager = *managerPtr;
  15. if (istream != NULL && (field->tag == sys_net_config_credentials_tag || field->tag == sys_status_wifi_scan_result_tag)) {
  16. if (manager == nullptr) {
  17. ESP_LOGE(TAG_CRED_MANAGER, "Invalid pointer to wifi list manager");
  18. return false;
  19. }
  20. ESP_LOGV(TAG_CRED_MANAGER, "Decoding credentials");
  21. sys_net_wifi_entry entry = sys_net_wifi_entry_init_default;
  22. if (!pb_decode(istream, &sys_net_wifi_entry_msg, &entry)) return false;
  23. printf("\nFound ssid %s, password %s\n", entry.ssid, entry.password);
  24. try {
  25. manager->AddUpdate(entry); // Add to the manager
  26. } catch (const std::exception& e) {
  27. ESP_LOGE(TAG_CRED_MANAGER, "decode exception: %s", e.what());
  28. return false;
  29. }
  30. ESP_LOGV(TAG_CRED_MANAGER, "Credentials decoding completed");
  31. } else if (ostream != NULL && (field->tag == sys_net_config_credentials_tag || field->tag == sys_status_wifi_scan_result_tag)) {
  32. if (manager == nullptr) {
  33. ESP_LOGV(TAG_CRED_MANAGER, "No wifi entries manager instance. nothing to encode");
  34. return true;
  35. }
  36. ESP_LOGV(TAG_CRED_MANAGER, "Encoding %d access points", manager->GetCount());
  37. for (int i = 0; i < manager->GetCount(); i++) {
  38. ESP_LOGV(TAG_CRED_MANAGER, "Encoding credential #%d: SSID: %s, PASS: %s", i, manager->GetIndex(i)->ssid, manager->GetIndex(i)->password);
  39. if (!pb_encode_tag_for_field(ostream, field)) {
  40. return false;
  41. }
  42. if (!pb_encode_submessage(ostream, &sys_net_wifi_entry_msg, manager->GetIndex(i))) {
  43. return false;
  44. }
  45. }
  46. ESP_LOGV(TAG_CRED_MANAGER, "Credentials encoding completed");
  47. }
  48. return true;
  49. }
  50. std::string WifiList::GetBSSID(const wifi_event_sta_connected_t* evt) {
  51. char buffer[18]={};
  52. FormatBSSID(buffer, sizeof(buffer), evt->bssid);
  53. ESP_LOGD(TAG_CRED_MANAGER, "Formatted BSSID: %s", buffer);
  54. return std::string(buffer);
  55. }
  56. bool WifiList::OffsetTimeStamp(google_protobuf_Timestamp* ts) {
  57. timeval tts;
  58. google_protobuf_Timestamp gts;
  59. gettimeofday((struct timeval*)&tts, NULL);
  60. gts.nanos = tts.tv_usec * 1000;
  61. gts.seconds = tts.tv_sec;
  62. if (tts.tv_sec < 1704143717) {
  63. ESP_LOGE(TAG_CRED_MANAGER, "Error updating time stamp. Clock doesn't seem right");
  64. return false;
  65. }
  66. if (ts && ts->seconds < 1704143717) {
  67. ESP_LOGV(TAG_CRED_MANAGER, "Updating time stamp based on new clock value");
  68. ts->seconds = gts.seconds - ts->seconds;
  69. ts->nanos = gts.nanos - ts->nanos;
  70. return true;
  71. }
  72. ESP_LOGD(TAG_CRED_MANAGER, "Time stamp already updated. Skipping");
  73. return false;
  74. }
  75. bool WifiList::UpdateTimeStamp(google_protobuf_Timestamp* ts, bool& has_flag_val) {
  76. ESP_RETURN_ON_FALSE(ts != nullptr, false, TAG_CRED_MANAGER, "Null pointer!");
  77. bool changed = false;
  78. timeval tts;
  79. google_protobuf_Timestamp gts;
  80. gettimeofday((struct timeval*)&tts, NULL);
  81. gts.nanos = tts.tv_usec * 1000;
  82. gts.seconds = tts.tv_sec;
  83. if (!has_flag_val || gts.nanos != ts->nanos || gts.seconds != ts->seconds) {
  84. ts->seconds = gts.seconds;
  85. ts->nanos = gts.nanos;
  86. has_flag_val = true;
  87. changed = true;
  88. }
  89. return changed;
  90. }
  91. bool WifiList::isEmpty(const char* str, size_t len) {
  92. for (size_t i = 0; i < len; ++i) {
  93. if (str[i] != '\0') {
  94. return false;
  95. }
  96. }
  97. return true;
  98. }
  99. bool WifiList::Update(const wifi_ap_record_t* ap, bool connected) {
  100. if (!Lock()) {
  101. throw std::runtime_error("Lock failed");
  102. }
  103. auto existing = Get(ap);
  104. if (!existing) {
  105. return false;
  106. }
  107. auto updated = ToSTAEntry(ap);
  108. updated.connected = connected;
  109. bool changed = Update(*existing, updated);
  110. Release(&updated);
  111. Unlock();
  112. return changed;
  113. }
  114. bool WifiList::Update(sys_net_wifi_entry& existingEntry, sys_net_wifi_entry& updated) {
  115. // Check if any relevant fields have changed
  116. bool hasChanged = false;
  117. if (!isEmpty(updated.ssid, sizeof(updated.ssid)) && memcmp(existingEntry.ssid, updated.ssid, sizeof(existingEntry.ssid)) != 0) {
  118. memcpy(existingEntry.ssid, updated.ssid, sizeof(existingEntry.ssid));
  119. hasChanged = true;
  120. }
  121. // Check and copy BSSID if the compared BSSID is not empty
  122. if (!isEmpty(updated.bssid, sizeof(updated.bssid)) && strcmp(updated.bssid, "00:00:00:00:00:00") != 0 &&
  123. memcmp(existingEntry.bssid, updated.bssid, sizeof(existingEntry.bssid)) != 0) {
  124. memcpy(existingEntry.bssid, updated.bssid, sizeof(existingEntry.bssid));
  125. hasChanged = true;
  126. }
  127. // Check and copy password if the compared password is not empty
  128. if (!isEmpty(updated.password, sizeof(updated.password)) &&
  129. memcmp(existingEntry.password, updated.password, sizeof(existingEntry.password)) != 0) {
  130. memcpy(existingEntry.password, updated.password, sizeof(existingEntry.password));
  131. hasChanged = true;
  132. }
  133. if (existingEntry.channel != updated.channel && updated.channel > 0) {
  134. existingEntry.channel = updated.channel;
  135. hasChanged = true;
  136. }
  137. if (existingEntry.auth_type != updated.auth_type && updated.auth_type != sys_net_auth_types_AUTH_UNKNOWN) {
  138. existingEntry.auth_type = updated.auth_type;
  139. hasChanged = true;
  140. }
  141. if (areRadioTypesDifferent(existingEntry.radio_type, existingEntry.radio_type_count, updated.radio_type, updated.radio_type_count) &&
  142. updated.radio_type_count > 0 && updated.radio_type[0] != sys_net_radio_types_UNKNOWN) {
  143. if (existingEntry.radio_type != nullptr) {
  144. // Free the old radio_type array if it exists
  145. delete[] existingEntry.radio_type;
  146. }
  147. // Allocate new memory and copy the updated radio types
  148. existingEntry.radio_type_count = updated.radio_type_count;
  149. existingEntry.radio_type = new sys_net_radio_types[updated.radio_type_count];
  150. std::copy(updated.radio_type, updated.radio_type + updated.radio_type_count, existingEntry.radio_type);
  151. hasChanged = true;
  152. }
  153. if (updated.has_last_try) {
  154. if (memcmp(&existingEntry.last_try, &updated.last_try, sizeof(existingEntry.last_try)) != 0) {
  155. memcpy(&existingEntry.last_try, &updated.last_try, sizeof(existingEntry.last_try));
  156. hasChanged = true;
  157. }
  158. }
  159. if (updated.has_last_seen) {
  160. if (memcmp(&existingEntry.last_seen, &updated.last_seen, sizeof(existingEntry.last_seen)) != 0) {
  161. memcpy(&existingEntry.last_seen, &updated.last_seen, sizeof(existingEntry.last_seen));
  162. hasChanged = true;
  163. }
  164. }
  165. if (existingEntry.has_last_seen != updated.has_last_seen && updated.has_last_seen) {
  166. existingEntry.has_last_seen = updated.has_last_seen;
  167. hasChanged = true;
  168. }
  169. if (existingEntry.has_last_try != updated.has_last_try && updated.has_last_try) {
  170. existingEntry.has_last_try = updated.has_last_try;
  171. hasChanged = true;
  172. }
  173. if (existingEntry.connected != updated.connected && updated.connected) {
  174. existingEntry.connected = updated.connected;
  175. hasChanged = true;
  176. }
  177. if (existingEntry.rssi != updated.rssi && updated.rssi != 0) {
  178. existingEntry.rssi = updated.rssi;
  179. hasChanged = true;
  180. }
  181. return hasChanged;
  182. }
  183. std::string WifiList::formatRadioTypes(const sys_net_radio_types* radioTypes, pb_size_t count) {
  184. std::string result;
  185. for (pb_size_t i = 0; i < count; ++i) {
  186. switch (radioTypes[i]) {
  187. case sys_net_radio_types_PHY_11B:
  188. result += "B";
  189. break;
  190. case sys_net_radio_types_PHY_11G:
  191. result += "G";
  192. break;
  193. case sys_net_radio_types_PHY_11N:
  194. result += "N";
  195. break;
  196. case sys_net_radio_types_LR:
  197. result += "L";
  198. break;
  199. case sys_net_radio_types_WPS:
  200. result += "W";
  201. break;
  202. case sys_net_radio_types_FTM_RESPONDER:
  203. result += "FR";
  204. break;
  205. case sys_net_radio_types_FTM_INITIATOR:
  206. result += "FI";
  207. break;
  208. case sys_net_radio_types_UNKNOWN:
  209. default:
  210. result += "U";
  211. break;
  212. }
  213. if (i < count - 1) {
  214. result += ",";
  215. }
  216. }
  217. return result;
  218. }
  219. bool WifiList::Update(const wifi_sta_config_t* sta, bool connected) {
  220. if (!sta) {
  221. return false; // Invalid input
  222. }
  223. if (!Lock()) {
  224. throw std::runtime_error("Lock failed");
  225. }
  226. sys_net_wifi_entry* existingEntry = Get(sta);
  227. // If the entry does not exist, nothing to update
  228. if (!existingEntry) {
  229. Unlock();
  230. return false;
  231. }
  232. auto updated = ToSTAEntry(sta);
  233. // Check if any relevant fields have changed
  234. bool hasChanged = false;
  235. if (strlen(updated.ssid) > 0 && memcmp(existingEntry->ssid, updated.ssid, sizeof(existingEntry->ssid)) != 0) {
  236. memcpy(existingEntry->ssid, updated.ssid, sizeof(existingEntry->ssid));
  237. hasChanged = true;
  238. }
  239. if (strlen(updated.bssid) > 0 && strcmp(updated.bssid, "00:00:00:00:00:00") != 0 &&
  240. memcmp(existingEntry->bssid, updated.bssid, sizeof(existingEntry->bssid)) != 0) {
  241. memcpy(existingEntry->bssid, updated.bssid, sizeof(existingEntry->bssid));
  242. hasChanged = true;
  243. }
  244. if (existingEntry->channel != updated.channel) {
  245. existingEntry->channel = updated.channel;
  246. hasChanged = true;
  247. }
  248. if (existingEntry->auth_type != updated.auth_type && updated.auth_type != sys_net_auth_types_AUTH_UNKNOWN) {
  249. existingEntry->auth_type = updated.auth_type;
  250. hasChanged = true;
  251. }
  252. if (areRadioTypesDifferent(existingEntry->radio_type, existingEntry->radio_type_count, updated.radio_type, updated.radio_type_count) &&
  253. updated.radio_type_count > 0 && updated.radio_type[0] != sys_net_radio_types_UNKNOWN) {
  254. // Free the old radio_type array if it exists
  255. delete[] existingEntry->radio_type;
  256. // Allocate new memory and copy the updated radio types
  257. existingEntry->radio_type_count = updated.radio_type_count;
  258. existingEntry->radio_type = new sys_net_radio_types[updated.radio_type_count];
  259. std::copy(updated.radio_type, updated.radio_type + updated.radio_type_count, existingEntry->radio_type);
  260. hasChanged = true;
  261. }
  262. if (updated.has_last_try) {
  263. if (memcmp(&existingEntry->last_try, &updated.last_try, sizeof(existingEntry->last_try)) != 0) {
  264. memcpy(&existingEntry->last_try, &updated.last_try, sizeof(existingEntry->last_try));
  265. hasChanged = true;
  266. }
  267. }
  268. if (updated.has_last_seen) {
  269. if (memcmp(&existingEntry->last_seen, &updated.last_seen, sizeof(existingEntry->last_seen)) != 0) {
  270. memcpy(&existingEntry->last_seen, &updated.last_seen, sizeof(existingEntry->last_seen));
  271. hasChanged = true;
  272. }
  273. }
  274. if (existingEntry->has_last_try != updated.has_last_try) {
  275. existingEntry->has_last_try = updated.has_last_try;
  276. hasChanged = true;
  277. }
  278. if (existingEntry->has_last_seen != updated.has_last_seen) {
  279. existingEntry->has_last_seen = updated.has_last_seen;
  280. hasChanged = true;
  281. }
  282. if (existingEntry->connected != (connected | updated.connected)) {
  283. existingEntry->connected = connected | updated.connected;
  284. hasChanged = true;
  285. }
  286. if (strlen(updated.password) == 0 && strlen(existingEntry->password) > 0) {
  287. ESP_LOGW(TAG_CRED_MANAGER, "Updated password is empty, while existing password is %s for %s. Ignoring.", existingEntry->password,
  288. existingEntry->ssid);
  289. } else {
  290. if (memcmp(existingEntry->password, updated.password, sizeof(existingEntry->password)) != 0) {
  291. memcpy(existingEntry->password, updated.password, sizeof(existingEntry->password));
  292. hasChanged = true;
  293. }
  294. }
  295. if (existingEntry->rssi != updated.rssi && updated.rssi != 0) {
  296. existingEntry->rssi = updated.rssi;
  297. hasChanged = true;
  298. }
  299. Release(&updated);
  300. Unlock();
  301. return hasChanged;
  302. }
  303. sys_net_wifi_entry WifiList::ToSTAEntry(const sys_net_wifi_entry* sta) {
  304. if (!sta) {
  305. throw std::runtime_error("Null STA entry provided");
  306. }
  307. sys_net_wifi_entry result = *sta;
  308. if (result.radio_type_count > 0) {
  309. std::unique_ptr<sys_net_radio_types[]> newRadioTypes(new sys_net_radio_types[result.radio_type_count]);
  310. if (!newRadioTypes) {
  311. throw std::runtime_error("Failed to allocate memory for radio types");
  312. }
  313. memcpy(newRadioTypes.get(), sta->radio_type, sizeof(sys_net_radio_types) * result.radio_type_count);
  314. result.radio_type = newRadioTypes.release();
  315. } else {
  316. result.radio_type = nullptr;
  317. }
  318. ESP_LOGD(TAG_CRED_MANAGER, "ToSTAEntry: SSID: %s, PASS: %s", result.ssid, result.password);
  319. return result;
  320. }
  321. sys_net_wifi_entry WifiList::ToSTAEntry(const wifi_sta_config_t* sta, sys_net_auth_types auth_type) {
  322. return ToSTAEntry(sta, GetRadioTypes(nullptr), auth_type);
  323. }
  324. sys_net_wifi_entry WifiList::ToSTAEntry(
  325. const wifi_sta_config_t* sta, const std::list<sys_net_radio_types>& radio_types, sys_net_auth_types auth_type) {
  326. sys_net_wifi_entry item = sys_net_wifi_entry_init_default;
  327. ESP_LOGD(TAG_CRED_MANAGER,"%s (sta_config)","toSTAEntry");
  328. auto result = ToSTAEntry(sta, item, radio_types);
  329. ESP_LOGV(TAG_CRED_MANAGER, "ToSTAEntry: SSID: %s, PASS: %s", result.ssid, result.password);
  330. return result;
  331. }
  332. sys_net_wifi_entry& WifiList::ToSTAEntry(const wifi_ap_record_t* ap, sys_net_wifi_entry& item) {
  333. if (ap) {
  334. auto radioTypes = GetRadioTypes(ap);
  335. item.radio_type_count=radioTypes.size();
  336. item.radio_type = new sys_net_radio_types[item.radio_type_count];
  337. int i = 0;
  338. for (const auto& type : radioTypes) {
  339. item.radio_type[i++] = type;
  340. }
  341. item.auth_type = GetAuthType(ap);
  342. FormatBSSID(ap, item);
  343. item.channel = ap->primary;
  344. item.rssi = ap->rssi;
  345. strncpy(item.ssid, GetSSID(ap).c_str(), sizeof(item.ssid));
  346. }
  347. return item;
  348. }
  349. sys_net_wifi_entry WifiList::ToSTAEntry(const wifi_ap_record_t* ap) {
  350. sys_net_wifi_entry item = sys_net_wifi_entry_init_default;
  351. return ToSTAEntry(ap, item);
  352. }
  353. sys_net_wifi_entry& WifiList::ToSTAEntry(
  354. const wifi_sta_config_t* sta, sys_net_wifi_entry& item, const std::list<sys_net_radio_types>& radio_types, sys_net_auth_types auth_type) {
  355. if (!sta) {
  356. ESP_LOGE(TAG_CRED_MANAGER, "Invalid access point entry");
  357. return item;
  358. }
  359. std::string ssid = GetSSID(sta); // Convert SSID to std::string
  360. std::string password = GetPassword(sta); // Convert password to std::string
  361. if (ssid.empty()) {
  362. ESP_LOGE(TAG_CRED_MANAGER, "Invalid access point ssid");
  363. return item;
  364. }
  365. memset(item.ssid, 0x00, sizeof(item.ssid));
  366. memset(item.password, 0x00, sizeof(item.password));
  367. strncpy(item.ssid, ssid.c_str(), sizeof(item.ssid)); // Copy SSID
  368. strncpy(item.password, password.c_str(), sizeof(item.password)); // Copy password
  369. if (LOG_LOCAL_LEVEL > ESP_LOG_DEBUG) {
  370. WifiList::FormatBSSID(item.bssid, sizeof(item.bssid), sta->bssid); // Format BSSID
  371. }
  372. item.channel = sta->channel;
  373. item.auth_type = auth_type;
  374. // Handle the radio_type array
  375. if (item.radio_type != nullptr) {
  376. delete[] item.radio_type; // Free existing array if any
  377. item.radio_type_count = 0;
  378. }
  379. item.radio_type_count = radio_types.size();
  380. item.radio_type = new sys_net_radio_types[item.radio_type_count];
  381. int i = 0;
  382. for (const auto& type : radio_types) {
  383. item.radio_type[i++] = type;
  384. }
  385. ESP_LOGV(TAG_CRED_MANAGER, "ToSTAEntry wifi : %s, password: %s", item.ssid, item.password);
  386. return item;
  387. }
  388. bool WifiList::RemoveCredential(const wifi_sta_config_t* sta) { return RemoveCredential(GetSSID(sta)); }
  389. bool WifiList::RemoveCredential(const std::string& ssid) {
  390. auto it = credentials_.find(ssid);
  391. if (it != credentials_.end()) {
  392. // Release any dynamically allocated fields in the structure
  393. Release(&it->second);
  394. // Erase the entry from the map
  395. credentials_.erase(it);
  396. return true;
  397. }
  398. return false;
  399. }
  400. void WifiList::Clear() {
  401. if (Lock()) {
  402. for (auto& e : credentials_) {
  403. Release( &e.second);
  404. }
  405. credentials_.clear();
  406. Unlock();
  407. }
  408. }
  409. bool WifiList::ResetRSSI() {
  410. if (!Lock()) {
  411. ESP_LOGE(TAG_CRED_MANAGER, "Unable to lock structure %s", name_.c_str());
  412. return false;
  413. }
  414. for (auto& e : credentials_) {
  415. e.second.rssi = 0;
  416. }
  417. Unlock();
  418. return true;
  419. }
  420. bool WifiList::ResetConnected() {
  421. if (!Lock()) {
  422. throw std::runtime_error("Lock failed");
  423. }
  424. for (auto& e : credentials_) {
  425. e.second.connected = false;
  426. }
  427. Unlock();
  428. return true;
  429. }
  430. sys_net_wifi_entry* WifiList::Get(const std::string& ssid) {
  431. auto it = credentials_.find(ssid);
  432. if (it != credentials_.end()) {
  433. return &(it->second);
  434. }
  435. return nullptr;
  436. }
  437. const sys_net_wifi_entry* WifiList::GetConnected() {
  438. if (!Lock()) {
  439. ESP_LOGE(TAG_CRED_MANAGER, "Unable to lock structure %s", name_.c_str());
  440. return nullptr;
  441. }
  442. for (auto& e : credentials_) {
  443. if (e.second.connected) {
  444. return &e.second;
  445. }
  446. }
  447. Unlock();
  448. return nullptr;
  449. }
  450. bool WifiList::SetConnected(const wifi_event_sta_connected_t* evt, bool connected) {
  451. auto ssid = GetSSID(evt);
  452. auto bssid = GetBSSID(evt);
  453. auto existing = Get(ssid);
  454. if (existing) {
  455. if (bssid != existing->bssid || existing->connected != connected || existing->auth_type != GetAuthType(evt->authmode) ||
  456. existing->channel != evt->channel) {
  457. ResetConnected();
  458. if (!Lock()) {
  459. throw std::runtime_error("Lock failed");
  460. }
  461. strncpy(existing->bssid, bssid.c_str(), sizeof(existing->bssid));
  462. existing->connected = connected;
  463. existing->auth_type = GetAuthType(evt->authmode);
  464. existing->channel = evt->channel;
  465. config_raise_changed(false);
  466. Unlock();
  467. return true;
  468. }
  469. } else {
  470. ESP_LOGE(TAG_CRED_MANAGER, "Cannot set unknown ssid %s as connected", ssid.c_str());
  471. }
  472. return false;
  473. }
  474. sys_net_wifi_entry& WifiList::AddUpdate(const wifi_sta_config_t* sta, sys_net_auth_types auth_type) {
  475. return AddUpdate(sta, GetRadioTypes(nullptr), auth_type);
  476. }
  477. sys_net_wifi_entry& WifiList::AddUpdate(
  478. const wifi_sta_config_t* sta, const std::list<sys_net_radio_types>& radio_types, sys_net_auth_types auth_type) {
  479. auto ssid = GetSSID(sta);
  480. if (!Exists(sta)) {
  481. auto entry = ToSTAEntry(sta, radio_types, auth_type);
  482. ESP_LOGD(TAG_CRED_MANAGER, "Added new entry %s to list %s", ssid.c_str(), name_.c_str());
  483. if (!Lock()) {
  484. throw std::runtime_error("Lock failed");
  485. }
  486. credentials_[ssid] = entry;
  487. Unlock();
  488. } else {
  489. Update(sta);
  490. }
  491. ESP_LOGV(TAG_CRED_MANAGER, "AddUpdate: SSID: %s, PASS: %s", ssid.c_str(), credentials_[ssid].password);
  492. return credentials_[ssid];
  493. }
  494. bool WifiList::UpdateFromClock() {
  495. bool changed = false;
  496. if (Lock()) {
  497. for (auto iter = credentials_.begin(); iter != credentials_.end(); ++iter) {
  498. bool entrychanged = false;
  499. if (iter->second.has_last_seen) {
  500. entrychanged |= OffsetTimeStamp(&iter->second.last_seen);
  501. }
  502. if (iter->second.has_last_try) {
  503. entrychanged |= OffsetTimeStamp(&iter->second.last_try);
  504. }
  505. if (entrychanged) {
  506. ESP_LOGD(TAG_CRED_MANAGER, "Updated from clock");
  507. PrintWifiSTAEntry(iter->second);
  508. }
  509. changed |= entrychanged;
  510. }
  511. Unlock();
  512. }
  513. return changed;
  514. }
  515. void WifiList::PrintTimeStamp(const google_protobuf_Timestamp* timestamp) {
  516. if (timestamp == NULL) {
  517. printf("Timestamp is NULL\n");
  518. return;
  519. }
  520. char buffer[80];
  521. // Check for special case of time == 0
  522. if (timestamp->seconds == 0) {
  523. if (timestamp->nanos != 0) {
  524. printf("nanos not empty!");
  525. }
  526. snprintf(buffer, sizeof(buffer), "%-26s", "N/A");
  527. }
  528. // Check for timestamps less than 1704143717 (2024-01-01)
  529. else if (timestamp->seconds > 0 && timestamp->seconds < 1704143717) {
  530. // Convert seconds to time_t for use with localtime
  531. time_t rawtime = (time_t)timestamp->seconds;
  532. struct tm* timeinfo = localtime(&rawtime);
  533. strftime(buffer, sizeof(buffer), "%H:%M:%S", timeinfo);
  534. } else {
  535. // Convert seconds to time_t for use with localtime
  536. time_t rawtime = (time_t)timestamp->seconds;
  537. struct tm* timeinfo = localtime(&rawtime);
  538. strftime(buffer, sizeof(buffer), "%Y-%m-%dT%H:%M:%S", timeinfo);
  539. }
  540. printf("%-26s", buffer);
  541. }
  542. bool WifiList::UpdateLastTry(const std::string ssid) {
  543. if (!Lock()) {
  544. throw std::runtime_error("Lock failed");
  545. }
  546. auto entry = Get(ssid);
  547. ESP_RETURN_ON_FALSE(entry != nullptr, false, TAG_CRED_MANAGER, "Unknown ssid %s", ssid.c_str());
  548. ESP_RETURN_ON_FALSE(entry->ssid != nullptr, false, TAG_CRED_MANAGER, "Invalid pointer!");
  549. bool changed = UpdateLastTry(entry);
  550. Unlock();
  551. return changed;
  552. }
  553. bool WifiList::UpdateLastTry(sys_net_wifi_entry* entry) {
  554. ESP_RETURN_ON_FALSE(entry != nullptr, false, TAG_CRED_MANAGER, "Invalid pointer!");
  555. ESP_RETURN_ON_FALSE(entry->ssid != nullptr, false, TAG_CRED_MANAGER, "Invalid pointer!");
  556. ESP_LOGV(TAG_CRED_MANAGER, "Updating last try for %s", entry->ssid);
  557. return UpdateTimeStamp(&entry->last_try, entry->has_last_try);
  558. }
  559. bool WifiList::UpdateLastTry(const wifi_sta_config_t* sta) { return UpdateLastTry(GetSSID(sta)); }
  560. bool WifiList::UpdateLastTry(const wifi_ap_record_t* ap) { return UpdateLastTry(GetSSID(ap)); }
  561. bool WifiList::UpdateLastSeen(const wifi_sta_config_t* sta) { return UpdateLastSeen(GetSSID(sta)); }
  562. bool WifiList::UpdateLastSeen(const wifi_ap_record_t* ap) { return UpdateLastSeen(GetSSID(ap)); }
  563. bool WifiList::UpdateLastSeen(const std::string ssid) {
  564. if (!Lock()) {
  565. throw std::runtime_error("Lock failed");
  566. }
  567. auto entry = Get(ssid);
  568. bool changed = false;
  569. if (entry != nullptr) {
  570. changed = UpdateLastSeen(entry);
  571. }
  572. Unlock();
  573. return changed;
  574. }
  575. bool WifiList::UpdateLastSeen(sys_net_wifi_entry* entry) {
  576. ESP_RETURN_ON_FALSE(entry != nullptr, false, TAG_CRED_MANAGER, "Invalid pointer!");
  577. ESP_LOGV(TAG_CRED_MANAGER, "Updating last seen for %s", entry->ssid);
  578. return UpdateTimeStamp(&entry->last_seen, entry->has_last_seen);
  579. }
  580. sys_net_wifi_entry& WifiList::AddUpdate(const wifi_ap_record_t* scan_rec) {
  581. auto ssid = GetSSID(scan_rec);
  582. if (!Exists(scan_rec)) {
  583. ESP_LOGV(TAG_CRED_MANAGER, "Added new entry %s to list %s", ssid.c_str(), name_.c_str());
  584. if (!Lock()) {
  585. throw std::runtime_error("Lock failed");
  586. }
  587. credentials_[ssid] = ToSTAEntry(scan_rec);
  588. Unlock();
  589. } else {
  590. Update(scan_rec);
  591. }
  592. return credentials_[ssid];
  593. }
  594. sys_net_wifi_entry& WifiList::AddUpdate(const char* ssid, const char* password) {
  595. if (ssid == nullptr || password == nullptr) {
  596. throw std::invalid_argument("SSID and password cannot be null");
  597. }
  598. // Ensure that the SSID and password are not too long
  599. if (std::strlen(ssid) >= sizeof(sys_net_wifi_entry::ssid) || std::strlen(password) >= sizeof(sys_net_wifi_entry::password)) {
  600. throw std::length_error("SSID or password is too long");
  601. }
  602. if (!Exists(ssid)) {
  603. if (!Lock()) {
  604. throw std::runtime_error("Lock failed");
  605. }
  606. sys_net_wifi_entry newEntry = sys_net_wifi_entry_init_default;
  607. // Copy the SSID and password into the new entry, ensuring null termination
  608. std::strncpy(newEntry.ssid, ssid, sizeof(newEntry.ssid) - 1);
  609. newEntry.ssid[sizeof(newEntry.ssid) - 1] = '\0';
  610. std::strncpy(newEntry.password, password, sizeof(newEntry.password) - 1);
  611. newEntry.password[sizeof(newEntry.password) - 1] = '\0';
  612. ESP_LOGV(TAG_CRED_MANAGER, "Added new entry %s to list %s", ssid, name_.c_str());
  613. credentials_[ssid] = newEntry;
  614. Unlock();
  615. } else {
  616. auto existing = Get(ssid);
  617. if (strncmp(existing->password, password, sizeof(existing->password)) != 0) {
  618. strncpy(existing->password, password, sizeof(existing->password));
  619. existing->password[sizeof(existing->password) - 1] = '\0';
  620. }
  621. }
  622. return credentials_[ssid];
  623. }
  624. sys_net_wifi_entry& WifiList::AddUpdate(const sys_net_wifi_entry* sta, const char* password) {
  625. if (sta == nullptr) {
  626. throw std::invalid_argument("Entry pointer cannot be null");
  627. }
  628. auto converted = ToSTAEntry(sta);
  629. strncpy(converted.password, password, sizeof(converted.password));
  630. if (!Exists(sta->ssid)) {
  631. ESP_LOGD(TAG_CRED_MANAGER, "Added new entry %s to list %s", sta->ssid, name_.c_str());
  632. if (!Lock()) {
  633. throw std::runtime_error("Lock failed");
  634. }
  635. credentials_[sta->ssid] = converted;
  636. Unlock();
  637. } else {
  638. auto existing = Get(sta->ssid);
  639. Update(*existing, converted);
  640. // release the converted structure now
  641. Release(&converted);
  642. }
  643. return credentials_[sta->ssid];
  644. }
  645. void WifiList::PrintString(const char* pData, size_t length, const char* format) {
  646. std::string buffer;
  647. for (size_t i = 0; i < length && pData[i]; i++) {
  648. if (isprint((char)pData[i])) {
  649. buffer += (char)pData[i]; // Print as a character
  650. } else {
  651. buffer += '?';
  652. }
  653. }
  654. printf(format, buffer.c_str());
  655. }
  656. void WifiList::PrintWifiSTAEntryTitle() {
  657. printf("-----------------------------------------------------------------------------------------------------------------------------------------"
  658. "--------------------\n");
  659. printf("CONN SSID PASSWORD BSSID RSSI CHAN AUTH RADIO LAST TRY LAST "
  660. "SEEN\n");
  661. printf("-----------------------------------------------------------------------------------------------------------------------------------------"
  662. "--------------------\n");
  663. }
  664. void WifiList::PrintWifiSTAEntry(const sys_net_wifi_entry& entry) {
  665. google_protobuf_Timestamp gts = google_protobuf_Timestamp_init_default;
  666. printf("%-5c", entry.connected ? 'X' : ' ');
  667. printf("%-20s", entry.ssid);
  668. PrintString(entry.password, sizeof(entry.password), "%-25s");
  669. PrintString(entry.bssid, sizeof(entry.bssid), "%-20s");
  670. printf("%-4ddB", entry.rssi);
  671. printf("%3u ", static_cast<unsigned>(entry.channel));
  672. printf("%-14s", sys_net_auth_types_name(entry.auth_type));
  673. printf("%-9s", formatRadioTypes(entry.radio_type, entry.radio_type_count).c_str());
  674. if (entry.has_last_try) {
  675. PrintTimeStamp(&entry.last_try);
  676. } else {
  677. PrintTimeStamp(&gts);
  678. }
  679. if (entry.has_last_seen) {
  680. PrintTimeStamp(&entry.last_seen);
  681. } else {
  682. PrintTimeStamp(&gts);
  683. }
  684. printf("\n");
  685. }
  686. sys_net_wifi_entry* WifiList::GetIndex(size_t index) {
  687. if (index >= credentials_.size()) {
  688. return nullptr;
  689. }
  690. auto it = credentials_.begin();
  691. std::advance(it, index);
  692. return &(it->second);
  693. }
  694. sys_net_wifi_entry& WifiList::GetStrongestSTA() {
  695. if (credentials_.empty()) {
  696. throw std::runtime_error("No credentials available");
  697. }
  698. auto strongestIter = credentials_.begin();
  699. for (auto iter = credentials_.begin(); iter != credentials_.end(); ++iter) {
  700. if (iter->second.rssi > strongestIter->second.rssi) {
  701. strongestIter = iter;
  702. }
  703. }
  704. return strongestIter->second;
  705. }
  706. std::list<sys_net_radio_types> WifiList::GetRadioTypes(const wifi_ap_record_t* sta) {
  707. std::list<sys_net_radio_types> result;
  708. if (sta == nullptr) {
  709. result.push_back(sys_net_radio_types_UNKNOWN);
  710. } else {
  711. // Check each bit field and return the corresponding enum value
  712. if (sta->phy_11b) {
  713. result.push_back(sys_net_radio_types_PHY_11B);
  714. }
  715. if (sta->phy_11g) {
  716. result.push_back(sys_net_radio_types_PHY_11G);
  717. }
  718. if (sta->phy_11n) {
  719. result.push_back(sys_net_radio_types_PHY_11N);
  720. }
  721. if (sta->phy_lr) {
  722. result.push_back(sys_net_radio_types_LR);
  723. }
  724. if (sta->wps) {
  725. result.push_back(sys_net_radio_types_WPS);
  726. }
  727. if (sta->ftm_responder) {
  728. result.push_back(sys_net_radio_types_FTM_RESPONDER);
  729. }
  730. if (sta->ftm_initiator) {
  731. result.push_back(sys_net_radio_types_FTM_INITIATOR);
  732. }
  733. }
  734. return result;
  735. }
  736. wifi_auth_mode_t WifiList::GetESPAuthMode(sys_net_auth_types auth_type) {
  737. switch (auth_type) {
  738. case sys_net_auth_types_OPEN:
  739. return WIFI_AUTH_OPEN;
  740. case sys_net_auth_types_WEP:
  741. return WIFI_AUTH_WEP;
  742. case sys_net_auth_types_WPA_PSK:
  743. return WIFI_AUTH_WPA_PSK;
  744. case sys_net_auth_types_WPA2_PSK:
  745. return WIFI_AUTH_WPA2_PSK;
  746. case sys_net_auth_types_WPA_WPA2_PSK:
  747. return WIFI_AUTH_WPA_WPA2_PSK;
  748. case sys_net_auth_types_WPA2_ENTERPRISE:
  749. return WIFI_AUTH_WPA2_ENTERPRISE;
  750. case sys_net_auth_types_WPA3_PSK:
  751. return WIFI_AUTH_WPA3_PSK;
  752. case sys_net_auth_types_WPA2_WPA3_PSK:
  753. return WIFI_AUTH_WPA2_WPA3_PSK;
  754. case sys_net_auth_types_WAPI_PSK:
  755. return WIFI_AUTH_WAPI_PSK;
  756. default:
  757. return WIFI_AUTH_OPEN; // Default case
  758. }
  759. }
  760. sys_net_auth_types WifiList::GetAuthType(const wifi_ap_record_t* ap) {
  761. return ap ? GetAuthType(ap->authmode) : sys_net_auth_types_AUTH_UNKNOWN;
  762. }
  763. sys_net_auth_types WifiList::GetAuthType(const wifi_auth_mode_t mode) {
  764. switch (mode) {
  765. case WIFI_AUTH_OPEN:
  766. return sys_net_auth_types_OPEN;
  767. case WIFI_AUTH_WEP:
  768. return sys_net_auth_types_WEP;
  769. case WIFI_AUTH_WPA_PSK:
  770. return sys_net_auth_types_WPA_PSK;
  771. case WIFI_AUTH_WPA2_PSK:
  772. return sys_net_auth_types_WPA2_PSK;
  773. case WIFI_AUTH_WPA_WPA2_PSK:
  774. return sys_net_auth_types_WPA_WPA2_PSK;
  775. case WIFI_AUTH_WPA2_ENTERPRISE:
  776. return sys_net_auth_types_WPA2_ENTERPRISE;
  777. case WIFI_AUTH_WPA3_PSK:
  778. return sys_net_auth_types_WPA3_PSK;
  779. case WIFI_AUTH_WPA2_WPA3_PSK:
  780. return sys_net_auth_types_WPA2_WPA3_PSK;
  781. case WIFI_AUTH_WAPI_PSK:
  782. return sys_net_auth_types_WAPI_PSK;
  783. case WIFI_AUTH_MAX:
  784. return sys_net_auth_types_OPEN;
  785. }
  786. return sys_net_auth_types_AUTH_UNKNOWN;
  787. }