mirror of
https://github.com/HASwitchPlate/openHASP.git
synced 2025-07-26 20:56:37 +00:00
Add Telnet authentication
This commit is contained in:
parent
0ad0f7c02d
commit
3f228e3f47
@ -54,6 +54,22 @@ Syslog syslog(syslogClient, debugSyslogHost.c_str(), debugSyslogPort, debugAppNa
|
|||||||
unsigned long debugLastMillis = 0;
|
unsigned long debugLastMillis = 0;
|
||||||
uint16_t debugTelePeriod = 300;
|
uint16_t debugTelePeriod = 300;
|
||||||
|
|
||||||
|
String debugHaspHeader()
|
||||||
|
{
|
||||||
|
String header((char *)0);
|
||||||
|
header.reserve(256);
|
||||||
|
header = F(" _____ _____ _____ _____\r\n"
|
||||||
|
" | | | _ | __| _ |\r\n"
|
||||||
|
" | | |__ | __|\r\n"
|
||||||
|
" |__|__|__|__|_____|__|\r\n"
|
||||||
|
" Home Automation Switch Plate\r\n");
|
||||||
|
char buffer[128];
|
||||||
|
snprintf(buffer, sizeof(buffer), PSTR(" Open Hardware edition v%u.%u.%u\r\n"), HASP_VERSION_MAJOR,
|
||||||
|
HASP_VERSION_MINOR, HASP_VERSION_REVISION);
|
||||||
|
header += buffer;
|
||||||
|
return header;
|
||||||
|
}
|
||||||
|
|
||||||
void debugStart()
|
void debugStart()
|
||||||
{
|
{
|
||||||
#if defined(ARDUINO_ARCH_ESP32)
|
#if defined(ARDUINO_ARCH_ESP32)
|
||||||
@ -64,10 +80,7 @@ void debugStart()
|
|||||||
|
|
||||||
Serial.flush();
|
Serial.flush();
|
||||||
Serial.println();
|
Serial.println();
|
||||||
Serial.printf_P(PSTR(" _____ _____ _____ _____\r\n | | | _ | __| _ |\r\n"
|
Serial.println(debugHaspHeader());
|
||||||
" | | |__ | __|\r\n |__|__|__|__|_____|__|\r\n"
|
|
||||||
" Home Automation Switch Plate\r\n Open Hardware edition v%u.%u.%u\r\n\r\n"),
|
|
||||||
HASP_VERSION_MAJOR, HASP_VERSION_MINOR, HASP_VERSION_REVISION);
|
|
||||||
Serial.flush();
|
Serial.flush();
|
||||||
|
|
||||||
// prepare syslog configuration here (can be anywhere before first call of
|
// prepare syslog configuration here (can be anywhere before first call of
|
||||||
|
@ -3,6 +3,8 @@
|
|||||||
|
|
||||||
#include "ArduinoJson.h"
|
#include "ArduinoJson.h"
|
||||||
|
|
||||||
|
String debugHaspHeader(void);
|
||||||
|
|
||||||
void debugPreSetup(JsonObject settings);
|
void debugPreSetup(JsonObject settings);
|
||||||
void debugSetup(JsonObject settings);
|
void debugSetup(JsonObject settings);
|
||||||
void debugLoop(void);
|
void debugLoop(void);
|
||||||
|
@ -3,6 +3,7 @@
|
|||||||
|
|
||||||
#include "hasp_conf.h"
|
#include "hasp_conf.h"
|
||||||
#include "hasp_log.h"
|
#include "hasp_log.h"
|
||||||
|
#include "hasp_debug.h"
|
||||||
#include "hasp_config.h"
|
#include "hasp_config.h"
|
||||||
#include "hasp_dispatch.h"
|
#include "hasp_dispatch.h"
|
||||||
#include "hasp_telnet.h"
|
#include "hasp_telnet.h"
|
||||||
@ -16,11 +17,31 @@
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
//#define telnetInputMax 128; // Size of user input buffer for user telnet session
|
//#define telnetInputMax 128; // Size of user input buffer for user telnet session
|
||||||
static char telnetInputBuffer[128];
|
|
||||||
|
|
||||||
uint8_t telnetEnabled = true; // Enable telnet debug output
|
|
||||||
WiFiServer * telnetServer; //(23);
|
WiFiServer * telnetServer; //(23);
|
||||||
WiFiClient * telnetClient;
|
WiFiClient * telnetClient;
|
||||||
|
uint8_t telnetEnabled = true; // Enable telnet debug output
|
||||||
|
uint8_t telnetLoginAttempt = 0; // Initial attempt
|
||||||
|
uint8_t telnetLoginState = 0; // Unauthenticated
|
||||||
|
uint8_t telnetInputIndex = 0; // Empty buffer
|
||||||
|
char telnetInputBuffer[128];
|
||||||
|
|
||||||
|
bool telnetExitCommand()
|
||||||
|
{
|
||||||
|
char buffer[128];
|
||||||
|
snprintf_P(buffer, sizeof(buffer), PSTR("exit"));
|
||||||
|
if(strcmp(telnetInputBuffer, buffer) == 0 || telnetLoginAttempt >= 3) {
|
||||||
|
snprintf_P(buffer, sizeof(buffer), PSTR("TELNET: Closing session from %s"),
|
||||||
|
telnetClient->remoteIP().toString().c_str());
|
||||||
|
debugPrintln(buffer);
|
||||||
|
telnetClient->stop();
|
||||||
|
telnetLoginState = 0; // Unauthenticated
|
||||||
|
telnetInputIndex = 0; // Empty buffer
|
||||||
|
telnetLoginAttempt = 0; // Initial attempt
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void telnetSetup(const JsonObject & settings)
|
void telnetSetup(const JsonObject & settings)
|
||||||
{
|
{
|
||||||
@ -28,29 +49,35 @@ void telnetSetup(const JsonObject & settings)
|
|||||||
|
|
||||||
if(telnetEnabled) { // Setup telnet server for remote debug output
|
if(telnetEnabled) { // Setup telnet server for remote debug output
|
||||||
telnetServer = new WiFiServer(23);
|
telnetServer = new WiFiServer(23);
|
||||||
telnetClient = new WiFiClient;
|
|
||||||
if(telnetServer) {
|
if(telnetServer) {
|
||||||
|
telnetClient = new WiFiClient;
|
||||||
telnetServer->setNoDelay(true);
|
telnetServer->setNoDelay(true);
|
||||||
telnetServer->begin();
|
telnetServer->begin();
|
||||||
debugPrintln(String(F("TELNET: debug server enabled at telnet:")) + WiFi.localIP().toString());
|
debugPrintln(String(F("TELNET: Debug console enabled at telnet://")) + WiFi.localIP().toString());
|
||||||
} else {
|
} else {
|
||||||
errorPrintln(F("TELNET: %sFailed to create telnet server"));
|
errorPrintln(F("TELNET: %sFailed to start telnet server"));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void telnetLoop(bool isConnected)
|
void telnetLoop()
|
||||||
{ // Basic telnet client handling code from: https://gist.github.com/tablatronix/4793677ca748f5f584c95ec4a2b10303
|
{ // Basic telnet client handling code from: https://gist.github.com/tablatronix/4793677ca748f5f584c95ec4a2b10303
|
||||||
if(!isConnected) return;
|
|
||||||
|
|
||||||
static unsigned long telnetInputIndex = 0;
|
|
||||||
if(telnetServer && telnetServer->hasClient()) { // client is connected
|
if(telnetServer && telnetServer->hasClient()) { // client is connected
|
||||||
if(!*telnetClient || !telnetClient->connected()) {
|
if(!*telnetClient || !telnetClient->connected()) {
|
||||||
if(telnetClient) {
|
if(telnetClient) {
|
||||||
telnetClient->stop(); // client disconnected
|
telnetClient->stop(); // client disconnected
|
||||||
}
|
}
|
||||||
*telnetClient = telnetServer->available(); // ready for new client
|
*telnetClient = telnetServer->available(); // ready for new client
|
||||||
|
char buffer[128];
|
||||||
|
snprintf_P(buffer, sizeof(buffer), PSTR("TELNET: Client connected from %s"),
|
||||||
|
telnetClient->remoteIP().toString().c_str());
|
||||||
|
debugPrintln(buffer);
|
||||||
|
snprintf_P(buffer, sizeof(buffer), PSTR("\r\nUsername: "));
|
||||||
|
telnetClient->print(buffer);
|
||||||
|
telnetLoginState = 0; // Unauthenticated
|
||||||
telnetInputIndex = 0; // reset input buffer index
|
telnetInputIndex = 0; // reset input buffer index
|
||||||
|
telnetLoginAttempt = 0; // Initial attempt
|
||||||
} else {
|
} else {
|
||||||
telnetServer->available().stop(); // have client, block new connections
|
telnetServer->available().stop(); // have client, block new connections
|
||||||
}
|
}
|
||||||
@ -63,22 +90,76 @@ void telnetLoop(bool isConnected)
|
|||||||
char telnetInputByte = telnetClient->read(); // Read client byte
|
char telnetInputByte = telnetClient->read(); // Read client byte
|
||||||
// debugPrintln(String("telnet in: 0x") + String(telnetInputByte, HEX));
|
// debugPrintln(String("telnet in: 0x") + String(telnetInputByte, HEX));
|
||||||
switch(telnetInputByte) {
|
switch(telnetInputByte) {
|
||||||
case 0x3:
|
case 0x01:
|
||||||
case 0x5:
|
case 0x03:
|
||||||
|
case 0x05:
|
||||||
case 0xff:
|
case 0xff:
|
||||||
|
case 0xfe:
|
||||||
|
case 0xfd:
|
||||||
|
case 0xfc:
|
||||||
|
case 0xfb:
|
||||||
case 0xf1:
|
case 0xf1:
|
||||||
|
case 0x1f:
|
||||||
|
case 10:
|
||||||
telnetInputIndex = 0;
|
telnetInputIndex = 0;
|
||||||
break;
|
break;
|
||||||
case 10:
|
case 0x08: // Backspace
|
||||||
|
if(telnetInputIndex > 0) {
|
||||||
|
telnetInputIndex--;
|
||||||
|
}
|
||||||
|
break;
|
||||||
case 13:
|
case 13:
|
||||||
telnetInputBuffer[telnetInputIndex] = 0; // null terminate our char array
|
telnetInputBuffer[telnetInputIndex] = 0; // null terminate our char array
|
||||||
if(telnetInputIndex > 0) dispatchCommand(telnetInputBuffer);
|
switch(telnetLoginState) {
|
||||||
|
case 0: {
|
||||||
|
char buffer[128];
|
||||||
|
snprintf_P(buffer, sizeof(buffer), PSTR("Password: %c%c%c"), 0xFF, 0xFB, 0x01);
|
||||||
|
telnetClient->print(buffer);
|
||||||
|
snprintf_P(buffer, sizeof(buffer), PSTR("admin"));
|
||||||
|
telnetLoginState =
|
||||||
|
strcmp(telnetInputBuffer, buffer) == 0 ? 10 : 99; // Username OK=1 or NOK=99
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 10:
|
||||||
|
case 99: {
|
||||||
|
char buffer[128];
|
||||||
|
snprintf_P(buffer, sizeof(buffer), PSTR("%c%c%c"), 0xFF, 0xFC, 0x01);
|
||||||
|
telnetClient->println(buffer);
|
||||||
|
snprintf_P(buffer, sizeof(buffer), PSTR("haspadmin"));
|
||||||
|
if(telnetLoginState == 10 && strcmp(telnetInputBuffer, buffer) == 0) { // Authenticated
|
||||||
|
telnetLoginState = 255; // Authenticated
|
||||||
|
telnetLoginAttempt = 0; // Initial attempt
|
||||||
|
telnetClient->println(debugHaspHeader());
|
||||||
|
snprintf_P(buffer, sizeof(buffer), PSTR("TELNET: Client login from %s"),
|
||||||
|
telnetClient->remoteIP().toString().c_str());
|
||||||
|
debugPrintln(buffer);
|
||||||
|
} else {
|
||||||
|
telnetLoginState = 0; // Unauthorized
|
||||||
|
telnetLoginAttempt++; // Subsequent attempt
|
||||||
|
snprintf_P(buffer, sizeof(buffer), PSTR("Authorization failed!\r\n"));
|
||||||
|
telnetClient->println(buffer);
|
||||||
|
snprintf_P(buffer, sizeof(buffer), PSTR("TELNET: %%Incorrect login attempt from %s"),
|
||||||
|
telnetClient->remoteIP().toString().c_str());
|
||||||
|
errorPrintln(buffer);
|
||||||
|
if(telnetLoginAttempt >= 3) {
|
||||||
|
telnetExitCommand();
|
||||||
|
} else {
|
||||||
|
snprintf_P(buffer, sizeof(buffer), PSTR("Username: "));
|
||||||
|
telnetClient->print(buffer);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
if(telnetInputIndex > 0 && !telnetExitCommand()) {
|
||||||
|
dispatchCommand(telnetInputBuffer);
|
||||||
|
}
|
||||||
|
}
|
||||||
telnetInputIndex = 0;
|
telnetInputIndex = 0;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
if(telnetInputIndex <
|
// If we have room left in our buffer add the current byte
|
||||||
sizeof(telnetInputBuffer)) { // If we have room left in our buffer add the current byte
|
if(telnetInputIndex < sizeof(telnetInputBuffer) - 1 && telnetInputByte >= 0x20) {
|
||||||
telnetInputBuffer[telnetInputIndex] = telnetInputByte;
|
telnetInputBuffer[telnetInputIndex] = telnetInputByte;
|
||||||
telnetInputIndex++;
|
telnetInputIndex++;
|
||||||
}
|
}
|
||||||
@ -87,19 +168,22 @@ void telnetLoop(bool isConnected)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
void telnetPrintln(const char * msg)
|
void telnetPrintln(const char * msg)
|
||||||
{
|
{
|
||||||
if(telnetEnabled && telnetClient && telnetClient->connected()) {
|
if(telnetEnabled && telnetClient && telnetClient->connected() && telnetLoginState == 255) {
|
||||||
telnetClient->println(msg);
|
telnetClient->println(msg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
void telnetPrint(const char * msg)
|
void telnetPrint(const char * msg)
|
||||||
{
|
{
|
||||||
if(telnetEnabled && telnetClient && telnetClient->connected()) {
|
if(telnetEnabled && telnetClient && telnetClient->connected() && telnetLoginState == 255) {
|
||||||
telnetClient->print(msg);
|
telnetClient->print(msg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
bool telnetGetConfig(const JsonObject & settings)
|
bool telnetGetConfig(const JsonObject & settings)
|
||||||
{
|
{
|
||||||
settings[FPSTR(F_CONFIG_ENABLE)] = telnetEnabled;
|
settings[FPSTR(F_CONFIG_ENABLE)] = telnetEnabled;
|
||||||
@ -108,8 +192,6 @@ bool telnetGetConfig(const JsonObject & settings)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
bool telnetSetConfig(const JsonObject & settings)
|
bool telnetSetConfig(const JsonObject & settings)
|
||||||
{
|
{
|
||||||
configOutput(settings);
|
configOutput(settings);
|
||||||
|
@ -8,7 +8,7 @@
|
|||||||
#include "ArduinoJson.h"
|
#include "ArduinoJson.h"
|
||||||
|
|
||||||
void telnetSetup(const JsonObject & settings);
|
void telnetSetup(const JsonObject & settings);
|
||||||
void telnetLoop(bool isConnected);
|
void telnetLoop(void);
|
||||||
void telnetStop(void);
|
void telnetStop(void);
|
||||||
|
|
||||||
void telnetPrint(const char * msg);
|
void telnetPrint(const char * msg);
|
||||||
|
Loading…
x
Reference in New Issue
Block a user