This repository was archived by the owner on Feb 4, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 8
/
Copy pathSimpleAuthentication.ino
238 lines (193 loc) · 6.91 KB
/
SimpleAuthentication.ino
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
/****************************************************************************************************************************
SimpleAuthentication.ino - Simple Arduino web server sample for ESP8266/ESP32 AT-command shield
For ESP8266/ESP32 AT-command running shields
ESP8266_AT_WebServer is a library for the ESP8266/ESP32 AT-command shields to run WebServer
Based on and modified from ESP8266 https://github.com/esp8266/Arduino/releases
Built by Khoi Hoang https://github.com/khoih-prog/ESP8266_AT_WebServer
Licensed under MIT license
Original author:
@file Esp8266WebServer.h
@author Ivan Grokhotkov
*****************************************************************************************************************************/
// Credits of [Miguel Alexandre Wisintainer](https://github.com/tcpipchip) for this simple yet effective method
// For some STM32, there is only definition of Serial in variant.h, and is used for Serial/USB Debugging
// For example, in Nucleo-144 F767ZI original variant.h
//
// #define SERIAL_PORT_MONITOR Serial
// #define SERIAL_PORT_HARDWARE Serial
//
// To use ESP8266/ESP32-AT, we need another Serial, such as Serial1
// To do this, first, in corresponding variant.h, modify as follows:
// #define SERIAL_PORT_HARDWARE Serial1
//
// then assign pins D0 = RX/D1 = TX to be Hardware Serial1 by putting in sketch as follows:
//
// #define EspSerial SERIAL_PORT_HARDWARE //Serial1
// HardwareSerial Serial1(D0, D1);
//
// This must be included in defines.h for each board you'd like to use ESPSerial as Serial1
//
// The pin usage must be modified according to your boards.
#include "defines.h"
int status = WL_IDLE_STATUS; // the Wifi radio's status
int reqCount = 0; // number of requests received
ESP8266_AT_WebServer server(80);
//Check if header is present and correct
bool is_authenticated()
{
Serial.println(F("Enter is_authenticated"));
if (server.hasHeader(F("Cookie")))
{
Serial.print(F("Found cookie: "));
String cookie = server.header(F("Cookie"));
Serial.println(cookie);
if (cookie.indexOf(F("ESPSESSIONID=1")) != -1)
{
Serial.println(F("Authentication Successful"));
return true;
}
}
Serial.println(F("Authentication Failed"));
return false;
}
//login page, also called for disconnect
void handleLogin()
{
String msg;
if (server.hasHeader(F("Cookie")))
{
Serial.print(F("Found cookie: "));
String cookie = server.header(F("Cookie"));
Serial.println(cookie);
}
if (server.hasArg("DISCONNECT"))
{
Serial.println(F("Disconnection"));
server.sendHeader(F("Location"), F("/login"));
server.sendHeader(F("Cache-Control"), F("no-cache"));
server.sendHeader(F("Set-Cookie"), F("ESPSESSIONID=0"));
server.send(301);
return;
}
if (server.hasArg(F("USERNAME")) && server.hasArg(F("PASSWORD")))
{
if (server.arg(F("USERNAME")) == F("admin") && server.arg(F("PASSWORD")) == F("admin"))
{
server.sendHeader(F("Location"), F("/"));
server.sendHeader(F("Cache-Control"), F("no-cache"));
server.sendHeader(F("Set-Cookie"), F("ESPSESSIONID=1"));
server.send(301);
Serial.println(F("Log in Successful"));
return;
}
msg = F("Wrong username/password! try again.");
Serial.println(F("Log in Failed"));
}
String content = F("<html><body><form action='/login' method='POST'>To log in, please use : admin/admin<br>");
content += F("User:<input type='text' name='USERNAME' placeholder='user name'><br>");
content += F("Password:<input type='password' name='PASSWORD' placeholder='password'><br>");
content += F("<input type='submit' name='SUBMIT' value='Submit'></form>");
content += msg;
content += F("<br>");
content += F("You also can go <a href='/inline'>here</a></body></html>");
server.send(200, F("text/html"), content);
}
#define ORIGINAL_STR_LEN 384
//root page can be accessed only if authentication is ok
void handleRoot()
{
static String content;
static uint16_t previousStrLen = ORIGINAL_STR_LEN;
Serial.println(F("Enter handleRoot"));
if (!is_authenticated())
{
server.sendHeader(F("Location"), F("/login"));
server.sendHeader(F("Cache-Control"), F("no-cache"));
server.send(301);
return;
}
content = F("<html><body><H2>Hello, you successfully connected to ESP8266_AT on ");
content += BOARD_NAME;
content += F("!</H2><br>");
if (server.hasHeader(F("User-Agent")))
{
content += F("the user agent used is : ");
content += server.header(F("User-Agent"));
content += F("<br><br>");
}
content += F("You can access this page until you <a href=\"/login?DISCONNECT=YES\">disconnect</a></body></html>");
if (content.length() > previousStrLen)
{
AT_LOGERROR3(F("String Len > "), previousStrLen, F(", extend to"), content.length() + 48);
previousStrLen = content.length() + 48;
content.reserve(previousStrLen);
}
else
{
AT_LOGDEBUG1(F("Len ="), content.length());
server.send(200, F("text/html"), content);
}
}
//no need authentication
void handleNotFound()
{
String message = F("File Not Found\n\n");
message += F("URI: ");
message += server.uri();
message += F("\nMethod: ");
message += (server.method() == HTTP_GET) ? F("GET") : F("POST");
message += F("\nArguments: ");
message += server.args();
message += F("\n");
for (uint8_t i = 0; i < server.args(); i++)
{
message += " " + server.argName(i) + ": " + server.arg(i) + "\n";
}
server.send(404, F("text/plain"), message);
}
void setup()
{
Serial.begin(115200);
while (!Serial && millis() < 5000);
Serial.print(F("\nStarting SimpleAuthentication on ")); Serial.print(BOARD_NAME);
Serial.print(F(" with ")); Serial.println(SHIELD_TYPE);
Serial.println(ESP8266_AT_WEBSERVER_VERSION);
// initialize serial for ESP module
EspSerial.begin(115200);
// initialize ESP module
WiFi.init(&EspSerial);
// check for the presence of the shield
if (WiFi.status() == WL_NO_SHIELD)
{
Serial.println(F("WiFi shield not present"));
// don't continue
while (true);
}
// attempt to connect to WiFi network
while ( status != WL_CONNECTED)
{
Serial.print(F("Connecting to WPA SSID: "));
Serial.println(ssid);
// Connect to WPA/WPA2 network
status = WiFi.begin(ssid, pass);
}
server.on(F("/"), handleRoot);
server.on(F("/login"), handleLogin);
server.on(F("/inline"), []()
{
server.send(200, F("text/plain"), F("This works without need of authentication"));
});
server.onNotFound(handleNotFound);
//here the list of headers to be recorded
const char * headerkeys[] = {"User-Agent", "Cookie"} ;
size_t headerkeyssize = sizeof(headerkeys) / sizeof(char*);
//ask server to track these headers
server.collectHeaders(headerkeys, headerkeyssize);
server.begin();
Serial.print(F("HTTP server started @ "));
Serial.println(WiFi.localIP());
}
void loop()
{
server.handleClient();
}