519 lines
16 KiB
C++
519 lines
16 KiB
C++
#include "WebCfgServer.h"
|
|
#include "PreferencesKeys.h"
|
|
#include "Version.h"
|
|
#include "hardware/WifiEthServer.h"
|
|
|
|
WebCfgServer::WebCfgServer(NukiWrapper* nuki, Network* network, EthServer* ethServer, Preferences* preferences, bool allowRestartToPortal)
|
|
: _server(ethServer),
|
|
_nuki(nuki),
|
|
_network(network),
|
|
_preferences(preferences),
|
|
_allowRestartToPortal(allowRestartToPortal)
|
|
{
|
|
_confirmCode = generateConfirmCode();
|
|
|
|
String str = _preferences->getString(preference_cred_user);
|
|
|
|
if(str.length() > 0)
|
|
{
|
|
_hasCredentials = true;
|
|
const char *user = str.c_str();
|
|
memcpy(&_credUser, user, str.length());
|
|
|
|
str = _preferences->getString(preference_cred_password);
|
|
const char *pass = str.c_str();
|
|
memcpy(&_credPassword, pass, str.length());
|
|
}
|
|
}
|
|
|
|
|
|
void WebCfgServer::initialize()
|
|
{
|
|
_server.on("/", [&]() {
|
|
if (_hasCredentials && !_server.authenticate(_credUser, _credPassword)) {
|
|
return _server.requestAuthentication();
|
|
}
|
|
String response = "";
|
|
buildHtml(response);
|
|
_server.send(200, "text/html", response);
|
|
});
|
|
_server.on("/cred", [&]() {
|
|
if (_hasCredentials && !_server.authenticate(_credUser, _credPassword)) {
|
|
return _server.requestAuthentication();
|
|
}
|
|
String response = "";
|
|
buildCredHtml(response);
|
|
_server.send(200, "text/html", response);
|
|
});
|
|
_server.on("/wifi", [&]() {
|
|
if (_hasCredentials && !_server.authenticate(_credUser, _credPassword)) {
|
|
return _server.requestAuthentication();
|
|
}
|
|
String response = "";
|
|
buildConfigureWifiHtml(response);
|
|
_server.send(200, "text/html", response);
|
|
});
|
|
_server.on("/unpair", [&]() {
|
|
if (_hasCredentials && !_server.authenticate(_credUser, _credPassword)) {
|
|
return _server.requestAuthentication();
|
|
}
|
|
|
|
processUnpair();
|
|
});
|
|
_server.on("/wifimanager", [&]() {
|
|
if (_hasCredentials && !_server.authenticate(_credUser, _credPassword)) {
|
|
return _server.requestAuthentication();
|
|
}
|
|
if(_allowRestartToPortal)
|
|
{
|
|
String response = "";
|
|
buildConfirmHtml(response, "Restarting. Connect to ESP access point to reconfigure WiFi.", 0);
|
|
_server.send(200, "text/html", response);
|
|
waitAndProcess(true, 2000);
|
|
_network->restartAndConfigureWifi();
|
|
}
|
|
});
|
|
_server.on("/method=get", [&]() {
|
|
if (_hasCredentials && !_server.authenticate(_credUser, _credPassword)) {
|
|
return _server.requestAuthentication();
|
|
}
|
|
String message = "";
|
|
bool restartEsp = processArgs(message);
|
|
if(restartEsp)
|
|
{
|
|
String response = "";
|
|
buildConfirmHtml(response, message);
|
|
_server.send(200, "text/html", response);
|
|
Serial.println(F("Restarting"));
|
|
|
|
waitAndProcess(true, 1000);
|
|
ESP.restart();
|
|
}
|
|
else
|
|
{
|
|
String response = "";
|
|
buildConfirmHtml(response, message, 3);
|
|
_server.send(200, "text/html", response);
|
|
waitAndProcess(false, 1000);
|
|
}
|
|
});
|
|
|
|
_server.begin();
|
|
}
|
|
|
|
bool WebCfgServer::processArgs(String& message)
|
|
{
|
|
bool configChanged = false;
|
|
bool clearMqttCredentials = false;
|
|
bool clearCredentials = false;
|
|
|
|
bool publishAuthData = false;
|
|
|
|
int count = _server.args();
|
|
for(int index = 0; index < count; index++)
|
|
{
|
|
String key = _server.argName(index);
|
|
String value = _server.arg(index);
|
|
|
|
if(key == "MQTTSERVER")
|
|
{
|
|
_preferences->putString(preference_mqtt_broker, value);
|
|
configChanged = true;
|
|
}
|
|
else if(key == "MQTTPORT")
|
|
{
|
|
_preferences->putInt(preference_mqtt_broker_port, value.toInt());
|
|
configChanged = true;
|
|
}
|
|
else if(key == "MQTTUSER")
|
|
{
|
|
if(value == "#")
|
|
{
|
|
clearMqttCredentials = true;
|
|
}
|
|
else
|
|
{
|
|
_preferences->putString(preference_mqtt_user, value);
|
|
configChanged = true;
|
|
}
|
|
}
|
|
else if(key == "MQTTPASS")
|
|
{
|
|
if(value != "*")
|
|
{
|
|
_preferences->putString(preference_mqtt_password, value);
|
|
configChanged = true;
|
|
}
|
|
}
|
|
else if(key == "MQTTPATH")
|
|
{
|
|
_preferences->putString(preference_mqtt_path, value);
|
|
configChanged = true;
|
|
}
|
|
else if(key == "HOSTNAME")
|
|
{
|
|
_preferences->putString(preference_hostname, value);
|
|
configChanged = true;
|
|
}
|
|
else if(key == "NETTIMEOUT")
|
|
{
|
|
_preferences->putInt(preference_network_timeout, value.toInt());
|
|
configChanged = true;
|
|
}
|
|
else if(key == "LSTINT")
|
|
{
|
|
_preferences->putInt(preference_query_interval_lockstate, value.toInt());
|
|
configChanged = true;
|
|
}
|
|
else if(key == "BATINT")
|
|
{
|
|
_preferences->putInt(preference_query_interval_battery, value.toInt());
|
|
configChanged = true;
|
|
}
|
|
else if(key == "PRDTMO")
|
|
{
|
|
_preferences->putInt(preference_presence_detection_timeout, value.toInt());
|
|
configChanged = true;
|
|
}
|
|
else if(key == "PUBAUTH")
|
|
{
|
|
publishAuthData = true;
|
|
}
|
|
else if(key == "CREDUSER")
|
|
{
|
|
if(value == "#")
|
|
{
|
|
clearCredentials = true;
|
|
}
|
|
else
|
|
{
|
|
_preferences->putString(preference_cred_user, value);
|
|
configChanged = true;
|
|
}
|
|
}
|
|
else if(key == "CREDPASS")
|
|
{
|
|
_preferences->putString(preference_cred_password, value);
|
|
configChanged = true;
|
|
}
|
|
else if(key == "NUKIPIN")
|
|
{
|
|
if(value == "#")
|
|
{
|
|
message = "PIN cleared";
|
|
_nuki->setPin(0xffff);
|
|
}
|
|
else
|
|
{
|
|
message = "PIN saved";
|
|
_nuki->setPin(value.toInt());
|
|
}
|
|
}
|
|
}
|
|
|
|
if(_preferences->getBool(preference_publish_authdata) != publishAuthData)
|
|
{
|
|
_preferences->putBool(preference_publish_authdata, publishAuthData);
|
|
configChanged = true;
|
|
}
|
|
|
|
if(clearMqttCredentials)
|
|
{
|
|
_preferences->putString(preference_mqtt_user, "");
|
|
_preferences->putString(preference_mqtt_password, "");
|
|
configChanged = true;
|
|
}
|
|
|
|
if(clearCredentials)
|
|
{
|
|
_preferences->putString(preference_cred_user, "");
|
|
_preferences->putString(preference_cred_password, "");
|
|
configChanged = true;
|
|
}
|
|
|
|
if(configChanged)
|
|
{
|
|
message = "Configuration saved ... restarting.";
|
|
_enabled = false;
|
|
_preferences->end();
|
|
}
|
|
|
|
return configChanged;
|
|
}
|
|
|
|
void WebCfgServer::update()
|
|
{
|
|
if(!_enabled) return;
|
|
|
|
_server.handleClient();
|
|
}
|
|
|
|
void WebCfgServer::buildHtml(String& response)
|
|
{
|
|
buildHtmlHeader(response);
|
|
|
|
response.concat("<br><h3>Info</h3>\n");
|
|
|
|
String version = " ";
|
|
version.concat(nuki_hub_version);
|
|
|
|
char lockstateArr[20];
|
|
NukiLock::lockstateToString(_nuki->keyTurnerState().lockState, lockstateArr);
|
|
String lockState = " ";
|
|
lockState.concat(lockstateArr);
|
|
|
|
response.concat("<table>");
|
|
printParameter(response, "Paired", _nuki->isPaired() ? " Yes" : " No");
|
|
printParameter(response, "MQTT Connected", _network->isMqttConnected() ? " Yes" : " No");
|
|
printParameter(response, "Lock state", lockState.c_str());
|
|
printParameter(response, "Firmware", version.c_str());
|
|
response.concat("</table><br><br>");
|
|
|
|
response.concat("<FORM ACTION=method=get >");
|
|
|
|
response.concat("<h3>Configuration</h3>");
|
|
response.concat("<table>");
|
|
printInputField(response, "MQTTSERVER", "MQTT Broker", _preferences->getString(preference_mqtt_broker).c_str(), 100);
|
|
printInputField(response, "MQTTPORT", "MQTT Broker port", _preferences->getInt(preference_mqtt_broker_port), 5);
|
|
printInputField(response, "MQTTUSER", "MQTT User (# to clear)", _preferences->getString(preference_mqtt_user).c_str(), 30);
|
|
printInputField(response, "MQTTPASS", "MQTT Password", "*", 30, true);
|
|
printInputField(response, "MQTTPATH", "MQTT Path", _preferences->getString(preference_mqtt_path).c_str(), 180);
|
|
printInputField(response, "HOSTNAME", "Host name", _preferences->getString(preference_hostname).c_str(), 100);
|
|
printInputField(response, "NETTIMEOUT", "Network Timeout until restart (seconds; -1 to disable)", _preferences->getInt(preference_network_timeout), 5);
|
|
printInputField(response, "LSTINT", "Query interval lock state (seconds)", _preferences->getInt(preference_query_interval_lockstate), 10);
|
|
printInputField(response, "BATINT", "Query interval battery (seconds)", _preferences->getInt(preference_query_interval_battery), 10);
|
|
printCheckBox(response, "PUBAUTH", "Publish auth data (May reduce battery life)", _preferences->getBool(preference_publish_authdata));
|
|
printInputField(response, "PRDTMO", "Presence detection timeout (seconds; -1 to disable)", _preferences->getInt(preference_presence_detection_timeout), 10);
|
|
response.concat("</table>");
|
|
|
|
response.concat("<br><INPUT TYPE=SUBMIT NAME=\"submit\" VALUE=\"Save\">");
|
|
|
|
response.concat("</FORM><BR><BR>");
|
|
|
|
response.concat("<h3>Credentials</h3>");
|
|
response.concat("<form method=\"get\" action=\"/cred\">");
|
|
response.concat("<button type=\"submit\">Edit</button>");
|
|
response.concat("</form>");
|
|
|
|
if(_allowRestartToPortal)
|
|
{
|
|
response.concat("<br><br><h3>WiFi</h3>");
|
|
response.concat("<form method=\"get\" action=\"/wifi\">");
|
|
response.concat("<button type=\"submit\">Restart and configure wifi</button>");
|
|
}
|
|
response.concat("</form>");
|
|
|
|
response.concat("</BODY>\n");
|
|
response.concat("</HTML>\n");
|
|
}
|
|
|
|
|
|
void WebCfgServer::buildCredHtml(String &response)
|
|
{
|
|
buildHtmlHeader(response);
|
|
|
|
response.concat("<FORM ACTION=method=get >");
|
|
response.concat("<h3>Credentials</h3>");
|
|
response.concat("<table>");
|
|
printInputField(response, "CREDUSER", "User (# to clear)", _preferences->getString(preference_cred_user).c_str(), 20);
|
|
printInputField(response, "CREDPASS", "Password", "*", 30, true);
|
|
response.concat("</table>");
|
|
response.concat("<br><INPUT TYPE=SUBMIT NAME=\"submit\" VALUE=\"Save\">");
|
|
response.concat("</FORM>");
|
|
|
|
response.concat("<br><br><FORM ACTION=method=get >");
|
|
response.concat("<h3>NUKI Pin Code</h3>");
|
|
response.concat("<table>");
|
|
printInputField(response, "NUKIPIN", "PIN Code (# to clear)", "*", 20, true);
|
|
response.concat("</table>");
|
|
response.concat("<br><INPUT TYPE=SUBMIT NAME=\"submit\" VALUE=\"Save\">");
|
|
response.concat("</FORM>");
|
|
|
|
_confirmCode = generateConfirmCode();
|
|
response.concat("<br><br><h3>Unpair NUKI</h3>");
|
|
response.concat("<form method=\"get\" action=\"/unpair\">");
|
|
String message = "Type ";
|
|
message.concat(_confirmCode);
|
|
message.concat(" to confirm unpair");
|
|
printInputField(response, "CONFIRMTOKEN", message.c_str(), "", 10);
|
|
response.concat("<br><br><button type=\"submit\">OK</button>");
|
|
|
|
response.concat("</BODY>\n");
|
|
response.concat("</HTML>\n");
|
|
}
|
|
|
|
void WebCfgServer::buildConfirmHtml(String &response, const String &message, uint32_t redirectDelay)
|
|
{
|
|
String delay(redirectDelay);
|
|
|
|
response.concat("<HTML>\n");
|
|
response.concat("<HEAD>\n");
|
|
response.concat("<TITLE>NUKI Hub</TITLE>\n");
|
|
response.concat("<meta http-equiv=\"Refresh\" content=\"");
|
|
response.concat(redirectDelay);
|
|
response.concat("; url=/\" />");
|
|
response.concat("\n</HEAD>\n");
|
|
response.concat("<BODY>\n");
|
|
response.concat(message);
|
|
|
|
response.concat("</BODY>\n");
|
|
response.concat("</HTML>\n");
|
|
}
|
|
|
|
|
|
void WebCfgServer::buildConfigureWifiHtml(String &response)
|
|
{
|
|
buildHtmlHeader(response);
|
|
|
|
response.concat("<h3>WiFi</h3>");
|
|
response.concat("Click confirm to restart ESP into WiFi configuration mode. After restart, connect to ESP access point to reconfigure WiFI.<br><br>");
|
|
response.concat("<form method=\"get\" action=\"/wifimanager\">");
|
|
response.concat("<button type=\"submit\">Confirm</button>");
|
|
response.concat("</form>");
|
|
|
|
response.concat("</BODY>\n");
|
|
response.concat("</HTML>\n");
|
|
}
|
|
|
|
//printInputField(response, "CONFIRMTOKEN", "Type confirm", "", 10);
|
|
|
|
|
|
//int count = _server.args();
|
|
//for(int index = 0; index < count; index++)
|
|
//{
|
|
//String key = _server.argName(index);
|
|
//String value = _server.arg(index);
|
|
void WebCfgServer::processUnpair()
|
|
{
|
|
String response = "";
|
|
if(_server.args() == 0)
|
|
{
|
|
buildConfirmHtml(response, "Confirm code is invalid.", 3);
|
|
_server.send(200, "text/html", response);
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
String key = _server.argName(0);
|
|
String value = _server.arg(0);
|
|
|
|
if(key != "CONFIRMTOKEN" || value != _confirmCode)
|
|
{
|
|
buildConfirmHtml(response, "Confirm code is invalid.", 3);
|
|
_server.send(200, "text/html", response);
|
|
return;
|
|
}
|
|
}
|
|
|
|
buildConfirmHtml(response, "Unpairing NUKI and restarting.", 3);
|
|
_server.send(200, "text/html", response);
|
|
_nuki->unpair();
|
|
waitAndProcess(false, 1000);
|
|
ESP.restart();
|
|
}
|
|
|
|
void WebCfgServer::buildHtmlHeader(String &response)
|
|
{
|
|
response.concat("<HTML>\n");
|
|
response.concat("<HEAD>\n");
|
|
response.concat("<TITLE>NUKI Hub</TITLE>\n");
|
|
response.concat("</HEAD>\n");
|
|
response.concat("<BODY>\n");
|
|
}
|
|
|
|
void WebCfgServer::printInputField(String& response,
|
|
const char *token,
|
|
const char *description,
|
|
const char *value,
|
|
const size_t maxLength,
|
|
const bool isPassword)
|
|
{
|
|
char maxLengthStr[20];
|
|
|
|
itoa(maxLength, maxLengthStr, 10);
|
|
|
|
response.concat("<tr>");
|
|
response.concat("<td>");
|
|
response.concat(description);
|
|
response.concat("</td>");
|
|
response.concat("<td>");
|
|
response.concat(" <INPUT TYPE=");
|
|
response.concat(isPassword ? "PASSWORD" : "TEXT");
|
|
response.concat(" VALUE=\"");
|
|
response.concat(value);
|
|
response.concat("\" NAME=\"");
|
|
response.concat(token);
|
|
response.concat("\" SIZE=\"25\" MAXLENGTH=\"");
|
|
response.concat(maxLengthStr);
|
|
response.concat("\\\"/>");
|
|
response.concat("</td>");
|
|
response.concat("</tr>");
|
|
}
|
|
|
|
void WebCfgServer::printInputField(String& response,
|
|
const char *token,
|
|
const char *description,
|
|
const int value,
|
|
size_t maxLength)
|
|
{
|
|
char valueStr[20];
|
|
itoa(value, valueStr, 10);
|
|
printInputField(response, token, description, valueStr, maxLength);
|
|
}
|
|
|
|
void WebCfgServer::printCheckBox(String &response, const char *token, const char *description, const bool value)
|
|
{
|
|
response.concat("<tr>");
|
|
response.concat("<td>");
|
|
response.concat(description);
|
|
response.concat("</td>");
|
|
response.concat("<td>");
|
|
response.concat(" <INPUT TYPE=");
|
|
response.concat("checkbox ");
|
|
response.concat("NAME=\"");
|
|
response.concat(token);
|
|
response.concat("\"");
|
|
response.concat(value ? " checked=\"checked\"" : "");
|
|
response.concat(" />");
|
|
response.concat("</td>");
|
|
response.concat("</tr>");
|
|
}
|
|
|
|
void WebCfgServer::printParameter(String& response, const char *description, const char *value)
|
|
{
|
|
response.concat("<tr>");
|
|
response.concat("<td>");
|
|
response.concat(description);
|
|
response.concat("</td>");
|
|
response.concat("<td>");
|
|
response.concat(value);
|
|
response.concat("</td>");
|
|
response.concat("</tr>");
|
|
|
|
}
|
|
|
|
|
|
String WebCfgServer::generateConfirmCode()
|
|
{
|
|
int code = random(1000,9999);
|
|
return String(code);
|
|
}
|
|
|
|
void WebCfgServer::waitAndProcess(const bool blocking, const uint32_t duration)
|
|
{
|
|
unsigned long timeout = millis() + duration;
|
|
while(millis() < timeout)
|
|
{
|
|
_server.handleClient();
|
|
if(blocking)
|
|
{
|
|
delay(10);
|
|
}
|
|
else
|
|
{
|
|
vTaskDelay( 50 / portTICK_PERIOD_MS);
|
|
}
|
|
}
|
|
}
|
|
|