806 lines
27 KiB
C++
806 lines
27 KiB
C++
/***************************************************
|
|
Main of FingerprintDoorbell
|
|
****************************************************/
|
|
|
|
#include <WiFi.h>
|
|
#include <DNSServer.h>
|
|
#include <time.h>
|
|
#include <ESPAsyncWebServer.h>
|
|
#include <ElegantOTA.h>
|
|
#include <SPIFFS.h>
|
|
#include <PubSubClient.h>
|
|
#include "FingerprintManager.h"
|
|
#include "SettingsManager.h"
|
|
#include "global.h"
|
|
|
|
enum class Mode { scan, enroll, wificonfig, maintenance };
|
|
|
|
const char* VersionInfo = "0.4";
|
|
|
|
// ===================================================================================================================
|
|
// Caution: below are not the credentials for connecting to your home network, they are for the Access Point mode!!!
|
|
// ===================================================================================================================
|
|
const char* WifiConfigSsid = "FingerprintDoorbell-Config"; // SSID used for WiFi when in Access Point mode for configuration
|
|
const char* WifiConfigPassword = "12345678"; // password used for WiFi when in Access Point mode for configuration. Min. 8 chars needed!
|
|
IPAddress WifiConfigIp(192, 168, 4, 1); // IP of access point in wifi config mode
|
|
|
|
const long gmtOffset_sec = 0; // UTC Time
|
|
const int daylightOffset_sec = 0; // UTC Time
|
|
const int doorbellOutputPin = 5; // pin connected to the doorbell (when using hardware connection instead of mqtt to ring the bell)
|
|
const int doorOpenerOutputPin = 6; // pin connected to the door opener (when using hardware connection instead of mqtt to open the door)
|
|
#ifdef CUSTOM_GPIOS
|
|
const int customOutput1 = 18; // not used internally, but can be set over MQTT
|
|
const int customOutput2 = 26; // not used internally, but can be set over MQTT
|
|
const int customInput1 = 21; // not used internally, but changes are published over MQTT
|
|
const int customInput2 = 22; // not used internally, but changes are published over MQTT
|
|
bool customInput1Value = false;
|
|
bool customInput2Value = false;
|
|
#endif
|
|
|
|
const int logMessagesCount = 5;
|
|
String logMessages[logMessagesCount]; // log messages, 0=most recent log message
|
|
bool shouldReboot = false;
|
|
unsigned long wifiReconnectPreviousMillis = 0;
|
|
unsigned long mqttReconnectPreviousMillis = 0;
|
|
|
|
String enrollId;
|
|
String enrollName;
|
|
Mode currentMode = Mode::scan;
|
|
|
|
FingerprintManager fingerManager;
|
|
SettingsManager settingsManager;
|
|
bool needMaintenanceMode = false;
|
|
|
|
const byte DNS_PORT = 53;
|
|
DNSServer dnsServer;
|
|
AsyncWebServer webServer(80); // AsyncWebServer on port 80
|
|
AsyncEventSource events("/events"); // event source (Server-Sent events)
|
|
|
|
WiFiClient espClient;
|
|
PubSubClient mqttClient(espClient);
|
|
long lastMsg = 0;
|
|
char msg[50];
|
|
int value = 0;
|
|
bool mqttConfigValid = true;
|
|
|
|
|
|
Match lastMatch;
|
|
|
|
void addLogMessage(const String& message) {
|
|
// shift all messages in array by 1, oldest message will die
|
|
for (int i=logMessagesCount-1; i>0; i--)
|
|
logMessages[i]=logMessages[i-1];
|
|
logMessages[0]=message;
|
|
}
|
|
|
|
String getLogMessagesAsHtml() {
|
|
String html = "";
|
|
for (int i=logMessagesCount-1; i>=0; i--) {
|
|
if (logMessages[i]!="")
|
|
html = html + logMessages[i] + "<br>";
|
|
}
|
|
return html;
|
|
}
|
|
|
|
String getTimestampString(){
|
|
struct tm timeinfo;
|
|
if(!getLocalTime(&timeinfo)){
|
|
Serial.println("Failed to obtain time");
|
|
return "no time";
|
|
}
|
|
|
|
char buffer[25];
|
|
strftime(buffer,sizeof(buffer),"%Y-%m-%d %H:%M:%S %Z", &timeinfo);
|
|
String datetime = String(buffer);
|
|
return datetime;
|
|
}
|
|
|
|
/* wait for maintenance mode or timeout 5s */
|
|
bool waitForMaintenanceMode() {
|
|
needMaintenanceMode = true;
|
|
unsigned long startMillis = millis();
|
|
while (currentMode != Mode::maintenance) {
|
|
if ((millis() - startMillis) >= 5000ul) {
|
|
needMaintenanceMode = false;
|
|
return false;
|
|
}
|
|
delay(50);
|
|
}
|
|
needMaintenanceMode = false;
|
|
return true;
|
|
}
|
|
|
|
// Replaces placeholder in HTML pages
|
|
String processor(const String& var){
|
|
if(var == "LOGMESSAGES"){
|
|
return getLogMessagesAsHtml();
|
|
} else if (var == "FINGERLIST") {
|
|
return fingerManager.getFingerListAsHtmlOptionList();
|
|
} else if (var == "HOSTNAME") {
|
|
return settingsManager.getWifiSettings().hostname;
|
|
} else if (var == "VERSIONINFO") {
|
|
return VersionInfo;
|
|
} else if (var == "WIFI_SSID") {
|
|
return settingsManager.getWifiSettings().ssid;
|
|
} else if (var == "WIFI_PASSWORD") {
|
|
if (settingsManager.getWifiSettings().password.isEmpty())
|
|
return "";
|
|
else
|
|
return "********"; // for security reasons the wifi password will not left the device once configured
|
|
} else if (var == "MQTT_SERVER") {
|
|
return settingsManager.getAppSettings().mqttServer;
|
|
} else if (var == "MQTT_USERNAME") {
|
|
return settingsManager.getAppSettings().mqttUsername;
|
|
} else if (var == "MQTT_PASSWORD") {
|
|
return settingsManager.getAppSettings().mqttPassword;
|
|
} else if (var == "MQTT_ROOTTOPIC") {
|
|
return settingsManager.getAppSettings().mqttRootTopic;
|
|
} else if (var == "NTP_SERVER") {
|
|
return settingsManager.getAppSettings().ntpServer;
|
|
}
|
|
|
|
return String();
|
|
}
|
|
|
|
|
|
// send LastMessage to websocket clients
|
|
void notifyClients(String message) {
|
|
String messageWithTimestamp = "[" + getTimestampString() + "]: " + message;
|
|
Serial.println(messageWithTimestamp);
|
|
addLogMessage(messageWithTimestamp);
|
|
events.send(getLogMessagesAsHtml().c_str(),"message",millis(),1000);
|
|
|
|
String mqttRootTopic = settingsManager.getAppSettings().mqttRootTopic;
|
|
mqttClient.publish((String(mqttRootTopic) + "/lastLogMessage").c_str(), message.c_str());
|
|
}
|
|
|
|
void updateClientsFingerlist(String fingerlist) {
|
|
Serial.println("New fingerlist was sent to clients");
|
|
events.send(fingerlist.c_str(),"fingerlist",millis(),1000);
|
|
}
|
|
|
|
|
|
bool doPairing() {
|
|
String newPairingCode = settingsManager.generateNewPairingCode();
|
|
|
|
if (fingerManager.setPairingCode(newPairingCode)) {
|
|
AppSettings settings = settingsManager.getAppSettings();
|
|
settings.sensorPairingCode = newPairingCode;
|
|
settings.sensorPairingValid = true;
|
|
settingsManager.saveAppSettings(settings);
|
|
notifyClients("Pairing successful.");
|
|
return true;
|
|
} else {
|
|
notifyClients("Pairing failed.");
|
|
return false;
|
|
}
|
|
|
|
}
|
|
|
|
|
|
bool checkPairingValid() {
|
|
AppSettings settings = settingsManager.getAppSettings();
|
|
|
|
if (!settings.sensorPairingValid) {
|
|
if (settings.sensorPairingCode.isEmpty()) {
|
|
// first boot, do pairing automatically so the user does not have to do this manually
|
|
return doPairing();
|
|
} else {
|
|
Serial.println("Pairing has been invalidated previously.");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
String actualSensorPairingCode = fingerManager.getPairingCode();
|
|
//Serial.println("Awaited pairing code: " + settings.sensorPairingCode);
|
|
//Serial.println("Actual pairing code: " + actualSensorPairingCode);
|
|
|
|
if (actualSensorPairingCode.equals(settings.sensorPairingCode))
|
|
return true;
|
|
else {
|
|
if (!actualSensorPairingCode.isEmpty()) {
|
|
// An empty code means there was a communication problem. So we don't have a valid code, but maybe next read will succeed and we get one again.
|
|
// But here we just got an non-empty pairing code that was different to the awaited one. So don't expect that will change in future until repairing was done.
|
|
// -> invalidate pairing for security reasons
|
|
AppSettings settings = settingsManager.getAppSettings();
|
|
settings.sensorPairingValid = false;
|
|
settingsManager.saveAppSettings(settings);
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
|
|
bool initWifi() {
|
|
// Connect to Wi-Fi
|
|
WifiSettings wifiSettings = settingsManager.getWifiSettings();
|
|
WiFi.mode(WIFI_STA);
|
|
WiFi.config(INADDR_NONE, INADDR_NONE, INADDR_NONE, INADDR_NONE);
|
|
WiFi.setHostname(wifiSettings.hostname.c_str()); //define hostname
|
|
WiFi.begin(wifiSettings.ssid.c_str(), wifiSettings.password.c_str());
|
|
int counter = 0;
|
|
while (WiFi.status() != WL_CONNECTED) {
|
|
delay(1000);
|
|
Serial.println("Waiting for WiFi connection...");
|
|
counter++;
|
|
if (counter > 30)
|
|
return false;
|
|
}
|
|
Serial.println("Connected!");
|
|
|
|
// Print ESP32 Local IP Address
|
|
Serial.println(WiFi.localIP());
|
|
|
|
return true;
|
|
}
|
|
|
|
void initWiFiAccessPointForConfiguration() {
|
|
WiFi.softAPConfig(WifiConfigIp, WifiConfigIp, IPAddress(255, 255, 255, 0));
|
|
WiFi.softAP(WifiConfigSsid, WifiConfigPassword);
|
|
|
|
// if DNSServer is started with "*" for domain name, it will reply with
|
|
// provided IP to all DNS request
|
|
dnsServer.start(DNS_PORT, "*", WifiConfigIp);
|
|
|
|
Serial.print("AP IP address: ");
|
|
Serial.println(WifiConfigIp);
|
|
}
|
|
|
|
void onOTAStart() {
|
|
// Log when OTA has started
|
|
Serial.println("OTA update started!");
|
|
// <Add your own code here>
|
|
}
|
|
|
|
void onOTAProgress(size_t current, size_t final) {
|
|
static unsigned long ota_progress_millis = 0;
|
|
// Log every 1 second
|
|
if (millis() - ota_progress_millis > 1000) {
|
|
ota_progress_millis = millis();
|
|
Serial.printf("OTA Progress Current: %u bytes, Final: %u bytes\n", current, final);
|
|
}
|
|
}
|
|
|
|
void onOTAEnd(bool success) {
|
|
// Log when OTA has finished
|
|
if (success) {
|
|
Serial.println("OTA update finished successfully!");
|
|
} else {
|
|
Serial.println("There was an error during OTA update!");
|
|
}
|
|
// <Add your own code here>
|
|
}
|
|
|
|
|
|
|
|
void startWebserver(){
|
|
|
|
// Initialize SPIFFS
|
|
if(!SPIFFS.begin(true)){
|
|
Serial.println("An Error has occurred while mounting SPIFFS");
|
|
return;
|
|
}
|
|
|
|
|
|
|
|
// webserver for normal operating or wifi config?
|
|
if (currentMode == Mode::wificonfig)
|
|
{
|
|
// =================
|
|
// WiFi config mode
|
|
// =================
|
|
|
|
webServer.on("/", HTTP_GET, [](AsyncWebServerRequest *request){
|
|
request->send(SPIFFS, "/wificonfig.html", String(), false, processor);
|
|
});
|
|
|
|
webServer.on("/save", HTTP_GET, [](AsyncWebServerRequest *request){
|
|
if(request->hasArg("hostname"))
|
|
{
|
|
Serial.println("Save wifi config");
|
|
WifiSettings settings = settingsManager.getWifiSettings();
|
|
settings.hostname = request->arg("hostname");
|
|
settings.ssid = request->arg("ssid");
|
|
if (request->arg("password").equals("********")) // password is replaced by wildcards when given to the browser, so if the user didn't changed it, don't save it
|
|
settings.password = settingsManager.getWifiSettings().password; // use the old, already saved, one
|
|
else
|
|
settings.password = request->arg("password");
|
|
settingsManager.saveWifiSettings(settings);
|
|
shouldReboot = true;
|
|
}
|
|
request->redirect("/");
|
|
});
|
|
|
|
|
|
webServer.onNotFound([](AsyncWebServerRequest *request){
|
|
AsyncResponseStream *response = request->beginResponseStream("text/html");
|
|
response->printf("<!DOCTYPE html><html><head><title>FingerprintDoorbell</title><meta http-equiv=\"refresh\" content=\"0; url=http://%s\" /></head><body>", WiFi.softAPIP().toString().c_str());
|
|
response->printf("<p>Please configure your WiFi settings <a href='http://%s'>here</a> to connect FingerprintDoorbell to your home network.</p>", WiFi.softAPIP().toString().c_str());
|
|
response->print("</body></html>");
|
|
request->send(response);
|
|
});
|
|
|
|
}
|
|
else
|
|
{
|
|
// =======================
|
|
// normal operating mode
|
|
// =======================
|
|
events.onConnect([](AsyncEventSourceClient *client){
|
|
if(client->lastId()){
|
|
Serial.printf("Client reconnected! Last message ID it got was: %u\n", client->lastId());
|
|
}
|
|
//send event with message "ready", id current millis
|
|
// and set reconnect delay to 1 second
|
|
client->send(getLogMessagesAsHtml().c_str(),"message",millis(),1000);
|
|
});
|
|
webServer.addHandler(&events);
|
|
|
|
|
|
// Route for root / web page
|
|
webServer.on("/", HTTP_GET, [](AsyncWebServerRequest *request){
|
|
request->send(SPIFFS, "/index.html", String(), false, processor);
|
|
});
|
|
|
|
webServer.on("/enroll", HTTP_GET, [](AsyncWebServerRequest *request){
|
|
if(request->hasArg("startEnrollment"))
|
|
{
|
|
enrollId = request->arg("newFingerprintId");
|
|
enrollName = request->arg("newFingerprintName");
|
|
currentMode = Mode::enroll;
|
|
}
|
|
request->redirect("/");
|
|
});
|
|
|
|
webServer.on("/editFingerprints", HTTP_GET, [](AsyncWebServerRequest *request){
|
|
if(request->hasArg("selectedFingerprint"))
|
|
{
|
|
if(request->hasArg("btnDelete"))
|
|
{
|
|
int id = request->arg("selectedFingerprint").toInt();
|
|
waitForMaintenanceMode();
|
|
fingerManager.deleteFinger(id);
|
|
currentMode = Mode::scan;
|
|
}
|
|
else if (request->hasArg("btnRename"))
|
|
{
|
|
int id = request->arg("selectedFingerprint").toInt();
|
|
String newName = request->arg("renameNewName");
|
|
fingerManager.renameFinger(id, newName);
|
|
}
|
|
}
|
|
request->redirect("/");
|
|
});
|
|
|
|
webServer.on("/settings", HTTP_GET, [](AsyncWebServerRequest *request){
|
|
if(request->hasArg("btnSaveSettings"))
|
|
{
|
|
Serial.println("Save settings");
|
|
AppSettings settings = settingsManager.getAppSettings();
|
|
settings.mqttServer = request->arg("mqtt_server");
|
|
settings.mqttUsername = request->arg("mqtt_username");
|
|
settings.mqttPassword = request->arg("mqtt_password");
|
|
settings.mqttRootTopic = request->arg("mqtt_rootTopic");
|
|
settings.ntpServer = request->arg("ntpServer");
|
|
settingsManager.saveAppSettings(settings);
|
|
request->redirect("/");
|
|
shouldReboot = true;
|
|
} else {
|
|
request->send(SPIFFS, "/settings.html", String(), false, processor);
|
|
}
|
|
});
|
|
|
|
|
|
webServer.on("/pairing", HTTP_GET, [](AsyncWebServerRequest *request){
|
|
if(request->hasArg("btnDoPairing"))
|
|
{
|
|
Serial.println("Do (re)pairing");
|
|
doPairing();
|
|
request->redirect("/");
|
|
} else {
|
|
request->send(SPIFFS, "/settings.html", String(), false, processor);
|
|
}
|
|
});
|
|
|
|
|
|
|
|
webServer.on("/factoryReset", HTTP_GET, [](AsyncWebServerRequest *request){
|
|
if(request->hasArg("btnFactoryReset"))
|
|
{
|
|
notifyClients("Factory reset initiated...");
|
|
|
|
if (!fingerManager.deleteAll())
|
|
notifyClients("Finger database could not be deleted.");
|
|
|
|
if (!settingsManager.deleteAppSettings())
|
|
notifyClients("App settings could not be deleted.");
|
|
|
|
if (!settingsManager.deleteWifiSettings())
|
|
notifyClients("Wifi settings could not be deleted.");
|
|
|
|
request->redirect("/");
|
|
shouldReboot = true;
|
|
} else {
|
|
request->send(SPIFFS, "/settings.html", String(), false, processor);
|
|
}
|
|
});
|
|
|
|
|
|
webServer.on("/deleteAllFingerprints", HTTP_GET, [](AsyncWebServerRequest *request){
|
|
if(request->hasArg("btnDeleteAllFingerprints"))
|
|
{
|
|
notifyClients("Deleting all fingerprints...");
|
|
|
|
if (!fingerManager.deleteAll())
|
|
notifyClients("Finger database could not be deleted.");
|
|
|
|
request->redirect("/");
|
|
|
|
} else {
|
|
request->send(SPIFFS, "/settings.html", String(), false, processor);
|
|
}
|
|
});
|
|
|
|
|
|
webServer.onNotFound([](AsyncWebServerRequest *request){
|
|
request->send(404);
|
|
});
|
|
|
|
|
|
} // end normal operating mode
|
|
|
|
|
|
// common url callbacks
|
|
webServer.on("/reboot", HTTP_GET, [](AsyncWebServerRequest *request){
|
|
request->redirect("/");
|
|
shouldReboot = true;
|
|
});
|
|
|
|
webServer.on("/bootstrap.min.css", HTTP_GET, [](AsyncWebServerRequest *request){
|
|
request->send(SPIFFS, "/bootstrap.min.css", "text/css");
|
|
});
|
|
|
|
|
|
// Enable Over-the-air updates at http://<IPAddress>/update
|
|
ElegantOTA.begin(&webServer);
|
|
ElegantOTA.onStart(onOTAStart);
|
|
ElegantOTA.onProgress(onOTAProgress);
|
|
ElegantOTA.onEnd(onOTAEnd);
|
|
// Start server
|
|
webServer.begin();
|
|
// Init time by NTP Client
|
|
configTime(gmtOffset_sec, daylightOffset_sec, "pool.ntp.org");
|
|
notifyClients("System booted successfully!");
|
|
|
|
}
|
|
|
|
|
|
void mqttCallback(char* topic, byte* message, unsigned int length) {
|
|
Serial.print("Message arrived on topic: ");
|
|
Serial.print(topic);
|
|
Serial.print(". Message: ");
|
|
String messageTemp;
|
|
|
|
for (int i = 0; i < length; i++) {
|
|
Serial.print((char)message[i]);
|
|
messageTemp += (char)message[i];
|
|
}
|
|
Serial.println();
|
|
|
|
// Check incomming message for interesting topics
|
|
if (String(topic) == String(settingsManager.getAppSettings().mqttRootTopic) + "/ignoreTouchRing") {
|
|
if(messageTemp == "on"){
|
|
fingerManager.setIgnoreTouchRing(true);
|
|
}
|
|
else if(messageTemp == "off"){
|
|
fingerManager.setIgnoreTouchRing(false);
|
|
}
|
|
}
|
|
|
|
#ifdef CUSTOM_GPIOS
|
|
if (String(topic) == String(settingsManager.getAppSettings().mqttRootTopic) + "/customOutput1") {
|
|
if(messageTemp == "on"){
|
|
digitalWrite(customOutput1, HIGH);
|
|
}
|
|
else if(messageTemp == "off"){
|
|
digitalWrite(customOutput1, LOW);
|
|
}
|
|
}
|
|
if (String(topic) == String(settingsManager.getAppSettings().mqttRootTopic) + "/customOutput2") {
|
|
if(messageTemp == "on"){
|
|
digitalWrite(customOutput2, HIGH);
|
|
}
|
|
else if(messageTemp == "off"){
|
|
digitalWrite(customOutput2, LOW);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
}
|
|
|
|
void connectMqttClient() {
|
|
if (!mqttClient.connected() && mqttConfigValid) {
|
|
Serial.print("(Re)connect to MQTT broker...");
|
|
// Attempt to connect
|
|
bool connectResult;
|
|
|
|
// connect with or witout authentication
|
|
String lastWillTopic = settingsManager.getAppSettings().mqttRootTopic + "/lastLogMessage";
|
|
String lastWillMessage = "FingerprintDoorbell disconnected unexpectedly";
|
|
if (settingsManager.getAppSettings().mqttUsername.isEmpty() || settingsManager.getAppSettings().mqttPassword.isEmpty())
|
|
connectResult = mqttClient.connect(settingsManager.getWifiSettings().hostname.c_str(),lastWillTopic.c_str(), 1, false, lastWillMessage.c_str());
|
|
else
|
|
connectResult = mqttClient.connect(settingsManager.getWifiSettings().hostname.c_str(), settingsManager.getAppSettings().mqttUsername.c_str(), settingsManager.getAppSettings().mqttPassword.c_str(), lastWillTopic.c_str(), 1, false, lastWillMessage.c_str());
|
|
|
|
if (connectResult) {
|
|
// success
|
|
Serial.println("connected");
|
|
// Subscribe
|
|
mqttClient.subscribe((settingsManager.getAppSettings().mqttRootTopic + "/ignoreTouchRing").c_str(), 1); // QoS = 1 (at least once)
|
|
#ifdef CUSTOM_GPIOS
|
|
mqttClient.subscribe((settingsManager.getAppSettings().mqttRootTopic + "/customOutput1").c_str(), 1); // QoS = 1 (at least once)
|
|
mqttClient.subscribe((settingsManager.getAppSettings().mqttRootTopic + "/customOutput2").c_str(), 1); // QoS = 1 (at least once)
|
|
#endif
|
|
|
|
|
|
|
|
} else {
|
|
if (mqttClient.state() == 4 || mqttClient.state() == 5) {
|
|
mqttConfigValid = false;
|
|
notifyClients("Failed to connect to MQTT Server: bad credentials or not authorized. Will not try again, please check your settings.");
|
|
} else {
|
|
notifyClients(String("Failed to connect to MQTT Server, rc=") + mqttClient.state() + ", try again in 30 seconds");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void doScan()
|
|
{
|
|
Match match = fingerManager.scanFingerprint();
|
|
String mqttRootTopic = settingsManager.getAppSettings().mqttRootTopic;
|
|
switch(match.scanResult)
|
|
{
|
|
case ScanResult::noFinger:
|
|
// standard case, occurs every iteration when no finger touchs the sensor
|
|
if (match.scanResult != lastMatch.scanResult) {
|
|
Serial.println("no finger");
|
|
mqttClient.publish((String(mqttRootTopic) + "/ring").c_str(), "off");
|
|
mqttClient.publish((String(mqttRootTopic) + "/matchId").c_str(), "-1");
|
|
mqttClient.publish((String(mqttRootTopic) + "/matchName").c_str(), "");
|
|
mqttClient.publish((String(mqttRootTopic) + "/matchConfidence").c_str(), "-1");
|
|
}
|
|
break;
|
|
case ScanResult::matchFound:
|
|
notifyClients( String("Match Found: ") + match.matchId + " - " + match.matchName + " with confidence of " + match.matchConfidence );
|
|
if (match.scanResult != lastMatch.scanResult) {
|
|
if (checkPairingValid()) {
|
|
digitalWrite(doorOpenerOutputPin, HIGH);
|
|
mqttClient.publish((String(mqttRootTopic) + "/ring").c_str(), "off");
|
|
mqttClient.publish((String(mqttRootTopic) + "/matchId").c_str(), String(match.matchId).c_str());
|
|
mqttClient.publish((String(mqttRootTopic) + "/matchName").c_str(), match.matchName.c_str());
|
|
mqttClient.publish((String(mqttRootTopic) + "/matchConfidence").c_str(), String(match.matchConfidence).c_str());
|
|
Serial.println("MQTT message sent: Open the door!");
|
|
delay(1000);
|
|
digitalWrite(doorOpenerOutputPin, LOW);
|
|
} else {
|
|
notifyClients("Security issue! Match was not sent by MQTT because of invalid sensor pairing! This could potentially be an attack! If the sensor is new or has been replaced by you do a (re)pairing in settings page.");
|
|
}
|
|
}
|
|
delay(3000); // wait some time before next scan to let the LED blink
|
|
break;
|
|
case ScanResult::noMatchFound:
|
|
notifyClients(String("No Match Found (Code ") + match.returnCode + ")");
|
|
if (match.scanResult != lastMatch.scanResult) {
|
|
digitalWrite(doorbellOutputPin, HIGH);
|
|
mqttClient.publish((String(mqttRootTopic) + "/ring").c_str(), "on");
|
|
mqttClient.publish((String(mqttRootTopic) + "/matchId").c_str(), "-1");
|
|
mqttClient.publish((String(mqttRootTopic) + "/matchName").c_str(), "");
|
|
mqttClient.publish((String(mqttRootTopic) + "/matchConfidence").c_str(), "-1");
|
|
Serial.println("MQTT message sent: ring the bell!");
|
|
delay(1000);
|
|
digitalWrite(doorbellOutputPin, LOW);
|
|
} else {
|
|
delay(1000); // wait some time before next scan to let the LED blink
|
|
}
|
|
break;
|
|
case ScanResult::error:
|
|
notifyClients(String("ScanResult Error (Code ") + match.returnCode + ")");
|
|
break;
|
|
};
|
|
lastMatch = match;
|
|
|
|
}
|
|
|
|
void doEnroll()
|
|
{
|
|
int id = enrollId.toInt();
|
|
if (id < 1 || id > 200) {
|
|
notifyClients("Invalid memory slot id '" + enrollId + "'");
|
|
return;
|
|
}
|
|
|
|
NewFinger finger = fingerManager.enrollFinger(id, enrollName);
|
|
if (finger.enrollResult == EnrollResult::ok) {
|
|
notifyClients("Enrollment successfull. You can now use your new finger for scanning.");
|
|
updateClientsFingerlist(fingerManager.getFingerListAsHtmlOptionList());
|
|
} else if (finger.enrollResult == EnrollResult::error) {
|
|
notifyClients(String("Enrollment failed. (Code ") + finger.returnCode + ")");
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void reboot()
|
|
{
|
|
notifyClients("System is rebooting now...");
|
|
delay(1000);
|
|
|
|
mqttClient.disconnect();
|
|
espClient.stop();
|
|
dnsServer.stop();
|
|
webServer.end();
|
|
WiFi.disconnect();
|
|
ESP.restart();
|
|
}
|
|
|
|
|
|
void setup()
|
|
{
|
|
// open serial monitor for debug infos
|
|
Serial.begin(115200);
|
|
//while (!Serial); // For Yun/Leo/Micro/Zero/...
|
|
//delay(2000);
|
|
Serial.println("Hello");
|
|
// initialize GPIOs
|
|
pinMode(doorbellOutputPin, OUTPUT);
|
|
#ifdef CUSTOM_GPIOS
|
|
pinMode(customOutput1, OUTPUT);
|
|
pinMode(customOutput2, OUTPUT);
|
|
pinMode(customInput1, INPUT_PULLDOWN);
|
|
pinMode(customInput2, INPUT_PULLDOWN);
|
|
#endif
|
|
Serial.println("Hello2");
|
|
settingsManager.loadWifiSettings();
|
|
settingsManager.loadAppSettings();
|
|
Serial.println("Hello3");
|
|
fingerManager.connect();
|
|
Serial.println("Hello4");
|
|
if (!checkPairingValid())
|
|
notifyClients("Security issue! Pairing with sensor is invalid. This could potentially be an attack! If the sensor is new or has been replaced by you do a (re)pairing in settings page. MQTT messages regarding matching fingerprints will not been sent until pairing is valid again.");
|
|
Serial.println("Hello5");
|
|
if (fingerManager.isFingerOnSensor() || !settingsManager.isWifiConfigured())
|
|
{
|
|
// ring touched during startup or no wifi settings stored -> wifi config mode
|
|
currentMode = Mode::wificonfig;
|
|
Serial.println("Started WiFi-Config mode");
|
|
fingerManager.setLedRingWifiConfig();
|
|
initWiFiAccessPointForConfiguration();
|
|
startWebserver();
|
|
|
|
} else {
|
|
Serial.println("Started normal operating mode");
|
|
currentMode = Mode::scan;
|
|
if (initWifi()) {
|
|
startWebserver();
|
|
if (settingsManager.getAppSettings().mqttServer.isEmpty()) {
|
|
mqttConfigValid = false;
|
|
notifyClients("Error: No MQTT Broker is configured! Please go to settings and enter your server URL + user credentials.");
|
|
} else {
|
|
delay(5000);
|
|
IPAddress mqttServerIp;
|
|
if (WiFi.hostByName(settingsManager.getAppSettings().mqttServer.c_str(), mqttServerIp))
|
|
{
|
|
mqttConfigValid = true;
|
|
Serial.println("IP used for MQTT server: " + mqttServerIp.toString());
|
|
mqttClient.setServer(mqttServerIp , 1883);
|
|
mqttClient.setCallback(mqttCallback);
|
|
connectMqttClient();
|
|
}
|
|
else {
|
|
mqttConfigValid = false;
|
|
notifyClients("MQTT Server '" + settingsManager.getAppSettings().mqttServer + "' not found. Please check your settings.");
|
|
}
|
|
}
|
|
if (fingerManager.connected)
|
|
fingerManager.setLedRingReady();
|
|
else
|
|
fingerManager.setLedRingError();
|
|
} else {
|
|
fingerManager.setLedRingError();
|
|
shouldReboot = true;
|
|
}
|
|
|
|
}
|
|
Serial.println("Hello6");
|
|
}
|
|
|
|
void loop()
|
|
{
|
|
ElegantOTA.loop();
|
|
// shouldReboot flag for supporting reboot through webui
|
|
if (shouldReboot) {
|
|
reboot();
|
|
}
|
|
|
|
// Reconnect handling
|
|
if (currentMode != Mode::wificonfig)
|
|
{
|
|
unsigned long currentMillis = millis();
|
|
// reconnect WiFi if down for 30s
|
|
if ((WiFi.status() != WL_CONNECTED) && (currentMillis - wifiReconnectPreviousMillis >= 30000ul)) {
|
|
Serial.println("Reconnecting to WiFi...");
|
|
WiFi.disconnect();
|
|
WiFi.reconnect();
|
|
wifiReconnectPreviousMillis = currentMillis;
|
|
}
|
|
|
|
// reconnect mqtt if down
|
|
if (!settingsManager.getAppSettings().mqttServer.isEmpty()) {
|
|
if (!mqttClient.connected() && (currentMillis - mqttReconnectPreviousMillis >= 30000ul)) {
|
|
connectMqttClient();
|
|
mqttReconnectPreviousMillis = currentMillis;
|
|
}
|
|
mqttClient.loop();
|
|
}
|
|
}
|
|
|
|
|
|
// do the actual loop work
|
|
switch (currentMode)
|
|
{
|
|
case Mode::scan:
|
|
if (fingerManager.connected)
|
|
doScan();
|
|
break;
|
|
|
|
case Mode::enroll:
|
|
doEnroll();
|
|
currentMode = Mode::scan; // switch back to scan mode after enrollment is done
|
|
break;
|
|
|
|
case Mode::wificonfig:
|
|
dnsServer.processNextRequest(); // used for captive portal redirect
|
|
break;
|
|
|
|
case Mode::maintenance:
|
|
// do nothing, give webserver exclusive access to sensor (not thread-safe for concurrent calls)
|
|
break;
|
|
|
|
}
|
|
|
|
// enter maintenance mode (no continous scanning) if requested
|
|
if (needMaintenanceMode)
|
|
currentMode = Mode::maintenance;
|
|
|
|
#ifdef CUSTOM_GPIOS
|
|
// read custom inputs and publish by MQTT
|
|
bool i1;
|
|
bool i2;
|
|
i1 = (digitalRead(customInput1) == HIGH);
|
|
i2 = (digitalRead(customInput2) == HIGH);
|
|
|
|
String mqttRootTopic = settingsManager.getAppSettings().mqttRootTopic;
|
|
if (i1 != customInput1Value) {
|
|
if (i1)
|
|
mqttClient.publish((String(mqttRootTopic) + "/customInput1").c_str(), "on");
|
|
else
|
|
mqttClient.publish((String(mqttRootTopic) + "/customInput1").c_str(), "off");
|
|
}
|
|
|
|
if (i2 != customInput2Value) {
|
|
if (i2)
|
|
mqttClient.publish((String(mqttRootTopic) + "/customInput2").c_str(), "on");
|
|
else
|
|
mqttClient.publish((String(mqttRootTopic) + "/customInput2").c_str(), "off");
|
|
}
|
|
|
|
customInput1Value = i1;
|
|
customInput2Value = i2;
|
|
|
|
#endif
|
|
|
|
}
|
|
|