Update
NetworkMessage->GetData fixed Plugin -> Splitted message types PluginManager is now thread safe TcpConnection added list of events Properly handling disconnection now Fixed bug in Socket with error translation It crashes when it exits, i think it has something to do with shared ptrs or something
This commit is contained in:
		| @ -89,7 +89,9 @@ namespace std::net | ||||
| 		template<typename T> | ||||
| 		T *GetData() const | ||||
| 		{ | ||||
| 			if (m_data) | ||||
| 				return (T*)m_data; | ||||
| 			else return nullptr; | ||||
| 		} | ||||
| 	}; | ||||
| } | ||||
| @ -1,6 +1,8 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include <VoidNet_HL/NetworkMessage.hpp> | ||||
| #include <VoidNet_HL/InternalTags.hpp> | ||||
| #include <string> | ||||
|  | ||||
| namespace std::net | ||||
| { | ||||
| @ -11,6 +13,29 @@ namespace std::net | ||||
| 		{ | ||||
| 		} | ||||
|  | ||||
| 		virtual void HandleMessage(const NetworkMessage& msg) = 0; | ||||
| 		void HandleMessage(const NetworkMessage& msg) | ||||
| 		{ | ||||
| 			if (msg.GetTag() == (uint32_t)InternalTags::Disconnect) | ||||
| 			{ | ||||
| 				OnDisconnect(*(msg.GetData<string>())); | ||||
| 			} | ||||
| 			else if (msg.GetTag() == (uint32_t)InternalTags::Connect) | ||||
| 			{ | ||||
| 				OnNewConnection(msg.GetSenderID(), msg.GetData<void>()); | ||||
| 			} | ||||
| 			else if (msg.GetTag() == (uint32_t)InternalTags::AssignID) | ||||
| 			{ | ||||
| 				OnConnection(); | ||||
| 			} | ||||
| 			else | ||||
| 			{ | ||||
| 				OnDataReceived(msg.GetSenderID(), msg.GetDistributionMode(), msg.GetDestinationID(), msg.GetTag(), msg.GetData<void>()); | ||||
| 			} | ||||
| 		} | ||||
|  | ||||
| 		virtual void OnDisconnect(string) abstract; | ||||
| 		virtual void OnNewConnection(uint32_t, void*) abstract; | ||||
| 		virtual void OnConnection() abstract; | ||||
| 		virtual void OnDataReceived(uint32_t, DistributionMode, uint32_t, uint32_t, void*) abstract; | ||||
| 	}; | ||||
| } | ||||
| @ -1,6 +1,7 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include <vector> | ||||
| #include <mutex> | ||||
|  | ||||
| #include "VoidNet_HL/Plugin/Plugin.hpp" | ||||
|  | ||||
| @ -14,16 +15,24 @@ namespace std::net | ||||
|  | ||||
| 		void HandleMessage(const NetworkMessage& msg) | ||||
| 		{ | ||||
| 			m_pluginsMutex.lock(); | ||||
|  | ||||
| 			for (size_t i = 0; i < m_plugins.size(); i++) | ||||
| 				m_plugins.at(i)->HandleMessage(msg); | ||||
|  | ||||
| 			m_pluginsMutex.unlock(); | ||||
| 		} | ||||
|  | ||||
| 		void AddPlugin(Plugin* p) | ||||
| 		{ | ||||
| 			m_pluginsMutex.lock(); | ||||
| 			m_plugins.emplace_back(p); | ||||
| 			m_pluginsMutex.unlock(); | ||||
| 		} | ||||
|  | ||||
| 	private: | ||||
| 		vector<Plugin*> m_plugins; | ||||
|  | ||||
| 		mutex m_pluginsMutex; | ||||
| 	}; | ||||
| } | ||||
| @ -21,30 +21,55 @@ namespace std::net | ||||
|  | ||||
| 		bool Connect(IPAddress addr); | ||||
|  | ||||
| 		bool Disconnect(); | ||||
|  | ||||
| 		template<typename T> | ||||
| 		void SendMessage(DistributionMode mode, uint32_t destinationId, uint32_t tag, void *data) | ||||
| 		bool SendMessage(DistributionMode mode, uint32_t destinationId, uint32_t tag, void *data) | ||||
| 		{ | ||||
| 			NetworkMessage msg(m_id, mode, destinationId, tag, data, sizeof(T)); | ||||
| 			sendMessage(msg); | ||||
| 			return sendMessage(msg); | ||||
| 		} | ||||
|  | ||||
| 		void SendMessage(DistributionMode mode, uint32_t destinationId, uint32_t tag) | ||||
| 		bool SendMessage(DistributionMode mode, uint32_t destinationId, uint32_t tag) | ||||
| 		{ | ||||
| 			NetworkMessage msg(m_id, mode, destinationId, tag, nullptr, 0); | ||||
| 			sendMessage(msg); | ||||
| 			return sendMessage(msg); | ||||
| 		} | ||||
|  | ||||
| 		void ReceiveData(); | ||||
|  | ||||
| 		function<void(uint32_t, DistributionMode, uint32_t, uint32_t, void*)> DataReceivedEvent; | ||||
| 		function<void(string)> DisconnectedEvent; | ||||
| 		function<void(uint32_t, void*)> NewConnectionEvent; | ||||
| 		function<void()> OnConnectionEvent; | ||||
| 		bool IsConnected; | ||||
|  | ||||
| 	private: | ||||
| 		bool sendMessage(const NetworkMessage &msg); | ||||
|  | ||||
| 		shared_ptr<TcpClient> m_client; | ||||
| 		uint32_t m_id; | ||||
|  | ||||
| 		vector<function<void(uint32_t, DistributionMode, uint32_t, uint32_t, void*)>> m_onDataReceived; | ||||
| 		vector<function<void(string)>> m_onDisconnect; | ||||
| 		vector<function<void(uint32_t, void*)>> m_onNewConnection; | ||||
| 		vector<function<void()>> m_onConnection; | ||||
|  | ||||
| 	public: | ||||
| 		void operator+=(const function<void(uint32_t, DistributionMode, uint32_t, uint32_t, void*)> &rhs) | ||||
| 		{ | ||||
| 			m_onDataReceived.push_back(rhs); | ||||
| 		} | ||||
|  | ||||
| 		void operator+=(const function<void(string)>& rhs) | ||||
| 		{ | ||||
| 			m_onDisconnect.push_back(rhs); | ||||
| 		} | ||||
|  | ||||
| 		void operator+=(const function<void(uint32_t, void*)> & rhs) | ||||
| 		{ | ||||
| 			m_onNewConnection.push_back(rhs); | ||||
| 		} | ||||
|  | ||||
| 		void operator+=(const function<void()>& rhs) | ||||
| 		{ | ||||
| 			m_onConnection.push_back(rhs); | ||||
| 		} | ||||
| 	}; | ||||
| } | ||||
| @ -47,7 +47,6 @@ namespace std::net | ||||
| 		uint32_t m_maxConnections = 0; | ||||
|  | ||||
| 		thread m_receiveThread; | ||||
| 		thread m_sendThread; | ||||
|  | ||||
| 		atomic_bool m_run; | ||||
|  | ||||
|  | ||||
| @ -3,6 +3,7 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include <map> | ||||
| #include <string> | ||||
|  | ||||
| namespace std::net | ||||
| { | ||||
|  | ||||
| @ -81,8 +81,8 @@ namespace std::net | ||||
| 	enum class SocketType | ||||
| 	{ | ||||
| 		Unknown = -1, | ||||
| 		Datagram = 2, | ||||
| 		Streaming = 1, | ||||
| 		Datagram = 2, //SOCK_DGRAM | ||||
| 		Streaming = 1, //SOCK_STREAM | ||||
| 	}; | ||||
|  | ||||
| 	enum class SocketProtocol | ||||
|  | ||||
| @ -3,6 +3,7 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include <map> | ||||
| #include <string> | ||||
|  | ||||
| namespace std::net | ||||
| { | ||||
|  | ||||
| @ -21,7 +21,7 @@ namespace std::net | ||||
| 			init(); | ||||
| 		} | ||||
|  | ||||
| 		virtual ~Socket() { Close(); } | ||||
| 		virtual ~Socket() {  } | ||||
|  | ||||
| 	public: | ||||
|  | ||||
|  | ||||
| @ -10,5 +10,9 @@ namespace std::net | ||||
|  | ||||
| 	PluginManager::~PluginManager() | ||||
| 	{ | ||||
| 		for (size_t i = 0; i < m_plugins.size(); i++) | ||||
| 		{ | ||||
| 			delete m_plugins[i]; | ||||
| 		} | ||||
| 	} | ||||
| } | ||||
| @ -6,33 +6,9 @@ | ||||
|  | ||||
| namespace std::net | ||||
| { | ||||
| 	void received(uint32_t, DistributionMode, uint32_t, uint32_t, void*) | ||||
| 	{ | ||||
| 		std::cout << "received" << std::endl; | ||||
| 	} | ||||
|  | ||||
| 	void disconnected(std::string s) | ||||
| 	{ | ||||
| 		std::cout << s << std::endl; | ||||
| 	} | ||||
|  | ||||
| 	void new_connection(uint32_t, void*) | ||||
| 	{ | ||||
| 		std::cout << "new client connection" << std::endl; | ||||
| 	} | ||||
|  | ||||
| 	void on_connect() | ||||
| 	{ | ||||
| 		std::cout << "i connected" << std::endl; | ||||
| 	} | ||||
|  | ||||
| 	TcpConnection::TcpConnection() : | ||||
| 		m_client(new TcpClient()) | ||||
| 	{ | ||||
| 		DataReceivedEvent = received; | ||||
| 		DisconnectedEvent = disconnected; | ||||
| 		NewConnectionEvent = new_connection; | ||||
| 		OnConnectionEvent = on_connect; | ||||
| 	} | ||||
|  | ||||
| 	TcpConnection::TcpConnection(TcpClient * client) | ||||
| @ -52,7 +28,12 @@ namespace std::net | ||||
|  | ||||
| 	bool TcpConnection::Connect(IPAddress addr) | ||||
| 	{ | ||||
| 		return m_client->Connect(addr); | ||||
| 		return IsConnected = m_client->Connect(addr); | ||||
| 	} | ||||
|  | ||||
| 	bool TcpConnection::Disconnect() | ||||
| 	{ | ||||
| 		return SendMessage(DistributionMode::AllAndServer, 0, (uint32_t)InternalTags::Disconnect); | ||||
| 	} | ||||
|  | ||||
| 	bool TcpConnection::sendMessage(const NetworkMessage & msg) | ||||
| @ -79,25 +60,39 @@ namespace std::net | ||||
|  | ||||
| 			if (message.GetTag() == (uint32_t)InternalTags::Disconnect) | ||||
| 			{ | ||||
| 				if (DisconnectedEvent) | ||||
| 					DisconnectedEvent(*(message.GetData<string>())); | ||||
| 				for (size_t i = 0; i < m_onDisconnect.size(); i++) | ||||
| 				{ | ||||
| 					//string* msgStr = message.GetData<string>(); | ||||
| 					m_onDisconnect[i](""); | ||||
| 				} | ||||
|  | ||||
| 				IsConnected = false; | ||||
| 				m_client->Close(); | ||||
| 			} | ||||
| 			else if (message.GetTag() == (uint32_t)InternalTags::Connect) | ||||
| 			{ | ||||
| 				if (NewConnectionEvent) | ||||
| 					NewConnectionEvent(message.GetSenderID(), message.GetData<void>()); | ||||
| 				for (size_t i = 0; i < m_onNewConnection.size(); i++) | ||||
| 				{ | ||||
| 					m_onNewConnection[i](message.GetSenderID(), message.GetData<void>()); | ||||
| 				} | ||||
| 			} | ||||
| 			else if (message.GetTag() == (uint32_t)InternalTags::AssignID) | ||||
| 			{ | ||||
| 				m_id = *(message.GetData<uint32_t>()); | ||||
|  | ||||
| 				if (OnConnectionEvent) | ||||
| 					OnConnectionEvent(); | ||||
| 				for (size_t i = 0; i < m_onConnection.size(); i++) | ||||
| 				{ | ||||
| 					m_onConnection[i](); | ||||
| 				} | ||||
|  | ||||
| 				IsConnected = true; | ||||
| 			} | ||||
| 			else | ||||
| 			{ | ||||
| 				if (DataReceivedEvent) | ||||
| 					DataReceivedEvent(message.GetSenderID(), message.GetDistributionMode(), message.GetDestinationID(), message.GetTag(), message.GetData<void>()); | ||||
| 				for (size_t i = 0; i < m_onDataReceived.size(); i++) | ||||
| 				{ | ||||
| 					m_onDataReceived[i](message.GetSenderID(), message.GetDistributionMode(), message.GetDestinationID(), message.GetTag(), message.GetData<void>()); | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
|  | ||||
| @ -7,6 +7,7 @@ | ||||
| #include "VoidNet_LL/TcpListener.hpp" | ||||
|  | ||||
| #include <chrono> | ||||
| #include <iostream> | ||||
|  | ||||
| namespace std::net | ||||
| { | ||||
| @ -112,7 +113,9 @@ namespace std::net | ||||
|  | ||||
| 	void TcpConnectionHandler::HandleConnections() | ||||
| 	{ | ||||
| 		int res = poll(m_pollFds.data(), m_pollFds.size(), -1); | ||||
| 		try | ||||
| 		{ | ||||
| 			int res = poll(m_pollFds.data(), m_pollFds.size(), 5); | ||||
|  | ||||
| 			if (res < 0) | ||||
| 			{ | ||||
| @ -159,16 +162,28 @@ namespace std::net | ||||
| 						NetworkMessage msg; | ||||
| 						msg.DeserializeWithoutHeader(buffer, net_header.Size); | ||||
|  | ||||
| 					if (msg.GetTag() == (uint32_t)InternalTags::Disconnect) | ||||
| 						// i? or i+1 | ||||
| 						m_pollFds.erase(m_pollFds.begin() + i); | ||||
|  | ||||
| 						// put this in a separate thread | ||||
| 						HandleMessage(msg); | ||||
|  | ||||
| 						if (msg.GetTag() == (uint32_t)InternalTags::Disconnect) | ||||
| 						{ | ||||
| 							for (size_t k = 0; k < m_list.size(); i++) | ||||
| 							{ | ||||
| 								if (m_list[k]->m_id == msg.GetSenderID()) | ||||
| 								{ | ||||
| 									std::shared_ptr<TcpConnection> c = m_list[k]; | ||||
| 									c->m_client->Close(); | ||||
| 									m_list.erase(m_list.begin() + k); | ||||
| 								} | ||||
| 							} | ||||
| 							m_pollFds.erase(m_pollFds.begin() + i); | ||||
| 						} | ||||
| 					} | ||||
| 				} | ||||
| 			} | ||||
| 		} | ||||
| 		catch (void*) {} | ||||
| 	} | ||||
|  | ||||
| 	void TcpConnectionHandler::HandleMessage(const NetworkMessage &msg) | ||||
| 	{ | ||||
|  | ||||
| @ -11,31 +11,65 @@ | ||||
|  | ||||
| class Plugin : public std::net::Plugin | ||||
| { | ||||
| 	virtual void HandleMessage(const std::net::NetworkMessage& message) override | ||||
| 	{ | ||||
| 		std::cout << "asd" << std::endl; | ||||
| 	} | ||||
|  | ||||
| }; | ||||
|  | ||||
| void dsfg(uint32_t, std::net::DistributionMode, uint32_t, uint32_t, void*) | ||||
| { | ||||
|  | ||||
| } | ||||
|  | ||||
| std::net::TcpConnection *con = new std::net::TcpConnection(); | ||||
|  | ||||
| void onData(uint32_t, std::net::DistributionMode, uint32_t, uint32_t, void*) | ||||
| { | ||||
| 	std::cout << "ondata" << std::endl; | ||||
| } | ||||
|  | ||||
| void onDisconnect(std::string) | ||||
| { | ||||
| 	std::cout << "ondisconnect" << std::endl; | ||||
| } | ||||
|  | ||||
| void onconnect() | ||||
| { | ||||
| 	std::cout << "onconnect" << std::endl; | ||||
| 	con->Disconnect(); | ||||
| } | ||||
|  | ||||
| int main() | ||||
| { | ||||
| 	std::net::Initialize(); | ||||
| 	std::net::Server server(1); | ||||
| 	server.AddPlugin(new Plugin()); | ||||
| 	server.Start(); | ||||
| 	std::net::Server *server = new std::net::Server(1); | ||||
| 	//server.AddPlugin(new Plugin()); | ||||
| 	server->Start(); | ||||
|  | ||||
| 	std::net::TcpConnection con; | ||||
| 	con.Connect(std::net::IPAddress("127.0.0.1")); | ||||
| 	//con->Connect(std::net::IPAddress("127.0.0.1")); | ||||
|  | ||||
| 	bool sent = false; | ||||
| 	//bool sent = false; | ||||
|  | ||||
| 	while (con) // 8% of my cpu | ||||
| 	//con += onData; | ||||
| 	//con += onDisconnect; | ||||
| 	//con += onconnect; | ||||
|  | ||||
| 	/*while (con.IsConnected) // 8% of my cpu | ||||
| 	{ | ||||
| 		con.ReceiveData(); | ||||
| 		if (!sent) | ||||
| 		{ | ||||
| 			sent = true; | ||||
| 			con.SendMessage(std::net::DistributionMode::Server, 0, 1); | ||||
| 			//con.SendMessage(std::net::DistributionMode::AllAndServer, 0, 1); | ||||
| 		} | ||||
| 	}*/ | ||||
|  | ||||
| 	try | ||||
|  | ||||
| 	{ | ||||
| 		server->Stop(); | ||||
| 		delete server; | ||||
| 	} | ||||
| 	catch (void* e) | ||||
| 	{ | ||||
| 		std::cout << "asd" << std::endl; | ||||
| 	} | ||||
| 	getchar(); | ||||
| } | ||||
| @ -3,6 +3,8 @@ | ||||
| #include "VoidNet_LL/Response.hpp" | ||||
| #include "VoidNet_LL/Parse.hpp" | ||||
|  | ||||
| #include <stdexcept> | ||||
|  | ||||
| namespace std::net | ||||
| { | ||||
| 	static ParseResult<HttpStatus> ParseStatus(const char* str) | ||||
|  | ||||
| @ -1,10 +1,13 @@ | ||||
| #include "VoidNet_LL/Socket.hpp" | ||||
| #include "VoidNet_LL/IPAddress.hpp" | ||||
| #include <VoidNet_LL\Init.hpp> | ||||
|  | ||||
| namespace std::net | ||||
| { | ||||
| 	void Socket::init() | ||||
| 	{ | ||||
| 		Initialize(); | ||||
|  | ||||
| 		if (GetSocketType() == SocketType::Unknown) | ||||
| 			throw invalid_argument("Unknown socket type"); | ||||
|  | ||||
| @ -55,11 +58,11 @@ namespace std::net | ||||
| 	bool Socket::Connect(const IPAddress& addr) | ||||
| 	{ | ||||
| 		sockaddr_in addr_in = addr.ToCAddr(); | ||||
| 		int32_t Return = connect(m_socket, (sockaddr*)&addr_in, sizeof(sockaddr_in)); | ||||
| 		SocketErrors Error = TranslateErrorCode(Return); | ||||
| 		int32_t retValue = connect(m_socket, (sockaddr*)&addr_in, sizeof(sockaddr_in)); | ||||
| 		SocketErrors error = TranslateErrorCode(retValue); | ||||
|  | ||||
| 		// "would block" is not an error | ||||
| 		return ((Error == SocketErrors::SE_NO_ERROR) || (Error == SocketErrors::SE_EWOULDBLOCK)); | ||||
| 		return ((error == SocketErrors::SE_NO_ERROR) || (error == SocketErrors::SE_EWOULDBLOCK)); | ||||
| 	} | ||||
|  | ||||
| 	bool Socket::WaitForPendingConnection(bool& hasPendingConnection, chrono::milliseconds t) | ||||
| @ -421,6 +424,7 @@ namespace std::net | ||||
|  | ||||
| 		return SocketErrors::SE_EINVAL; | ||||
| #else | ||||
| 		code = WSAGetLastError(); | ||||
| 		// handle the generic -1 error | ||||
| 		if (code == SOCKET_ERROR) | ||||
| 		{ | ||||
|  | ||||
		Reference in New Issue
	
	Block a user
	 TheDoctor
					TheDoctor