Skip to content
Snippets Groups Projects
netserver.h 4.06 KiB
Newer Older
  • Learn to ignore specific revisions
  •  * 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"
    
    #include "libwebsockets.h"
    #include "opc.h"
    
        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;
    
        // 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;
    
            // OPC and protocol-detection receive buffer.
            OPCBuffer *opcBuffer;
    
        OPC::callback_t mOpcCallback;
        jsonCallback_t mJsonCallback;
    
        std::set<libwebsocket*> mClients;
    
        typedef rapidjson::GenericStringBuffer<rapidjson::UTF8<>> jsonBuffer_t;
        std::vector<jsonBuffer_t*> mBroadcastList;
        tthread::mutex mBroadcastMutex;
    
    
        static HTTPDocument httpDocumentList[];
    
    
        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();