#pragma once #ifndef _NET_H #define _NET_H #ifdef _NET_SMALL_BUF #define BUFSIZE 512 #define BUF_2_MAX 2048 #else #define BUFSIZE 16384 #define BUF_2_MAX 16384 #endif #define WIN32_LEAN_AND_MEAN // Ping flag tells the recieving host to drop the current ulong_64b, as it is sent to check if the connection is still alive #define FLAG_PING (ulong_64b)-1 #include "Crypto.h" #include "ArchAbstract.h" #include #include #include #include namespace IO { enum CryptoLevel { None, Prefer, Force }; struct Packet { ulong_64b size; char packetUID; char* message; }; class NetServer; class NetClient { friend class NetServer; // Allow NetServer to access all members of NetClient private: volatile bool _open; // Whether or not connection is open bool canWrite; // Whether or not writing to peer is possible bool noThread; // Whether or not reading incoming data should be / is being done in a separate thread char rBuf[BUFSIZE]; // Recieve buffer CryptoLevel preferEncrypted = CryptoLevel::None;// Whether or not the socket should attempt to request an encrypted channel bool encrypted = false; // Whether or not negotiation determined the use of an encrypted channel bool firstMessage = true; // Whether or not negotiation has yet ocurred ulong_64b fm_neg_size; bool fm_neg_hasLevel = false; bool fm_neg_hasSize = false; bool startNegotiate = false; char expectedNextPUID = 0; char remotePUID = 0; std::vector* sparse; std::vector* outPacketBuf; Crypto::RSA::KeyData keys; // Client's keysets (if using encryption) CryptoPP::RSAFunction pK; // Remote host's public key (if using encryption) NetClient(SOCKET, bool, CryptoLevel, bool);// Special setup constructor NetClient(SOCKET, bool, Crypto::RSA::KeyData&, CryptoLevel = CryptoLevel::None, bool = false);// Create wrapper for existing socket void sharedSetup(); // Setup function for all constructor bool _write(char*, ulong_64b); // Internal write function. Doesn't do any of the fancy auto encryption: just raw write... bool writeBufferedPackets(); // Flushes and deletes buffer void update(); // Read incoming data and store in buffers bool ping(); // Check if connection is alive by pinging remote host protected: std::thread listener; // Incoming data listener (optional) SOCKET _socket; // Underlying socket used for communication std::vector* packets; // Basically a set containing a backlog of unprocessed data. Will oly be used if event handler doesn't exist std::function evt; // New data event handler std::function onDestroy; // Event handler called when NetClient object is destroyed public: time_t commTime; // Latest time a transaction occurred std::vector associatedData; NetClient(char* ipAddr, char* port, CryptoLevel = CryptoLevel::None);// Standard constructor for creating connection ~NetClient(); bool close(); void closeWrite(); bool isEncrypted(); size_t getBOPCount(); // Should return the amount of buffered packets to be sent to server bool write(void* message, ulong_64b size); bool write(char* message); Packet read(); void setEventHandler(std::function); // Register a callback that is guaranteed to be called when the socket has at least one unprocessed packet void setOnDestroy(std::function); bool isOpen(); ulong_64b available(); }; class NetServer { friend class NetClient; private: CryptoLevel pref; Crypto::RSA::KeyData keys; // Server's keysets (if using encryption) std::function onDestroy; volatile bool _open; void updateClients(); protected: std::thread clientListener; std::vector>* handlers; std::vector* clients; public: std::function timeoutHandler; NetServer(char* port, std::function, CryptoLevel); ~NetServer(); bool isOpen(); CryptoLevel getCryptoPreference(); void addHandler(std::function); void clearHandlers(); void setOnDestroy(std::function); bool close(); }; } #endif