mirror of
https://github.com/1technophile/OpenMQTTGateway.git
synced 2026-03-04 06:24:25 +01:00
Gaterway states to improve LED management Make the sleep function accessible outside BT BREAKING remove sleep for ESP8266 BREAKING lowpowermode API changed to powermode and new mode definition Co-authored-by: Florian <1technophile@users.noreply.github.com>
308 lines
12 KiB
C++
308 lines
12 KiB
C++
/*
|
|
OpenMQTTGateway - ESP8266 or Arduino program for home automation
|
|
|
|
Act as a gateway between your 433mhz, infrared IR, BLE, LoRa signal and one interface like an MQTT broker
|
|
Send and receiving command by MQTT
|
|
|
|
This program enables to:
|
|
- receive MQTT data from a topic and send signal (RF, IR, BLE, GSM) corresponding to the received MQTT data
|
|
- publish MQTT data to a different topic related to received signals (RF, IR, BLE, GSM)
|
|
|
|
Copyright: (c)Florian ROBERT
|
|
|
|
This file is part of OpenMQTTGateway.
|
|
|
|
OpenMQTTGateway is free software: you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
(at your option) any later version.
|
|
|
|
OpenMQTTGateway is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
#if defined(ESP32) && defined(USE_BLUFI)
|
|
# include "NimBLEDevice.h"
|
|
# include "esp_blufi_api.h"
|
|
|
|
extern "C" {
|
|
# include "esp_blufi.h"
|
|
}
|
|
|
|
/* store the station info for send back to phone */
|
|
//static bool gl_sta_connected = false;
|
|
bool omg_blufi_ble_connected = false;
|
|
static uint8_t gl_sta_bssid[6];
|
|
static uint8_t gl_sta_ssid[32];
|
|
static uint8_t gl_sta_passwd[64];
|
|
static int gl_sta_ssid_len;
|
|
static bool gl_sta_is_connecting = false;
|
|
static esp_blufi_extra_info_t gl_sta_conn_info;
|
|
|
|
static void example_event_callback(esp_blufi_cb_event_t event, esp_blufi_cb_param_t* param);
|
|
void wifi_event_handler(arduino_event_id_t event);
|
|
esp_err_t blufi_security_init(void);
|
|
void blufi_dh_negotiate_data_handler(uint8_t* data, int len, uint8_t** output_data, int* output_len, bool* need_free);
|
|
int blufi_aes_encrypt(uint8_t iv8, uint8_t* crypt_data, int crypt_len);
|
|
int blufi_aes_decrypt(uint8_t iv8, uint8_t* crypt_data, int crypt_len);
|
|
uint16_t blufi_crc_checksum(uint8_t iv8, uint8_t* data, int len);
|
|
void blufi_security_deinit(void);
|
|
|
|
static void example_event_callback(esp_blufi_cb_event_t event, esp_blufi_cb_param_t* param) {
|
|
/* actually, should post to blufi_task handle the procedure,
|
|
* now, as a example, we do it more simply */
|
|
switch (event) {
|
|
case ESP_BLUFI_EVENT_INIT_FINISH:
|
|
Log.trace(F("BLUFI init finish" CR));
|
|
esp_blufi_adv_start();
|
|
break;
|
|
case ESP_BLUFI_EVENT_DEINIT_FINISH:
|
|
Log.trace(F("BLUFI deinit finish" CR));
|
|
NimBLEDevice::deinit(true);
|
|
break;
|
|
case ESP_BLUFI_EVENT_BLE_CONNECT:
|
|
Log.trace(F("BLUFI BLE connect" CR));
|
|
gatewayState = GatewayState::ONBOARDING;
|
|
omg_blufi_ble_connected = true;
|
|
esp_blufi_adv_stop();
|
|
blufi_security_init();
|
|
break;
|
|
case ESP_BLUFI_EVENT_BLE_DISCONNECT:
|
|
Log.trace(F("BLUFI BLE disconnect" CR));
|
|
omg_blufi_ble_connected = false;
|
|
blufi_security_deinit();
|
|
if (WiFi.isConnected()) {
|
|
gatewayState = GatewayState::NTWK_CONNECTED;
|
|
esp_blufi_deinit();
|
|
# ifndef ESPWifiManualSetup
|
|
wifiManager.stopConfigPortal();
|
|
# endif
|
|
} else {
|
|
gatewayState = GatewayState::WAITING_ONBOARDING;
|
|
esp_blufi_adv_start();
|
|
}
|
|
break;
|
|
case ESP_BLUFI_EVENT_REQ_CONNECT_TO_AP:
|
|
Log.trace(F("BLUFI request wifi connect to AP" CR));
|
|
WiFi.begin((char*)gl_sta_ssid, (char*)gl_sta_passwd);
|
|
gl_sta_is_connecting = true;
|
|
blufiConnectAP = true;
|
|
break;
|
|
case ESP_BLUFI_EVENT_REQ_DISCONNECT_FROM_AP:
|
|
Log.trace(F("BLUFI request wifi disconnect from AP\n" CR));
|
|
WiFi.disconnect();
|
|
blufiConnectAP = false;
|
|
break;
|
|
case ESP_BLUFI_EVENT_REPORT_ERROR:
|
|
Log.trace(F("BLUFI report error, error code %d\n" CR), param->report_error.state);
|
|
esp_blufi_send_error_info(param->report_error.state);
|
|
break;
|
|
case ESP_BLUFI_EVENT_GET_WIFI_STATUS: {
|
|
esp_blufi_extra_info_t info;
|
|
if (WiFi.isConnected()) {
|
|
memset(&info, 0, sizeof(esp_blufi_extra_info_t));
|
|
memcpy(info.sta_bssid, gl_sta_bssid, 6);
|
|
info.sta_bssid_set = true;
|
|
info.sta_ssid = gl_sta_ssid;
|
|
info.sta_ssid_len = gl_sta_ssid_len;
|
|
esp_blufi_send_wifi_conn_report(WIFI_MODE_STA, ESP_BLUFI_STA_CONN_SUCCESS, 0, &info);
|
|
} else if (gl_sta_is_connecting) {
|
|
esp_blufi_send_wifi_conn_report(WIFI_MODE_STA, ESP_BLUFI_STA_CONNECTING, 0, &gl_sta_conn_info);
|
|
} else {
|
|
esp_blufi_send_wifi_conn_report(WIFI_MODE_STA, ESP_BLUFI_STA_CONN_FAIL, 0, &gl_sta_conn_info);
|
|
}
|
|
|
|
break;
|
|
}
|
|
case ESP_BLUFI_EVENT_RECV_SLAVE_DISCONNECT_BLE:
|
|
Log.trace(F("BLUFI recv slave disconnect a ble connection" CR));
|
|
esp_blufi_disconnect();
|
|
break;
|
|
case ESP_BLUFI_EVENT_RECV_STA_SSID:
|
|
strncpy((char*)gl_sta_ssid, (char*)param->sta_ssid.ssid, param->sta_ssid.ssid_len);
|
|
gl_sta_ssid[param->sta_ssid.ssid_len] = '\0';
|
|
Log.notice(F("Recv STA SSID %s" CR), gl_sta_ssid);
|
|
break;
|
|
case ESP_BLUFI_EVENT_RECV_STA_PASSWD:
|
|
strncpy((char*)gl_sta_passwd, (char*)param->sta_passwd.passwd, param->sta_passwd.passwd_len);
|
|
gl_sta_passwd[param->sta_passwd.passwd_len] = '\0';
|
|
Log.notice(F("Recv STA PASSWORD" CR));
|
|
break;
|
|
case ESP_BLUFI_EVENT_GET_WIFI_LIST: {
|
|
WiFi.scanNetworks(true);
|
|
break;
|
|
}
|
|
case ESP_BLUFI_EVENT_RECV_CUSTOM_DATA: {
|
|
Log.notice(F("Recv Custom Data %" PRIu32 CR), param->custom_data.data_len);
|
|
esp_log_buffer_hex("Custom Data", param->custom_data.data, param->custom_data.data_len);
|
|
|
|
DynamicJsonDocument json(JSON_MSG_BUFFER_MAX);
|
|
auto error = deserializeJson(json, param->custom_data.data);
|
|
if (error) {
|
|
Log.error(F("deserialize config failed: %s, buffer capacity: %u" CR), error.c_str(), json.capacity());
|
|
break;
|
|
}
|
|
if (!json.isNull()) {
|
|
Log.trace(F("\nparsed json, size: %u" CR), json.memoryUsage());
|
|
# if !MQTT_BROKER_MODE
|
|
if (json.containsKey("mqtt_server") && json["mqtt_server"].is<const char*>() && json["mqtt_server"].as<String>().length() > 0 && json["mqtt_server"].as<String>().length() < parameters_size)
|
|
strcpy(cnt_parameters_array[CNT_DEFAULT_INDEX].mqtt_server, json["mqtt_server"]);
|
|
if (json.containsKey("mqtt_port") && json["mqtt_port"].is<const char*>() && json["mqtt_port"].as<String>().length() > 0 && json["mqtt_port"].as<String>().length() < parameters_size)
|
|
strcpy(cnt_parameters_array[CNT_DEFAULT_INDEX].mqtt_port, json["mqtt_port"]);
|
|
if (json.containsKey("mqtt_user") && json["mqtt_user"].is<const char*>() && json["mqtt_user"].as<String>().length() > 0 && json["mqtt_user"].as<String>().length() < parameters_size)
|
|
strcpy(cnt_parameters_array[CNT_DEFAULT_INDEX].mqtt_user, json["mqtt_user"]);
|
|
if (json.containsKey("mqtt_pass") && json["mqtt_pass"].is<const char*>() && json["mqtt_pass"].as<String>().length() > 0 && json["mqtt_pass"].as<String>().length() < parameters_size)
|
|
strcpy(cnt_parameters_array[CNT_DEFAULT_INDEX].mqtt_pass, json["mqtt_pass"]);
|
|
|
|
if (json.containsKey("mqtt_broker_secure") && json["mqtt_broker_secure"].is<bool>())
|
|
cnt_parameters_array[CNT_DEFAULT_INDEX].isConnectionSecure = json["mqtt_broker_secure"].as<bool>();
|
|
if (json.containsKey("mqtt_iscertvalid") && json["mqtt_iscertvalid"].is<bool>())
|
|
cnt_parameters_array[CNT_DEFAULT_INDEX].isCertValidate = json["mqtt_iscertvalid"].as<bool>();
|
|
if (json.containsKey("mqtt_broker_cert") && json["mqtt_broker_cert"].is<const char*>() && json["mqtt_broker_cert"].as<String>().length() > MIN_CERT_LENGTH)
|
|
cnt_parameters_array[CNT_DEFAULT_INDEX].server_cert = json["mqtt_broker_cert"].as<const char*>();
|
|
if (json.containsKey("mqtt_client_cert") && json["mqtt_client_cert"].is<const char*>() && json["mqtt_client_cert"].as<String>().length() > MIN_CERT_LENGTH)
|
|
cnt_parameters_array[CNT_DEFAULT_INDEX].client_cert = json["mqtt_client_cert"].as<const char*>();
|
|
if (json.containsKey("mqtt_client_key") && json["mqtt_client_key"].is<const char*>() && json["mqtt_client_key"].as<String>().length() > MIN_CERT_LENGTH)
|
|
cnt_parameters_array[CNT_DEFAULT_INDEX].client_key = json["mqtt_client_key"].as<const char*>();
|
|
if (json.containsKey("ota_server_cert") && json["ota_server_cert"].is<const char*>() && json["ota_server_cert"].as<String>().length() > MIN_CERT_LENGTH)
|
|
cnt_parameters_array[CNT_DEFAULT_INDEX].ota_server_cert = json["ota_server_cert"].as<const char*>();
|
|
|
|
if (json.containsKey("cnt_index") && json["cnt_index"].is<int>() && json["cnt_index"].as<int>() > 0 && json["cnt_index"].as<int>() < 3) {
|
|
cnt_index = json["cnt_index"].as<int>();
|
|
} else {
|
|
cnt_index = CNT_DEFAULT_INDEX;
|
|
}
|
|
|
|
// We suppose the connection is valid (could be tested before)
|
|
cnt_parameters_array[cnt_index].validConnection = true;
|
|
# endif
|
|
|
|
if (json.containsKey("gateway_name"))
|
|
strcpy(gateway_name, json["gateway_name"]);
|
|
if (json.containsKey("mqtt_topic"))
|
|
strcpy(mqtt_topic, json["mqtt_topic"]);
|
|
if (json.containsKey("ota_pass"))
|
|
strcpy(ota_pass, json["ota_pass"]);
|
|
|
|
saveConfig();
|
|
}
|
|
break;
|
|
}
|
|
case ESP_BLUFI_EVENT_RECV_USERNAME:
|
|
break;
|
|
case ESP_BLUFI_EVENT_RECV_CA_CERT:
|
|
break;
|
|
case ESP_BLUFI_EVENT_RECV_CLIENT_CERT:
|
|
break;
|
|
case ESP_BLUFI_EVENT_RECV_SERVER_CERT:
|
|
break;
|
|
case ESP_BLUFI_EVENT_RECV_CLIENT_PRIV_KEY:
|
|
break;
|
|
;
|
|
case ESP_BLUFI_EVENT_RECV_SERVER_PRIV_KEY:
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void wifi_event_handler(arduino_event_id_t event) {
|
|
switch (event) {
|
|
case ARDUINO_EVENT_WIFI_STA_GOT_IP6:
|
|
case ARDUINO_EVENT_WIFI_STA_GOT_IP: {
|
|
gl_sta_is_connecting = false;
|
|
esp_blufi_extra_info_t info;
|
|
memset(&info, 0, sizeof(esp_blufi_extra_info_t));
|
|
memcpy(info.sta_bssid, gl_sta_bssid, 6);
|
|
info.sta_bssid_set = true;
|
|
info.sta_ssid = gl_sta_ssid;
|
|
info.sta_ssid_len = gl_sta_ssid_len;
|
|
if (omg_blufi_ble_connected == true) {
|
|
esp_blufi_send_wifi_conn_report(WIFI_MODE_STA, ESP_BLUFI_STA_CONN_SUCCESS, 0, &info);
|
|
}
|
|
|
|
break;
|
|
}
|
|
case ARDUINO_EVENT_WIFI_SCAN_DONE: {
|
|
uint16_t apCount = WiFi.scanComplete();
|
|
if (apCount == 0) {
|
|
Log.notice(F("No AP found" CR));
|
|
break;
|
|
}
|
|
wifi_ap_record_t* ap_list = (wifi_ap_record_t*)malloc(sizeof(wifi_ap_record_t) * apCount);
|
|
if (!ap_list) {
|
|
Log.error(F("malloc error, ap_list is NULL"));
|
|
break;
|
|
}
|
|
ESP_ERROR_CHECK(esp_wifi_scan_get_ap_records(&apCount, ap_list));
|
|
esp_blufi_ap_record_t* blufi_ap_list = (esp_blufi_ap_record_t*)malloc(apCount * sizeof(esp_blufi_ap_record_t));
|
|
if (!blufi_ap_list) {
|
|
if (ap_list) {
|
|
free(ap_list);
|
|
}
|
|
Log.error(F("malloc error, blufi_ap_list is NULL" CR));
|
|
break;
|
|
}
|
|
for (int i = 0; i < apCount; ++i) {
|
|
blufi_ap_list[i].rssi = ap_list[i].rssi;
|
|
memcpy(blufi_ap_list[i].ssid, ap_list[i].ssid, sizeof(ap_list[i].ssid));
|
|
}
|
|
|
|
if (omg_blufi_ble_connected == true) {
|
|
esp_blufi_send_wifi_list(apCount, blufi_ap_list);
|
|
}
|
|
|
|
free(ap_list);
|
|
free(blufi_ap_list);
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
return;
|
|
}
|
|
|
|
static esp_blufi_callbacks_t example_callbacks = {
|
|
.event_cb = example_event_callback,
|
|
.negotiate_data_handler = blufi_dh_negotiate_data_handler,
|
|
.encrypt_func = blufi_aes_encrypt,
|
|
.decrypt_func = blufi_aes_decrypt,
|
|
.checksum_func = blufi_crc_checksum,
|
|
};
|
|
|
|
bool startBlufi() {
|
|
esp_err_t ret = ESP_OK;
|
|
WiFi.onEvent(wifi_event_handler);
|
|
|
|
ret = esp_blufi_register_callbacks(&example_callbacks);
|
|
if (ret) {
|
|
Log.error(F("%s blufi register failed, error code = %x" CR), __func__, ret);
|
|
return false;
|
|
}
|
|
|
|
if (NimBLEDevice::getInitialized()) {
|
|
NimBLEDevice::deinit(true);
|
|
delay(50);
|
|
}
|
|
esp_blufi_btc_init();
|
|
uint8_t mac[6];
|
|
esp_read_mac(mac, ESP_MAC_WIFI_STA);
|
|
char advName[17] = {0};
|
|
// Check length of Gateway_Short_Name
|
|
if (strlen(Gateway_Short_Name) > 3) {
|
|
Log.error(F("Gateway_Short_Name is too long, max 3 characters" CR));
|
|
return false;
|
|
}
|
|
snprintf(advName, sizeof(advName), Gateway_Short_Name "_%02X%02X%02X%02X%02X%02X", mac[0], mac[1], mac[2], mac[3], mac[4], mac[5]);
|
|
NimBLEDevice::init(advName);
|
|
esp_blufi_gatt_svr_init();
|
|
ble_gatts_start();
|
|
return esp_blufi_profile_init() == ESP_OK;
|
|
}
|
|
|
|
#endif // defined(ESP32) && defined(USE_BLUFI)
|