Files
IoTManager/src/utils/WiFiUtils.cpp

579 lines
16 KiB
C++
Raw Normal View History

#include "utils/WiFiUtils.h"
2023-09-22 20:57:50 +03:00
#include <vector>
2024-11-17 22:35:32 +03:00
#if defined(ESP32)
#include <esp_task_wdt.h>
#endif
2025-02-15 00:20:44 +03:00
#define TRIESONE 20 // количество секунд ожидания подключения к одной сети из несколких
#define TRIES 30 // количество секунд ожидания подключения сети если она одна
#ifndef ESP8266
std::vector<String> _ssidList;
std::vector<String> _passwordList;
// номер сети, для перебирания в момент подключения к сетям из массива
volatile uint8_t currentNetwork = 0;
volatile bool wifiConnecting = false;
volatile uint8_t connectionAttempts = 0;
//------------------------------------------
// Обработчики событий Wi-Fi
//------------------------------------------
void WiFiEvent(arduino_event_t *event)
2024-09-20 12:45:17 +03:00
{
2025-02-15 00:20:44 +03:00
switch (event->event_id)
{
#if defined(LIBRETINY) || defined(esp32c6_4mb) || defined(esp32c6_8mb)
case ARDUINO_EVENT_WIFI_STA_CONNECTED:
#else
case SYSTEM_EVENT_STA_CONNECTED:
#endif
// Подключились к STA
SerialPrint("i", "WIFI", "Connected to AP: " + WiFi.SSID());
// TODO если подключились, но не получили IP что будет?
break;
#if defined(LIBRETINY) || defined(esp32c6_4mb) || defined(esp32c6_8mb)
case ARDUINO_EVENT_WIFI_STA_GOT_IP:
#else
case SYSTEM_EVENT_STA_GOT_IP:
#endif
// Получили IP от роутера
// wifiReconnectTicker.detach();
ts.remove(WIFI_SCAN);
ts.remove(WIFI_CONN);
#ifdef LIBRETINY
SerialPrint("i", "WIFI", "http://" + ipToString(WiFi.localIP()));
jsonWriteStr(settingsFlashJson, "ip", ipToString(WiFi.localIP()));
#else
SerialPrint("i", "WIFI", "http://" + WiFi.localIP().toString());
jsonWriteStr(settingsFlashJson, "ip", WiFi.localIP().toString());
#endif
2025-03-13 20:01:57 +03:00
createItemFromNet("onWifi", "1", 1);
2025-02-15 00:20:44 +03:00
// запускаем MQTT
mqttInit();
SerialPrint("i", F("WIFI"), F("Network Init"));
2024-09-20 12:45:17 +03:00
2025-02-15 00:20:44 +03:00
// Отключаем AP при успешном подключении
WiFi.softAPdisconnect(true);
break;
#if defined(LIBRETINY) || defined(esp32c6_4mb) || defined(esp32c6_8mb)
case ARDUINO_EVENT_WIFI_STA_DISCONNECTED:
#else
case SYSTEM_EVENT_STA_DISCONNECTED:
#endif
// Отключились от STA
SerialPrint("i", "WIFI", "Disconnected from STA");
// Завершаем задачу проверки сети
ts.remove(WIFI_CONN);
if (wifiConnecting)
{ // если у нас ещё не закончились попытки подключения, то перезапускаем задачу
Serial.print(".");
checkConnection();
// wifiReconnectTicker.once_ms(WIFI_CHECK_INTERVAL, checkConnection);
}
else
{ // если попытки подключения исчерпаны, то переходим в AP
startAPMode();
}
break;
#if defined(LIBRETINY) || defined(esp32c6_4mb) || defined(esp32c6_8mb)
case ARDUINO_EVENT_WIFI_SCAN_DONE:
#else
case SYSTEM_EVENT_SCAN_DONE:
#endif
2025-03-17 00:32:18 +03:00
if (WiFi.scanComplete() >= 0) {
Serial.println("Valid Scan completed");
handleScanResults();
WiFi.scanDelete(); // Очищаем только при успешном сканировании
} else {
//Serial.println("Empty scan or error");
//WiFi.scanDelete(); // Принудительная очистка буфера
}
2025-02-15 00:20:44 +03:00
// Завершилось сканирование сетей
2025-03-17 00:32:18 +03:00
//Serial.println("Scan completed");
//handleScanResults();
2025-02-15 00:20:44 +03:00
break;
}
}
2025-02-15 00:20:44 +03:00
//------------------------------------------
// Обработка результатов сканирования
//------------------------------------------
void handleScanResults()
{
ssidListHeapJson = "{}";
_ssidList.clear();
_passwordList.clear();
jsonReadArray(settingsFlashJson, "routerssid", _ssidList);
jsonReadArray(settingsFlashJson, "routerpass", _passwordList);
int16_t numNetworks = WiFi.scanComplete();
if (numNetworks <= 0)
{
SerialPrint("i", "WIFI", "no networks found");
return;
}
// Ищем известные сети
int connectNumNet = -1;
// SerialPrint("i", "WIFI", "Count found: "+numNetworks);
for (int n = 0; n < numNetworks; ++n)
{
String ssid = WiFi.SSID(n);
jsonWriteStr_(ssidListHeapJson, String(n), ssid);
for (size_t i = 0; i < _ssidList.size(); i++)
{
if (ssid == _ssidList[i])
{
Serial.printf("Found known network: %s\n", _ssidList[i]);
if (connectNumNet < 0)
connectNumNet = i;
}
}
// if
}
2025-03-17 00:32:18 +03:00
sendStringToWs("ssidli", ssidListHeapJson, -1);
2025-02-15 00:20:44 +03:00
SerialPrint("i", "WIFI", "Scan Found: " + ssidListHeapJson);
2025-03-17 00:32:18 +03:00
if (connectNumNet >= 0 && !isNetworkActive())
2025-02-15 00:20:44 +03:00
{
// ts.remove(WIFI_SCAN);
connectToNextNetwork();
}
// checkConnection();
// connectToSTA(_ssidList[connectNumNet], _passwordList[connectNumNet]);
WiFi.scanDelete();
}
void WiFiUtilsItit()
{
#if !defined LIBRETINY
#if defined(esp32c6_4mb) || defined(esp32c6_8mb)
WiFi.setAutoReconnect(false);
#else
WiFi.setAutoConnect(false);
#endif
2025-03-17 00:32:18 +03:00
WiFi.persistent(true); // Сохраняет текущую сеть при сканировании
WiFi.setSleep(false);
2025-02-15 00:20:44 +03:00
#endif
WiFi.mode(WIFI_STA);
WiFi.onEvent(WiFiEvent);
2025-03-17 00:32:18 +03:00
_ssidList.clear();
_passwordList.clear();
2025-02-15 00:20:44 +03:00
jsonReadArray(settingsFlashJson, "routerssid", _ssidList);
jsonReadArray(settingsFlashJson, "routerpass", _passwordList);
if (_ssidList.empty() || _passwordList.empty())
{
SerialPrint("E", "WIFI", "No networks configured");
startAPMode();
return;
}
currentNetwork = 0;
connectionAttempts = 0;
connectToNextNetwork();
}
void connectToNextNetwork()
{
// все сети перебрали
if (currentNetwork >= _ssidList.size())
{
SerialPrint("i", "WIFI", "All networks tried");
ts.remove(WIFI_CONN);
startAPMode();
return;
}
wifiConnecting = true;
// connectionAttempts++;
const char *ssid = _ssidList[currentNetwork].c_str();
const char *pass = _passwordList[currentNetwork].c_str();
// Пробуем подключиться к сети
SerialPrint("i", "WIFI", "Connecting to: " + String(ssid));
WiFi.begin(ssid, pass);
#if defined(ESP32)
WiFi.setTxPower(WIFI_POWER_19_5dBm);
#elif defined(ESP8266)
WiFi.setOutputPower(20.5);
#endif
// проверяем статус подключения и перебираем сети если таймаут не вышел
checkConnection();
// wifiReconnectTicker.once_ms(WIFI_CHECK_INTERVAL, checkConnection);
}
void checkConnection()
{
ts.add(
WIFI_CONN, 1000,
[&](void *)
{
connectionAttempts++;
if (WiFi.status() == WL_CONNECTED)
{
connectionAttempts = 0;
wifiConnecting = false;
return;
}
if (connectionAttempts >= (_ssidList.size() > 1 ? TRIESONE : TRIES))
{
SerialPrint("i", "WIFI", "Max attempts reached");
currentNetwork++;
connectionAttempts = 0;
wifiConnecting = false;
}
if (wifiConnecting)
{
#ifdef ESP8266
if (WiFi.status() == WL_CONNECT_FAILED || WiFi.status() == WL_WRONG_PASSWORD)
#else
if (WiFi.status() == WL_CONNECT_FAILED)
#endif
{
SerialPrint("E", "WIFI", "Connection failed, wrong password?");
jsonWriteInt(errorsHeapJson, "passer", 1);
currentNetwork++;
connectionAttempts = 0;
}
// wifiReconnectTicker.once_ms(WIFI_CHECK_INTERVAL, checkConnection);
}
if (!wifiConnecting)
{
connectToNextNetwork();
}
},
nullptr, true);
}
//------------------------------------------
// Неблокирующее подключение к STA
//------------------------------------------
void connectToSTA(const char *ssid, const char *pass)
{
if (isNetworkActive())
return;
SerialPrint("i", "WIFI", "Connecting to ... " + String(ssid));
// SerialPrint("i", "WIFI", "pass connect: " + _passwordList[i]);
WiFi.begin(ssid, pass);
#if defined(ESP32)
WiFi.setTxPower(WIFI_POWER_19_5dBm);
#elif defined(ESP8266)
WiFi.setOutputPower(20.5);
#endif
}
void ScanAsync()
{
// bool res = false;
int n = WiFi.scanComplete();
SerialPrint("i", "WIFI", "scan result: " + String(n, DEC));
if (n == -1)
{ // Сканирование все еще выполняется
SerialPrint("i", "WIFI", "scanning in progress");
}
else
{
SerialPrint("i", "WIFI", "start scanning");
WiFi.scanNetworks(true, false);
}
}
#else
// ESP8266
2023-09-24 20:58:51 +03:00
void routerConnect()
{
2024-09-20 12:45:17 +03:00
#if !defined LIBRETINY
2024-11-22 22:45:10 +03:00
#if defined(esp32c6_4mb) || defined(esp32c6_8mb)
WiFi.setAutoReconnect(false);
#else
2023-09-24 20:58:51 +03:00
WiFi.setAutoConnect(false);
2024-11-22 22:45:10 +03:00
#endif
2023-09-24 20:58:51 +03:00
WiFi.persistent(false);
2024-09-20 12:45:17 +03:00
#endif
/* String s_staip = "192.168.2.62";
String s_gateway = "192.168.2.1";
String s_netmask = "255.255.255.0";
String s_dns = "192.168.2.1";
SerialPrint("i", "WIFI", "Use static IP");
WiFi.config(stringToIp(s_staip), stringToIp(s_gateway), stringToIp(s_netmask), stringToIp(s_dns));
// bool config(IPAddress local_ip, IPAddress gateway, IPAddress subnet, IPAddress dns1 = (uint32_t)0x00000000, IPAddress dns2 = (uint32_t)0x00000000);
SerialPrint("i", "WIFI", "Static IP: " + s_staip);
SerialPrint("i", "WIFI", "Gateway: " + s_gateway);
SerialPrint("i", "WIFI", "Netmask: " + s_netmask);
SerialPrint("i", "WIFI", "DNS: " + s_dns); */
2023-09-24 20:58:51 +03:00
WiFi.mode(WIFI_STA);
byte triesOne = TRIESONE;
std::vector<String> _ssidList;
std::vector<String> _passwordList;
2023-09-24 20:58:51 +03:00
jsonReadArray(settingsFlashJson, "routerssid", _ssidList);
jsonReadArray(settingsFlashJson, "routerpass", _passwordList);
if (_ssidList.size() > 1)
triesOne = TRIES;
if (_passwordList.size() == 0 && _ssidList[0] == "" && _passwordList[0] == "")
{
2024-09-20 12:45:17 +03:00
#ifndef LIBRETINY
2023-09-24 20:58:51 +03:00
WiFi.begin();
2024-09-20 12:45:17 +03:00
#endif
2023-09-24 20:58:51 +03:00
}
else
{
WiFi.begin(_ssidList[0].c_str(), _passwordList[0].c_str());
2024-09-20 12:45:17 +03:00
#if defined (ESP32)
2023-09-24 20:58:51 +03:00
WiFi.setTxPower(WIFI_POWER_19_5dBm);
2024-09-20 12:45:17 +03:00
#elif defined (ESP8266)
2023-09-24 20:58:51 +03:00
WiFi.setOutputPower(20.5);
#endif
2023-09-24 20:58:51 +03:00
String _ssid;
String _password;
for (int8_t i = 0; i < _ssidList.size(); i++)
{
_ssid = _ssid + _ssidList[i] + "; ";
}
2023-09-24 20:58:51 +03:00
for (int8_t i = 0; i < _passwordList.size(); i++)
{
_password = _password + _passwordList[i] + "; ";
}
2023-09-24 20:58:51 +03:00
SerialPrint("i", "WIFI", "ssid list: " + _ssid);
SerialPrint("i", "WIFI", "pass list: " + _password);
}
for (size_t i = 0; i < _ssidList.size(); i++)
{
triesOne = TRIESONE;
if (WiFi.status() == WL_CONNECTED)
break;
WiFi.begin(_ssidList[i].c_str(), _passwordList[i].c_str());
SerialPrint("i", "WIFI", "ssid connect: " + _ssidList[i]);
SerialPrint("i", "WIFI", "pass connect: " + _passwordList[i]);
2023-09-24 20:58:51 +03:00
while (--triesOne && WiFi.status() != WL_CONNECTED)
{
2023-09-22 20:57:50 +03:00
// SerialPrint("i", "WIFI", ": " + String((int)WiFi.status()));
2023-09-22 21:54:51 +03:00
#ifdef ESP8266
2023-09-24 20:58:51 +03:00
if (WiFi.status() == WL_CONNECT_FAILED || WiFi.status() == WL_WRONG_PASSWORD)
2023-09-22 21:54:51 +03:00
#else
2023-09-24 20:58:51 +03:00
if (WiFi.status() == WL_CONNECT_FAILED)
#endif
{
SerialPrint("E", "WIFI", "password is not correct");
triesOne = 1;
jsonWriteInt(errorsHeapJson, "passer", 1);
break;
}
2024-11-17 22:35:32 +03:00
#if defined(ESP32)
//SerialPrint("i", "Task", "Resetting WDT...");
#if !defined(esp32c6_4mb) && !defined(esp32c6_8mb) //TODO esp32-c6 переписать esp_task_wdt_init
2024-11-17 22:35:32 +03:00
esp_task_wdt_reset();
#endif
2024-11-17 22:35:32 +03:00
#endif
2023-09-24 20:58:51 +03:00
Serial.print(".");
delay(1000);
}
2023-09-24 20:58:51 +03:00
Serial.println("");
}
if (WiFi.status() != WL_CONNECTED)
{
Serial.println("");
startAPMode();
}
else
{
Serial.println("");
2024-09-20 12:45:17 +03:00
#ifdef LIBRETINY
SerialPrint("i", "WIFI", "http://" + ipToString(WiFi.localIP()));
jsonWriteStr(settingsFlashJson, "ip", ipToString(WiFi.localIP()));
#else
2023-09-24 20:58:51 +03:00
SerialPrint("i", "WIFI", "http://" + WiFi.localIP().toString());
jsonWriteStr(settingsFlashJson, "ip", WiFi.localIP().toString());
2024-09-20 12:45:17 +03:00
#endif
2025-03-13 20:01:57 +03:00
createItemFromNet("onWifi", "1", 1);
2023-09-24 20:58:51 +03:00
mqttInit();
}
SerialPrint("i", F("WIFI"), F("Network Init"));
}
2025-02-15 00:20:44 +03:00
#endif
2023-09-24 20:58:51 +03:00
bool startAPMode()
{
2025-02-15 00:20:44 +03:00
#ifndef ESP8266
wifiConnecting = false;
currentNetwork = 0;
connectionAttempts = 0;
#endif
2023-09-24 20:58:51 +03:00
SerialPrint("i", "WIFI", "AP Mode");
2023-09-24 20:58:51 +03:00
WiFi.disconnect();
WiFi.mode(WIFI_AP);
2023-09-24 20:58:51 +03:00
String _ssidAP = jsonReadStr(settingsFlashJson, "apssid");
String _passwordAP = jsonReadStr(settingsFlashJson, "appass");
2024-09-20 12:45:17 +03:00
if (_passwordAP == "")
2025-02-15 00:20:44 +03:00
WiFi.softAP(_ssidAP.c_str(), NULL, 6);
2024-09-20 12:45:17 +03:00
else
2025-02-15 00:20:44 +03:00
WiFi.softAP(_ssidAP.c_str(), _passwordAP.c_str(), 6);
2023-09-24 20:58:51 +03:00
IPAddress myIP = WiFi.softAPIP();
2024-09-20 12:45:17 +03:00
#ifdef LIBRETINY
SerialPrint("i", "WIFI", "AP IP: " + ipToString(myIP));
jsonWriteStr(settingsFlashJson, "ip", ipToString(myIP));
#else
2023-09-24 20:58:51 +03:00
SerialPrint("i", "WIFI", "AP IP: " + myIP.toString());
jsonWriteStr(settingsFlashJson, "ip", myIP.toString());
2024-09-20 12:45:17 +03:00
#endif
2023-09-24 20:58:51 +03:00
if (jsonReadInt(errorsHeapJson, "passer") != 1)
{
ts.add(
WIFI_SCAN, 30 * 1000,
[&](void *)
{
2025-02-15 00:20:44 +03:00
#ifdef ESP8266
std::vector<String> jArray;
2023-09-24 20:58:51 +03:00
jsonReadArray(settingsFlashJson, "routerssid", jArray);
for (int8_t i = 0; i < jArray.size(); i++)
{
SerialPrint("i", "WIFI", "scanning for " + jArray[i]);
2023-09-24 20:58:51 +03:00
}
if (RouterFind(jArray))
{
ts.remove(WIFI_SCAN);
WiFi.scanDelete();
routerConnect();
}
2025-02-15 00:20:44 +03:00
#else
ScanAsync();
#endif
2023-09-24 20:58:51 +03:00
},
nullptr, true);
}
return true;
}
2024-09-20 12:45:17 +03:00
2025-02-15 00:20:44 +03:00
#if defined(ESP8266)
boolean RouterFind(std::vector<String> jArray)
2023-09-24 20:58:51 +03:00
{
bool res = false;
int n = WiFi.scanComplete();
SerialPrint("i", "WIFI", "scan result: " + String(n, DEC));
if (n == -2)
{ // Сканирование не было запущено, запускаем
SerialPrint("i", "WIFI", "start scanning");
WiFi.scanNetworks(true, false); // async, show_hidden
}
else if (n == -1)
{ // Сканирование все еще выполняется
SerialPrint("i", "WIFI", "scanning in progress");
}
else if (n == 0)
{ // ни одна сеть не найдена
SerialPrint("i", "WIFI", "no networks found");
WiFi.scanNetworks(true, false);
}
else if (n > 0)
{
for (int8_t i = 0; i < n; i++)
2023-09-22 20:57:50 +03:00
{
2023-09-24 20:58:51 +03:00
for (int8_t k = 0; k < jArray.size(); k++)
{
2023-09-24 21:35:50 +03:00
if (WiFi.SSID(i) == jArray[k])
2023-09-22 20:57:50 +03:00
{
2023-09-24 20:58:51 +03:00
res = true;
}
2023-09-24 20:58:51 +03:00
}
// SerialPrint("i", "WIFI", (res ? "*" : "") + String(i, DEC) + ") " + WiFi.SSID(i));
jsonWriteStr_(ssidListHeapJson, String(i), WiFi.SSID(i));
2023-09-24 20:58:51 +03:00
// String(WiFi.RSSI(i)
}
2023-09-24 20:58:51 +03:00
}
SerialPrint("i", "WIFI", ssidListHeapJson);
WiFi.scanDelete();
return res;
}
2024-09-20 12:45:17 +03:00
#endif
2025-02-15 00:20:44 +03:00
boolean isNetworkActive()
{
return WiFi.status() == WL_CONNECTED;
}
2025-02-15 00:20:44 +03:00
uint8_t getNumAPClients()
{
return WiFi.softAPgetStationNum();
}
2025-02-15 00:20:44 +03:00
uint8_t RSSIquality()
{
uint8_t res = 0;
if (isNetworkActive())
{
int rssi = WiFi.RSSI();
if (rssi >= -50)
{
res = 6; //"Excellent";
}
2025-02-15 00:20:44 +03:00
else if (rssi < -50 && rssi >= -60)
{
res = 5; //"Very good";
}
else if (rssi < -60 && rssi >= -70)
{
res = 4; //"Good";
}
else if (rssi < -70 && rssi >= -80)
{
res = 3; //"Low";
}
else if (rssi < -80 && rssi > -100)
{
res = 2; //"Very low";
}
else if (rssi <= -100)
{
res = 1; //"No signal";
}
}
return res;
}
2024-09-20 12:45:17 +03:00
#ifdef LIBRETINY
String httpGetString(HTTPClient &http)
{
2025-02-15 00:20:44 +03:00
String payload = "";
int len = http.getSize();
uint8_t buff[128] = {0};
WiFiClient *stream = http.getStreamPtr();
// read all data from server
while (http.connected() && (len > 0 || len == -1))
{
// get available data size
size_t size = stream->available();
if (size)
{
// read up to 128 byte
int c = stream->readBytes(buff, ((size > sizeof(buff)) ? sizeof(buff) : size));
// write it to Serial
// Serial.write(buff,c);
// payload += String((char*)buff);
char charBuff[c + 1]; // Create a character array with space for null terminator
memcpy(charBuff, buff, c); // Copy the data to the character array
charBuff[c] = '\0'; // Null-terminate the character array
payload += String(charBuff); // Append the character array to the payload
if (len > 0)
{
len -= c;
}
}
delay(1);
}
return payload;
2024-09-20 12:45:17 +03:00
}
#endif