2015-05-22 14:19:01 +02:00
|
|
|
/**
|
|
|
|
* @file WebSocketsServer.cpp
|
|
|
|
* @date 20.05.2015
|
|
|
|
* @author Markus Sattler
|
|
|
|
*
|
|
|
|
* Copyright (c) 2015 Markus Sattler. All rights reserved.
|
|
|
|
* This file is part of the WebSockets for Arduino.
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library 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
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "WebSockets.h"
|
|
|
|
#include "WebSocketsServer.h"
|
|
|
|
|
|
|
|
WebSocketsServer::WebSocketsServer(uint16_t port) {
|
|
|
|
_port = port;
|
|
|
|
_server = new WiFiServer(port);
|
2015-05-22 23:02:47 +02:00
|
|
|
|
|
|
|
_cbEvent = NULL;
|
|
|
|
|
2015-05-22 14:19:01 +02:00
|
|
|
}
|
2015-05-22 20:35:51 +02:00
|
|
|
|
2015-05-22 14:19:01 +02:00
|
|
|
WebSocketsServer::~WebSocketsServer() {
|
|
|
|
// todo how to close server?
|
|
|
|
}
|
|
|
|
|
2015-05-22 23:02:47 +02:00
|
|
|
/**
|
|
|
|
* calles to init the Websockets server
|
|
|
|
*/
|
2015-05-22 14:19:01 +02:00
|
|
|
void WebSocketsServer::begin(void) {
|
2015-05-22 20:35:51 +02:00
|
|
|
WSclient_t * client;
|
2015-05-22 14:19:01 +02:00
|
|
|
|
|
|
|
// init client storage
|
|
|
|
for(uint8_t i = 0; i < WEBSOCKETS_SERVER_CLIENT_MAX; i++) {
|
|
|
|
client = &_clients[i];
|
|
|
|
|
2015-05-22 14:40:46 +02:00
|
|
|
client->num = i;
|
2015-05-22 14:19:01 +02:00
|
|
|
client->cUrl = "";
|
|
|
|
client->cKey = "";
|
|
|
|
client->cProtocol = "";
|
|
|
|
client->cVersion = 0;
|
|
|
|
client->cIsUpgrade = false;
|
|
|
|
client->cIsWebsocket = false;
|
|
|
|
|
|
|
|
client->status = WSC_NOT_CONNECTED;
|
|
|
|
}
|
|
|
|
|
2015-05-24 15:40:47 +02:00
|
|
|
// todo find better seed
|
|
|
|
randomSeed(millis());
|
|
|
|
|
2015-05-22 14:19:01 +02:00
|
|
|
_server->begin();
|
|
|
|
}
|
|
|
|
|
2015-05-23 09:20:44 +02:00
|
|
|
/**
|
|
|
|
* called in arduino loop
|
|
|
|
*/
|
2015-05-22 14:19:01 +02:00
|
|
|
void WebSocketsServer::loop(void) {
|
|
|
|
handleNewClients();
|
|
|
|
handleClientData();
|
|
|
|
}
|
|
|
|
|
2015-05-22 23:02:47 +02:00
|
|
|
/**
|
|
|
|
* set callback function
|
|
|
|
* @param cbEvent WebSocketServerEvent
|
|
|
|
*/
|
|
|
|
void WebSocketsServer::onEvent(WebSocketServerEvent cbEvent) {
|
2015-05-23 23:51:32 +02:00
|
|
|
_cbEvent = cbEvent;
|
2015-05-22 23:02:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* send text data to client
|
|
|
|
* @param num uint8_t client id
|
|
|
|
* @param payload uint8_t *
|
|
|
|
* @param length size_t
|
2015-10-11 10:11:32 +02:00
|
|
|
* @param headerToPayload bool (see sendFrame for more details)
|
2015-05-22 23:02:47 +02:00
|
|
|
*/
|
2015-10-11 10:11:32 +02:00
|
|
|
void WebSocketsServer::sendTXT(uint8_t num, uint8_t * payload, size_t length, bool headerToPayload) {
|
2015-05-22 23:02:47 +02:00
|
|
|
if(num >= WEBSOCKETS_SERVER_CLIENT_MAX) {
|
|
|
|
return;
|
|
|
|
}
|
create overloaded functions for send and broadcast for easy usage
void sendTXT(uint8_t num, uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, const uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, char * payload, size_t length = 0);
void sendTXT(uint8_t num, const char * payload, size_t length = 0);
void sendTXT(uint8_t num, String payload);
void broadcastTXT(uint8_t * payload, size_t length = 0);
void broadcastTXT(const uint8_t * payload, size_t length = 0);
void broadcastTXT(char * payload, size_t length = 0);
void broadcastTXT(const char * payload, size_t length = 0);
void broadcastTXT(String payload);
void sendBIN(uint8_t num, uint8_t * payload, size_t length);
void sendBIN(uint8_t num, const uint8_t * payload, size_t length);
void broadcastBIN(uint8_t * payload, size_t length);
void broadcastBIN(const uint8_t * payload, size_t length);
send URL as payload on WStype_CONNECTED event
move Sec-WebSocket-Accept generation in function
2015-05-23 09:02:59 +02:00
|
|
|
if(length == 0) {
|
|
|
|
length = strlen((const char *) payload);
|
|
|
|
}
|
2015-05-22 23:02:47 +02:00
|
|
|
WSclient_t * client = &_clients[num];
|
|
|
|
if(clientIsConnected(client)) {
|
2015-10-11 10:11:32 +02:00
|
|
|
sendFrame(client, WSop_text, payload, length, false, true, headerToPayload);
|
2015-05-22 23:02:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
create overloaded functions for send and broadcast for easy usage
void sendTXT(uint8_t num, uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, const uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, char * payload, size_t length = 0);
void sendTXT(uint8_t num, const char * payload, size_t length = 0);
void sendTXT(uint8_t num, String payload);
void broadcastTXT(uint8_t * payload, size_t length = 0);
void broadcastTXT(const uint8_t * payload, size_t length = 0);
void broadcastTXT(char * payload, size_t length = 0);
void broadcastTXT(const char * payload, size_t length = 0);
void broadcastTXT(String payload);
void sendBIN(uint8_t num, uint8_t * payload, size_t length);
void sendBIN(uint8_t num, const uint8_t * payload, size_t length);
void broadcastBIN(uint8_t * payload, size_t length);
void broadcastBIN(const uint8_t * payload, size_t length);
send URL as payload on WStype_CONNECTED event
move Sec-WebSocket-Accept generation in function
2015-05-23 09:02:59 +02:00
|
|
|
void WebSocketsServer::sendTXT(uint8_t num, const uint8_t * payload, size_t length) {
|
|
|
|
sendTXT(num, (uint8_t *) payload, length);
|
|
|
|
}
|
|
|
|
|
2015-10-11 10:11:32 +02:00
|
|
|
void WebSocketsServer::sendTXT(uint8_t num, char * payload, size_t length, bool headerToPayload) {
|
|
|
|
sendTXT(num, (uint8_t *) payload, length, headerToPayload);
|
create overloaded functions for send and broadcast for easy usage
void sendTXT(uint8_t num, uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, const uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, char * payload, size_t length = 0);
void sendTXT(uint8_t num, const char * payload, size_t length = 0);
void sendTXT(uint8_t num, String payload);
void broadcastTXT(uint8_t * payload, size_t length = 0);
void broadcastTXT(const uint8_t * payload, size_t length = 0);
void broadcastTXT(char * payload, size_t length = 0);
void broadcastTXT(const char * payload, size_t length = 0);
void broadcastTXT(String payload);
void sendBIN(uint8_t num, uint8_t * payload, size_t length);
void sendBIN(uint8_t num, const uint8_t * payload, size_t length);
void broadcastBIN(uint8_t * payload, size_t length);
void broadcastBIN(const uint8_t * payload, size_t length);
send URL as payload on WStype_CONNECTED event
move Sec-WebSocket-Accept generation in function
2015-05-23 09:02:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebSocketsServer::sendTXT(uint8_t num, const char * payload, size_t length) {
|
|
|
|
sendTXT(num, (uint8_t *) payload, length);
|
|
|
|
}
|
|
|
|
|
2015-05-22 23:02:47 +02:00
|
|
|
void WebSocketsServer::sendTXT(uint8_t num, String payload) {
|
2015-05-23 07:33:26 +02:00
|
|
|
sendTXT(num, (uint8_t *) payload.c_str(), payload.length());
|
2015-05-22 23:02:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* send text data to client all
|
|
|
|
* @param payload uint8_t *
|
|
|
|
* @param length size_t
|
2015-10-11 10:11:32 +02:00
|
|
|
* @param headerToPayload bool (see sendFrame for more details)
|
2015-05-22 23:02:47 +02:00
|
|
|
*/
|
2015-10-11 10:11:32 +02:00
|
|
|
void WebSocketsServer::broadcastTXT(uint8_t * payload, size_t length, bool headerToPayload) {
|
2015-05-22 23:02:47 +02:00
|
|
|
WSclient_t * client;
|
create overloaded functions for send and broadcast for easy usage
void sendTXT(uint8_t num, uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, const uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, char * payload, size_t length = 0);
void sendTXT(uint8_t num, const char * payload, size_t length = 0);
void sendTXT(uint8_t num, String payload);
void broadcastTXT(uint8_t * payload, size_t length = 0);
void broadcastTXT(const uint8_t * payload, size_t length = 0);
void broadcastTXT(char * payload, size_t length = 0);
void broadcastTXT(const char * payload, size_t length = 0);
void broadcastTXT(String payload);
void sendBIN(uint8_t num, uint8_t * payload, size_t length);
void sendBIN(uint8_t num, const uint8_t * payload, size_t length);
void broadcastBIN(uint8_t * payload, size_t length);
void broadcastBIN(const uint8_t * payload, size_t length);
send URL as payload on WStype_CONNECTED event
move Sec-WebSocket-Accept generation in function
2015-05-23 09:02:59 +02:00
|
|
|
if(length == 0) {
|
|
|
|
length = strlen((const char *) payload);
|
|
|
|
}
|
2015-05-23 23:51:32 +02:00
|
|
|
|
2015-05-22 23:02:47 +02:00
|
|
|
for(uint8_t i = 0; i < WEBSOCKETS_SERVER_CLIENT_MAX; i++) {
|
|
|
|
client = &_clients[i];
|
|
|
|
if(clientIsConnected(client)) {
|
2015-10-11 10:11:32 +02:00
|
|
|
sendFrame(client, WSop_text, payload, length, false, true, headerToPayload);
|
2015-05-22 23:02:47 +02:00
|
|
|
}
|
2015-05-24 15:40:47 +02:00
|
|
|
#ifdef ESP8266
|
2015-05-23 23:51:32 +02:00
|
|
|
delay(0);
|
2015-05-24 15:40:47 +02:00
|
|
|
#endif
|
2015-05-22 23:02:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
create overloaded functions for send and broadcast for easy usage
void sendTXT(uint8_t num, uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, const uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, char * payload, size_t length = 0);
void sendTXT(uint8_t num, const char * payload, size_t length = 0);
void sendTXT(uint8_t num, String payload);
void broadcastTXT(uint8_t * payload, size_t length = 0);
void broadcastTXT(const uint8_t * payload, size_t length = 0);
void broadcastTXT(char * payload, size_t length = 0);
void broadcastTXT(const char * payload, size_t length = 0);
void broadcastTXT(String payload);
void sendBIN(uint8_t num, uint8_t * payload, size_t length);
void sendBIN(uint8_t num, const uint8_t * payload, size_t length);
void broadcastBIN(uint8_t * payload, size_t length);
void broadcastBIN(const uint8_t * payload, size_t length);
send URL as payload on WStype_CONNECTED event
move Sec-WebSocket-Accept generation in function
2015-05-23 09:02:59 +02:00
|
|
|
void WebSocketsServer::broadcastTXT(const uint8_t * payload, size_t length) {
|
|
|
|
broadcastTXT((uint8_t *) payload, length);
|
|
|
|
}
|
|
|
|
|
2015-10-11 10:11:32 +02:00
|
|
|
void WebSocketsServer::broadcastTXT(char * payload, size_t length, bool headerToPayload) {
|
|
|
|
broadcastTXT((uint8_t *) payload, length, headerToPayload);
|
create overloaded functions for send and broadcast for easy usage
void sendTXT(uint8_t num, uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, const uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, char * payload, size_t length = 0);
void sendTXT(uint8_t num, const char * payload, size_t length = 0);
void sendTXT(uint8_t num, String payload);
void broadcastTXT(uint8_t * payload, size_t length = 0);
void broadcastTXT(const uint8_t * payload, size_t length = 0);
void broadcastTXT(char * payload, size_t length = 0);
void broadcastTXT(const char * payload, size_t length = 0);
void broadcastTXT(String payload);
void sendBIN(uint8_t num, uint8_t * payload, size_t length);
void sendBIN(uint8_t num, const uint8_t * payload, size_t length);
void broadcastBIN(uint8_t * payload, size_t length);
void broadcastBIN(const uint8_t * payload, size_t length);
send URL as payload on WStype_CONNECTED event
move Sec-WebSocket-Accept generation in function
2015-05-23 09:02:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void WebSocketsServer::broadcastTXT(const char * payload, size_t length) {
|
|
|
|
broadcastTXT((uint8_t *) payload, length);
|
|
|
|
}
|
|
|
|
|
2015-05-22 23:02:47 +02:00
|
|
|
void WebSocketsServer::broadcastTXT(String payload) {
|
2015-05-23 07:33:26 +02:00
|
|
|
broadcastTXT((uint8_t *) payload.c_str(), payload.length());
|
2015-05-22 23:02:47 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* send binary data to client
|
|
|
|
* @param num uint8_t client id
|
|
|
|
* @param payload uint8_t *
|
|
|
|
* @param length size_t
|
2015-10-11 10:11:32 +02:00
|
|
|
* @param headerToPayload bool (see sendFrame for more details)
|
2015-05-22 23:02:47 +02:00
|
|
|
*/
|
2015-10-11 10:11:32 +02:00
|
|
|
void WebSocketsServer::sendBIN(uint8_t num, uint8_t * payload, size_t length, bool headerToPayload) {
|
2015-05-22 23:02:47 +02:00
|
|
|
if(num >= WEBSOCKETS_SERVER_CLIENT_MAX) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
WSclient_t * client = &_clients[num];
|
|
|
|
if(clientIsConnected(client)) {
|
2015-10-11 10:11:32 +02:00
|
|
|
sendFrame(client, WSop_binary, payload, length, false, true, headerToPayload);
|
2015-05-22 23:02:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
create overloaded functions for send and broadcast for easy usage
void sendTXT(uint8_t num, uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, const uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, char * payload, size_t length = 0);
void sendTXT(uint8_t num, const char * payload, size_t length = 0);
void sendTXT(uint8_t num, String payload);
void broadcastTXT(uint8_t * payload, size_t length = 0);
void broadcastTXT(const uint8_t * payload, size_t length = 0);
void broadcastTXT(char * payload, size_t length = 0);
void broadcastTXT(const char * payload, size_t length = 0);
void broadcastTXT(String payload);
void sendBIN(uint8_t num, uint8_t * payload, size_t length);
void sendBIN(uint8_t num, const uint8_t * payload, size_t length);
void broadcastBIN(uint8_t * payload, size_t length);
void broadcastBIN(const uint8_t * payload, size_t length);
send URL as payload on WStype_CONNECTED event
move Sec-WebSocket-Accept generation in function
2015-05-23 09:02:59 +02:00
|
|
|
void WebSocketsServer::sendBIN(uint8_t num, const uint8_t * payload, size_t length) {
|
|
|
|
sendBIN(num, (uint8_t *) payload, length);
|
|
|
|
}
|
|
|
|
|
2015-05-22 23:02:47 +02:00
|
|
|
/**
|
|
|
|
* send binary data to client all
|
|
|
|
* @param payload uint8_t *
|
|
|
|
* @param length size_t
|
2015-10-11 10:11:32 +02:00
|
|
|
* @param headerToPayload bool (see sendFrame for more details)
|
2015-05-22 23:02:47 +02:00
|
|
|
*/
|
2015-10-11 10:11:32 +02:00
|
|
|
void WebSocketsServer::broadcastBIN(uint8_t * payload, size_t length, bool headerToPayload) {
|
2015-05-22 23:02:47 +02:00
|
|
|
WSclient_t * client;
|
|
|
|
for(uint8_t i = 0; i < WEBSOCKETS_SERVER_CLIENT_MAX; i++) {
|
|
|
|
client = &_clients[i];
|
|
|
|
if(clientIsConnected(client)) {
|
2015-10-11 10:11:32 +02:00
|
|
|
sendFrame(client, WSop_binary, payload, length, false, true, headerToPayload);
|
2015-05-22 23:02:47 +02:00
|
|
|
}
|
2015-05-24 15:40:47 +02:00
|
|
|
#ifdef ESP8266
|
2015-05-23 23:51:32 +02:00
|
|
|
delay(0);
|
2015-05-24 15:40:47 +02:00
|
|
|
#endif
|
2015-05-22 23:02:47 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-23 09:20:44 +02:00
|
|
|
void WebSocketsServer::broadcastBIN(const uint8_t * payload, size_t length) {
|
create overloaded functions for send and broadcast for easy usage
void sendTXT(uint8_t num, uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, const uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, char * payload, size_t length = 0);
void sendTXT(uint8_t num, const char * payload, size_t length = 0);
void sendTXT(uint8_t num, String payload);
void broadcastTXT(uint8_t * payload, size_t length = 0);
void broadcastTXT(const uint8_t * payload, size_t length = 0);
void broadcastTXT(char * payload, size_t length = 0);
void broadcastTXT(const char * payload, size_t length = 0);
void broadcastTXT(String payload);
void sendBIN(uint8_t num, uint8_t * payload, size_t length);
void sendBIN(uint8_t num, const uint8_t * payload, size_t length);
void broadcastBIN(uint8_t * payload, size_t length);
void broadcastBIN(const uint8_t * payload, size_t length);
send URL as payload on WStype_CONNECTED event
move Sec-WebSocket-Accept generation in function
2015-05-23 09:02:59 +02:00
|
|
|
broadcastBIN((uint8_t *) payload, length);
|
|
|
|
}
|
|
|
|
|
2015-05-23 09:20:44 +02:00
|
|
|
/**
|
|
|
|
* disconnect all clients
|
|
|
|
*/
|
|
|
|
void WebSocketsServer::disconnect(void) {
|
|
|
|
WSclient_t * client;
|
|
|
|
for(uint8_t i = 0; i < WEBSOCKETS_SERVER_CLIENT_MAX; i++) {
|
|
|
|
client = &_clients[i];
|
|
|
|
if(clientIsConnected(client)) {
|
|
|
|
WebSockets::clientDisconnect(client, 1000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* disconnect one client
|
2015-05-23 09:47:39 +02:00
|
|
|
* @param num uint8_t client id
|
2015-05-23 09:20:44 +02:00
|
|
|
*/
|
|
|
|
void WebSocketsServer::disconnect(uint8_t num) {
|
|
|
|
if(num >= WEBSOCKETS_SERVER_CLIENT_MAX) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
WSclient_t * client = &_clients[num];
|
|
|
|
if(clientIsConnected(client)) {
|
|
|
|
WebSockets::clientDisconnect(client, 1000);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-23 09:47:39 +02:00
|
|
|
/**
|
|
|
|
* get an IP for a client
|
|
|
|
* @param num uint8_t client id
|
|
|
|
* @return IPAddress
|
|
|
|
*/
|
|
|
|
IPAddress WebSocketsServer::remoteIP(uint8_t num) {
|
|
|
|
if(num < WEBSOCKETS_SERVER_CLIENT_MAX) {
|
|
|
|
WSclient_t * client = &_clients[num];
|
|
|
|
if(clientIsConnected(client)) {
|
|
|
|
return client->tcp.remoteIP();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return IPAddress();
|
|
|
|
}
|
|
|
|
|
2015-05-22 14:19:01 +02:00
|
|
|
//#################################################################################
|
|
|
|
//#################################################################################
|
|
|
|
//#################################################################################
|
|
|
|
|
2015-05-22 23:02:47 +02:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @param client WSclient_t * ptr to the client struct
|
|
|
|
* @param opcode WSopcode_t
|
|
|
|
* @param payload uint8_t *
|
|
|
|
* @param lenght size_t
|
|
|
|
*/
|
|
|
|
void WebSocketsServer::messageRecived(WSclient_t * client, WSopcode_t opcode, uint8_t * payload, size_t lenght) {
|
|
|
|
WStype_t type = WStype_ERROR;
|
|
|
|
|
|
|
|
switch(opcode) {
|
|
|
|
case WSop_text:
|
|
|
|
type = WStype_TEXT;
|
|
|
|
break;
|
|
|
|
case WSop_binary:
|
|
|
|
type = WStype_BIN;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(_cbEvent) {
|
|
|
|
_cbEvent(client->num, type, payload, lenght);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-05-22 14:19:01 +02:00
|
|
|
/**
|
|
|
|
* Disconnect an client
|
2015-05-22 23:02:47 +02:00
|
|
|
* @param client WSclient_t * ptr to the client struct
|
2015-05-22 14:19:01 +02:00
|
|
|
*/
|
2015-05-22 20:35:51 +02:00
|
|
|
void WebSocketsServer::clientDisconnect(WSclient_t * client) {
|
2015-05-22 14:19:01 +02:00
|
|
|
|
|
|
|
if(client->tcp) {
|
2015-05-23 23:51:32 +02:00
|
|
|
client->tcp.flush();
|
2015-05-22 14:19:01 +02:00
|
|
|
client->tcp.stop();
|
|
|
|
}
|
2015-05-22 14:40:46 +02:00
|
|
|
|
|
|
|
client->cUrl = "";
|
|
|
|
client->cKey = "";
|
|
|
|
client->cProtocol = "";
|
|
|
|
client->cVersion = 0;
|
|
|
|
client->cIsUpgrade = false;
|
|
|
|
client->cIsWebsocket = false;
|
|
|
|
|
2015-05-22 14:19:01 +02:00
|
|
|
client->status = WSC_NOT_CONNECTED;
|
|
|
|
|
2015-05-22 14:40:46 +02:00
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d] client disconnected.\n", client->num);
|
2015-05-22 14:19:01 +02:00
|
|
|
|
2015-05-22 23:02:47 +02:00
|
|
|
if(_cbEvent) {
|
|
|
|
_cbEvent(client->num, WStype_DISCONNECTED, NULL, 0);
|
|
|
|
}
|
2015-05-22 14:19:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get client state
|
2015-05-22 23:02:47 +02:00
|
|
|
* @param client WSclient_t * ptr to the client struct
|
2015-05-22 14:19:01 +02:00
|
|
|
* @return true = conneted
|
|
|
|
*/
|
2015-05-22 20:35:51 +02:00
|
|
|
bool WebSocketsServer::clientIsConnected(WSclient_t * client) {
|
2015-05-22 14:19:01 +02:00
|
|
|
|
|
|
|
if(client->status != WSC_NOT_CONNECTED && client->tcp.connected()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(client->status != WSC_NOT_CONNECTED) {
|
|
|
|
// cleanup
|
2015-05-22 14:40:46 +02:00
|
|
|
clientDisconnect(client);
|
2015-05-22 14:19:01 +02:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle incomming Connection Request
|
|
|
|
*/
|
|
|
|
void WebSocketsServer::handleNewClients(void) {
|
2015-05-22 20:35:51 +02:00
|
|
|
WSclient_t * client;
|
2015-05-22 14:19:01 +02:00
|
|
|
while(_server->hasClient()) {
|
|
|
|
bool ok = false;
|
|
|
|
// search free list entry for client
|
|
|
|
for(uint8_t i = 0; i < WEBSOCKETS_SERVER_CLIENT_MAX; i++) {
|
|
|
|
client = &_clients[i];
|
|
|
|
|
|
|
|
// state is not connected or tcp connection is lost
|
2015-05-22 14:40:46 +02:00
|
|
|
if(!clientIsConnected(client)) {
|
2015-05-22 14:19:01 +02:00
|
|
|
|
|
|
|
// store new connection
|
|
|
|
client->tcp = _server->available();
|
2015-05-24 15:40:47 +02:00
|
|
|
#ifdef ESP8266
|
2015-05-22 14:40:46 +02:00
|
|
|
client->tcp.setNoDelay(true);
|
2015-05-24 15:40:47 +02:00
|
|
|
#endif
|
2015-05-22 14:19:01 +02:00
|
|
|
// set Timeout for readBytesUntil and readStringUntil
|
2015-05-22 20:35:51 +02:00
|
|
|
client->tcp.setTimeout(WEBSOCKETS_TCP_TIMEOUT);
|
2015-05-22 14:19:01 +02:00
|
|
|
client->status = WSC_HEADER;
|
|
|
|
|
|
|
|
IPAddress ip = client->tcp.remoteIP();
|
2015-05-22 14:40:46 +02:00
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d] new client from %d.%d.%d.%d\n", client->num, ip[0], ip[1], ip[2], ip[3]);
|
2015-05-22 14:19:01 +02:00
|
|
|
ok = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(!ok) {
|
|
|
|
// no free space to handle client
|
|
|
|
WiFiClient tcpClient = _server->available();
|
|
|
|
IPAddress ip = tcpClient.remoteIP();
|
|
|
|
DEBUG_WEBSOCKETS("[WS-Server] no free space new client from %d.%d.%d.%d\n", ip[0], ip[1], ip[2], ip[3]);
|
|
|
|
tcpClient.stop();
|
|
|
|
}
|
|
|
|
|
2015-05-24 15:40:47 +02:00
|
|
|
#ifdef ESP8266
|
2015-05-22 14:19:01 +02:00
|
|
|
delay(0);
|
2015-05-24 15:40:47 +02:00
|
|
|
#endif
|
2015-05-22 14:19:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handel incomming data from Client
|
|
|
|
*/
|
|
|
|
void WebSocketsServer::handleClientData(void) {
|
|
|
|
|
2015-05-22 20:35:51 +02:00
|
|
|
WSclient_t * client;
|
2015-05-22 14:19:01 +02:00
|
|
|
for(uint8_t i = 0; i < WEBSOCKETS_SERVER_CLIENT_MAX; i++) {
|
2015-05-22 14:40:46 +02:00
|
|
|
client = &_clients[i];
|
|
|
|
if(clientIsConnected(client)) {
|
2015-05-22 18:44:16 +02:00
|
|
|
int len = client->tcp.available();
|
2015-05-22 14:19:01 +02:00
|
|
|
if(len > 0) {
|
|
|
|
|
|
|
|
switch(client->status) {
|
|
|
|
case WSC_HEADER:
|
2015-05-22 14:40:46 +02:00
|
|
|
handleHeader(client);
|
2015-05-22 14:19:01 +02:00
|
|
|
break;
|
|
|
|
case WSC_CONNECTED:
|
2015-05-22 20:35:51 +02:00
|
|
|
WebSockets::handleWebsocket(client);
|
2015-05-22 14:19:01 +02:00
|
|
|
break;
|
|
|
|
default:
|
create overloaded functions for send and broadcast for easy usage
void sendTXT(uint8_t num, uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, const uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, char * payload, size_t length = 0);
void sendTXT(uint8_t num, const char * payload, size_t length = 0);
void sendTXT(uint8_t num, String payload);
void broadcastTXT(uint8_t * payload, size_t length = 0);
void broadcastTXT(const uint8_t * payload, size_t length = 0);
void broadcastTXT(char * payload, size_t length = 0);
void broadcastTXT(const char * payload, size_t length = 0);
void broadcastTXT(String payload);
void sendBIN(uint8_t num, uint8_t * payload, size_t length);
void sendBIN(uint8_t num, const uint8_t * payload, size_t length);
void broadcastBIN(uint8_t * payload, size_t length);
void broadcastBIN(const uint8_t * payload, size_t length);
send URL as payload on WStype_CONNECTED event
move Sec-WebSocket-Accept generation in function
2015-05-23 09:02:59 +02:00
|
|
|
WebSockets::clientDisconnect(client, 1002);
|
2015-05-22 14:19:01 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-05-24 15:40:47 +02:00
|
|
|
#ifdef ESP8266
|
2015-05-22 14:19:01 +02:00
|
|
|
delay(0);
|
2015-05-24 15:40:47 +02:00
|
|
|
#endif
|
2015-05-22 14:19:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2015-05-22 20:35:51 +02:00
|
|
|
* handle the WebSocket header reading
|
2015-05-22 23:02:47 +02:00
|
|
|
* @param client WSclient_t * ptr to the client struct
|
2015-05-22 14:19:01 +02:00
|
|
|
*/
|
2015-05-22 20:35:51 +02:00
|
|
|
void WebSocketsServer::handleHeader(WSclient_t * client) {
|
2015-05-22 14:19:01 +02:00
|
|
|
|
|
|
|
String headerLine = client->tcp.readStringUntil('\n');
|
|
|
|
headerLine.trim(); // remove \r
|
|
|
|
|
|
|
|
if(headerLine.length() > 0) {
|
2015-05-22 14:40:46 +02:00
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d][handleHeader] RX: %s\n", client->num, headerLine.c_str());
|
2015-05-22 14:19:01 +02:00
|
|
|
|
2015-05-23 07:33:26 +02:00
|
|
|
// websocket request starts allways with GET see rfc6455
|
2015-05-22 14:19:01 +02:00
|
|
|
if(headerLine.startsWith("GET ")) {
|
|
|
|
// cut URL out
|
|
|
|
client->cUrl = headerLine.substring(4, headerLine.indexOf(' ', 4));
|
2015-07-26 11:08:02 +02:00
|
|
|
} else if(headerLine.indexOf(':')) {
|
|
|
|
String headerName = headerLine.substring(0, headerLine.indexOf(':'));
|
|
|
|
String headerValue = headerLine.substring(headerLine.indexOf(':') + 2);
|
|
|
|
|
|
|
|
if(headerName.equalsIgnoreCase("Connection")) {
|
|
|
|
if(headerValue.indexOf("Upgrade") >= 0) {
|
|
|
|
client->cIsUpgrade = true;
|
|
|
|
}
|
|
|
|
} else if(headerName.equalsIgnoreCase("Upgrade")) {
|
|
|
|
if(headerValue.equalsIgnoreCase("websocket")) {
|
|
|
|
client->cIsWebsocket = true;
|
|
|
|
}
|
|
|
|
} else if(headerName.equalsIgnoreCase("Sec-WebSocket-Version")) {
|
|
|
|
client->cVersion = headerValue.toInt();
|
|
|
|
} else if(headerName.equalsIgnoreCase("Sec-WebSocket-Key")) {
|
|
|
|
client->cKey = headerValue;
|
|
|
|
client->cKey.trim(); // see rfc6455
|
|
|
|
} else if(headerName.equalsIgnoreCase("Sec-WebSocket-Protocol")) {
|
|
|
|
client->cProtocol = headerValue;
|
|
|
|
} else if(headerName.equalsIgnoreCase("Sec-WebSocket-Extensions")) {
|
|
|
|
client->cExtensions = headerValue;
|
2015-06-17 10:37:20 +02:00
|
|
|
}
|
2015-07-26 11:08:02 +02:00
|
|
|
} else {
|
|
|
|
DEBUG_WEBSOCKETS("[WS-Client][handleHeader] Header error (%s)\n", headerLine.c_str());
|
2015-05-22 14:19:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
2015-05-22 14:40:46 +02:00
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d][handleHeader] Header read fin.\n", client->num);
|
2015-05-22 14:19:01 +02:00
|
|
|
|
2015-05-23 07:33:26 +02:00
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d][handleHeader] - cURL: %s\n", client->num, client->cUrl.c_str());
|
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d][handleHeader] - cIsUpgrade: %d\n", client->num, client->cIsUpgrade);
|
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d][handleHeader] - cIsWebsocket: %d\n", client->num, client->cIsWebsocket);
|
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d][handleHeader] - cKey: %s\n", client->num, client->cKey.c_str());
|
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d][handleHeader] - cProtocol: %s\n", client->num, client->cProtocol.c_str());
|
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d][handleHeader] - cExtensions: %s\n", client->num, client->cExtensions.c_str());
|
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d][handleHeader] - cVersion: %d\n", client->num, client->cVersion);
|
2015-05-22 14:19:01 +02:00
|
|
|
|
|
|
|
bool ok = (client->cIsUpgrade && client->cIsWebsocket);
|
|
|
|
|
|
|
|
if(ok) {
|
|
|
|
if(client->cUrl.length() == 0) {
|
|
|
|
ok = false;
|
|
|
|
}
|
|
|
|
if(client->cKey.length() == 0) {
|
|
|
|
ok = false;
|
|
|
|
}
|
2015-05-22 14:40:46 +02:00
|
|
|
if(client->cVersion != 13) {
|
2015-05-22 14:19:01 +02:00
|
|
|
ok = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(ok) {
|
|
|
|
|
2015-05-22 14:40:46 +02:00
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d][handleHeader] Websocket connection incomming.\n", client->num);
|
2015-05-22 14:19:01 +02:00
|
|
|
|
2015-05-22 18:44:16 +02:00
|
|
|
// generate Sec-WebSocket-Accept key
|
2015-05-23 09:34:09 +02:00
|
|
|
String sKey = acceptKey(client->cKey);
|
2015-05-22 18:44:16 +02:00
|
|
|
|
2015-05-23 09:34:09 +02:00
|
|
|
DEBUG_WEBSOCKETS("[WS-Server][%d][handleHeader] - sKey: %s\n", client->num, sKey.c_str());
|
2015-05-22 18:44:16 +02:00
|
|
|
|
|
|
|
client->status = WSC_CONNECTED;
|
|
|
|
|
|
|
|
client->tcp.write("HTTP/1.1 101 Switching Protocols\r\n"
|
2015-05-24 18:00:35 +02:00
|
|
|
"Server: arduino-WebSocketsServer\r\n"
|
2015-05-22 18:44:16 +02:00
|
|
|
"Upgrade: websocket\r\n"
|
|
|
|
"Connection: Upgrade\r\n"
|
|
|
|
"Sec-WebSocket-Version: 13\r\n"
|
|
|
|
"Sec-WebSocket-Accept: ");
|
2015-05-23 09:34:09 +02:00
|
|
|
client->tcp.write(sKey.c_str(), sKey.length());
|
2015-05-22 20:35:51 +02:00
|
|
|
client->tcp.write("\r\n");
|
|
|
|
|
|
|
|
if(client->cProtocol.length() > 0) {
|
|
|
|
// todo add api to set Protocol of Server
|
2015-05-24 15:40:47 +02:00
|
|
|
client->tcp.write("Sec-WebSocket-Protocol: arduino\r\n");
|
2015-05-22 20:35:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// header end
|
|
|
|
client->tcp.write("\r\n");
|
2015-05-22 14:19:01 +02:00
|
|
|
|
create overloaded functions for send and broadcast for easy usage
void sendTXT(uint8_t num, uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, const uint8_t * payload, size_t length = 0);
void sendTXT(uint8_t num, char * payload, size_t length = 0);
void sendTXT(uint8_t num, const char * payload, size_t length = 0);
void sendTXT(uint8_t num, String payload);
void broadcastTXT(uint8_t * payload, size_t length = 0);
void broadcastTXT(const uint8_t * payload, size_t length = 0);
void broadcastTXT(char * payload, size_t length = 0);
void broadcastTXT(const char * payload, size_t length = 0);
void broadcastTXT(String payload);
void sendBIN(uint8_t num, uint8_t * payload, size_t length);
void sendBIN(uint8_t num, const uint8_t * payload, size_t length);
void broadcastBIN(uint8_t * payload, size_t length);
void broadcastBIN(const uint8_t * payload, size_t length);
send URL as payload on WStype_CONNECTED event
move Sec-WebSocket-Accept generation in function
2015-05-23 09:02:59 +02:00
|
|
|
// send ping
|
|
|
|
WebSockets::sendFrame(client, WSop_ping);
|
|
|
|
|
2015-05-22 23:02:47 +02:00
|
|
|
if(_cbEvent) {
|
2015-05-23 09:20:44 +02:00
|
|
|
_cbEvent(client->num, WStype_CONNECTED, (uint8_t *) client->cUrl.c_str(), client->cUrl.length());
|
2015-05-22 23:02:47 +02:00
|
|
|
}
|
|
|
|
|
2015-05-22 14:19:01 +02:00
|
|
|
} else {
|
2015-08-08 19:54:36 +02:00
|
|
|
handleNonWebsocketConnection(client);
|
2015-05-22 14:19:01 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2015-05-22 20:35:51 +02:00
|
|
|
|
2015-08-08 19:54:36 +02:00
|
|
|
|
|
|
|
|