Newer
Older
* Open Pixel Control and HTTP server for Fadecandy
*
* Copyright (c) 2013 Micah Elizabeth Scott
*
* Permission is hereby granted, free of charge, to any person obtaining a copy of
* this software and associated documentation files (the "Software"), to deal in
* the Software without restriction, including without limitation the rights to
* use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
* the Software, and to permit persons to whom the Software is furnished to do so,
* subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
* FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
* IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
#pragma once
#include <vector>
#include <set>
#include "rapidjson/document.h"
#include "rapidjson/stringbuffer.h"
Micah Elizabeth Scott
committed
#include "tinythread.h"
#include "libwebsockets.h"
#include "opc.h"
class NetServer {
typedef void (*jsonCallback_t)(libwebsocket *wsi, rapidjson::Document &message, void *context);
NetServer(OPC::callback_t opcCallback, jsonCallback_t jsonCallback,
void *context, bool verbose = false);
// Start the event loop on a separate thread
bool start(const char *host, int port);
// Reply callback, for use only on the NetServer thread. Call this inside jsonCallback.
int jsonReply(libwebsocket *wsi, rapidjson::Document &message);
// Broadcast JSON to all clients, from any thread.
void jsonBroadcast(rapidjson::Document &message);
private:
enum ClientState {
CLIENT_STATE_PROTOCOL_DETECT = 0,
CLIENT_STATE_OPEN_PIXEL_CONTROL,
CLIENT_STATE_HTTP
// In-memory database of static files to serve over HTTP
struct HTTPDocument {
const char *path;
const char *body;
const char *contentType;
int contentLength;
Micah Elizabeth Scott
committed
// Buffer used for protocol-detection and Open Pixel Control. Big enough for two OPC packets.
// This buffer is jettisonned
struct OPCBuffer {
unsigned bufferLength;
uint8_t buffer[2 * sizeof(OPC::Message)];
};
struct Client {
ClientState state;
// HTTP response state
const char *httpBody;
int httpLength;
// OPC and protocol-detection receive buffer.
OPCBuffer *opcBuffer;
Micah Elizabeth Scott
committed
};
OPC::callback_t mOpcCallback;
jsonCallback_t mJsonCallback;
void *mUserContext;
Micah Elizabeth Scott
committed
tthread::thread *mThread;
std::set<libwebsocket*> mClients;
typedef rapidjson::GenericStringBuffer<rapidjson::UTF8<>> jsonBuffer_t;
std::vector<jsonBuffer_t*> mBroadcastList;
tthread::mutex mBroadcastMutex;
static HTTPDocument httpDocumentList[];
// libwebsockets server
static void threadFunc(void *arg);
static int lwsCallback(libwebsocket_context *context, libwebsocket *wsi,
enum libwebsocket_callback_reasons reason, void *user, void *in, size_t len);
// HTTP Server
int httpBegin(libwebsocket_context *context, libwebsocket *wsi, Client &client, const char *path);
int httpWrite(libwebsocket_context *context, libwebsocket *wsi, Client &client);
static bool httpPathEqual(const char *a, const char *b);
// Open Pixel Control server
int opcRead(libwebsocket_context *context, libwebsocket *wsi, Client &client, uint8_t *in, size_t len);
// WebSockets server
int wsRead(libwebsocket_context *context, libwebsocket *wsi, Client &client, uint8_t *in, size_t len);
void jsonBufferPrepare(jsonBuffer_t &buffer, rapidjson::Value &value);
int jsonBufferSend(jsonBuffer_t &buffer, libwebsocket *wsi);
void flushBroadcastList();