TDME2 1.9.121
UDPServerClient.cpp
Go to the documentation of this file.
1#include <stdio.h>
2
3#include <map>
4#include <sstream>
5#include <string>
6#include <typeinfo>
7
8#include <tdme/tdme.h>
12#include <tdme/utilities/RTTI.h>
13#include <tdme/utilities/Time.h>
14
15using std::ios_base;
16using std::map;
17using std::pair;
18using std::string;
19using std::stringstream;
20
26
27UDPServerClient::UDPServerClient(const uint32_t clientId, const string& ip, const unsigned int port) :
28 server(NULL),
29 ioThread(NULL),
30 clientId(clientId),
31 ip(ip),
32 port(port),
33 shutdownRequested(false),
34 messageMapSafeMutex("nioudpserverclient_messagemapsafe") {
35 // key
36 ostringstream tmp;
37 tmp << KEY_PREFIX_UNNAMED;
38 tmp << clientId;
39 key = tmp.str();
40}
41
43}
44
46 return server;
47}
48
50 return clientId;
51}
52
53const string& UDPServerClient::getIp() const {
54 return ip;
55}
56
57const unsigned int UDPServerClient::getPort() const {
58 return port;
59}
60
61const string& UDPServerClient::getKey() const {
62 return key;
63}
64
65const bool UDPServerClient::setKey(const string &key) {
66 if (key.size() > 0 &&
67 key.size() < 256 &&
68 server->setClientKey(this, key) == true) {
69 this->key = key;
70 return true;
71 } else {
72 return false;
73 }
74}
75
77 stringstream* frame = new stringstream();
78 frame->exceptions(std::ios_base::failbit | std::ios_base::badbit);
80 return frame;
81}
82
83void UDPServerClient::send(stringstream* frame, bool safe, bool deleteFrame) {
84 try {
85 // seek writing to end of stream
86 frame->seekp(0, ios_base::end);
87
88 // check size
89 if (frame->tellp() > 512) {
90 delete frame;
91 throw NetworkServerException("message too big");
92 }
93
94 server->sendMessage(this, frame, safe, deleteFrame, UDPServer::MESSAGETYPE_MESSAGE);
95 } catch (NetworkServerException &exception) {
96 // shut down client
97 shutdown();
98
99 // log
100 Console::println(
101 "UDPServerClient::send(): send failed for client '" +
102 (ip) +
103 "': " +
104 (RTTI::demangle(typeid(exception).name())) +
105 ": " +
106 (exception.what())
107 );
108 }
109}
110
111bool UDPServerClient::processSafeMessage(const uint32_t messageId) {
112 bool messageProcessed = false;
113 MessageMapSafe::iterator it;
114
115 //
117
118 // check if message has been already processed
119 it = messageMapSafe.find(messageId);
120 if (it != messageMapSafe.end()) {
121 // yep, we did
122 messageProcessed = true;
123 Message* message = &it->second;
124 message->receptions++;
125 } else {
126 // nope, just remember message
127 Message message;
128 message.messageId = messageId;
129 message.receptions = 1;
130 message.time = Time::getCurrentMillis();
131 // TODO: check for overflow
132 messageMapSafe.insert(it, pair<uint32_t, Message>(messageId, message));
133 }
134
135 //
137
138 // always send acknowlegdement to client
139 stringstream* frame = createFrame();
140 try {
141 server->sendMessage(this, frame, false, true, UDPServer::MESSAGETYPE_ACKNOWLEDGEMENT, messageId);
142 } catch (NetworkServerException &exception) {
143 // shut down client
144 shutdown();
145
146 // log
147 Console::println(
148 "UDPServerClient::sendAcknowledgement(): send failed for client '" +
149 (ip) +
150 "': " +
151 (RTTI::demangle(typeid(exception).name())) +
152 ": " +
153 (exception.what())
154 );
155 }
156
157 // return if message should be processed
158 return messageProcessed == true?false:true;
159}
160
162 stringstream* frame = createFrame();
163 try {
164 char keySize = key.size();
165 frame->write(&keySize, 1);
166 frame->write(key.data(), keySize);
167 server->sendMessage(this, frame, true, true, UDPServer::MESSAGETYPE_CONNECT);
168 } catch (NetworkServerException &exception) {
169 // shut down client
170 shutdown();
171
172 // log
173 Console::println(
174 "UDPServerClient::sendConnected(): send failed for client '" +
175 (ip) +
176 "': " +
177 (RTTI::demangle(typeid(exception).name())) +
178 ": " +
179 (exception.what())
180 );
181 }
182}
183
185 shutdownRequested = true;
186}
187
188void UDPServerClient::onFrameReceived(stringstream* frame, const uint32_t messageId, const uint8_t retries) {
189 // create request
190 ServerRequest* request = new ServerRequest(
192 this,
194 frame,
195 messageId,
196 retries
197 );
198 // delegate it to thread pool, but make it declinable
199 if (server->workerThreadPool->addElement(request, true) == false) {
200 // element was declined
201 Console::println("UDPServerClient::onFrameReceived(): client request declined from '" + (ip) + "'. Shutting down client");
202 // release client reference
204 // delete frame
205 delete frame;
206 // delete request
207 delete request;
208 // shutdown client
209 shutdown();
210 }
211}
212
214 // create request
215 ServerRequest* request = new ServerRequest(
217 this,
219 NULL,
222 );
223 // delegate it to thread pool, but make close request not declinable
224 server->workerThreadPool->addElement(request, false);
225 // server call back
226 server->closeClient(this);
227}
228
230 // acquire reference for worker
232
233 // create request
234 ServerRequest* request = new ServerRequest(
236 this,
238 NULL,
241 );
242 // delegate it to thread pool, but make close request not declinable
243 server->workerThreadPool->addElement(request, false);
244}
245
246void UDPServerClient::fireEvent(const string &type) {
247 // acquire reference for worker
249
250 // create request
251 ServerRequest* request = new ServerRequest(
253 this,
254 type,
255 NULL,
258 );
259
260 // delegate it to thread pool, but make close request not declinable
261 server->workerThreadPool->addElement(request, false);
262}
263
264
266 //
268
269 // check if message has been already processed
270 uint64_t now = Time::getCurrentMillis();
271 MessageMapSafe::iterator it = messageMapSafe.begin();
272 while (it != messageMapSafe.end()) {
273 Message* message = &it->second;
274 if (message->time < now - MESSAGESSAFE_KEEPTIME) {
275 messageMapSafe.erase(it++);
276 continue;
277 }
278 ++it;
279 }
280
281 //
283}
284
285uint64_t UDPServerClient::getRetryTime(const uint8_t retries) {
286 if (retries == 0) return 0L;
289}
290
Base exception class for network server exceptions.
static STATIC_DLL_IMPEXT const char * KEY_PREFIX_UNNAMED
Definition: ServerClient.h:37
static const uint32_t MESSAGE_ID_UNSUPPORTED
Definition: ServerRequest.h:37
static const uint8_t MESSAGE_RETRIES_NONE
Definition: ServerRequest.h:38
bool setClientKey(CLIENT *client, const std::string &clientKey)
sets a client identification key
Definition: Server.h:155
void closeClient(CLIENT *client)
closes a client connection
Definition: Server.h:181
Base class for network UDP server clients.
void cleanUpSafeMessages()
Clean up safe messages.
void init()
initiates this network client
static stringstream * createFrame()
Creates a frame to be used with send.
void fireEvent(const string &type)
fires an custom event
void close()
Shuts down this network client.
uint64_t getRetryTime(const uint8_t retries)
const string & getIp() const
returns client's ip
void shutdown()
Shuts down this network client.
void send(stringstream *frame, bool safe=true, bool deleteFrame=true)
Sends a frame to client, takes over ownership of frame.
const string & getKey() const
Client identification key.
const unsigned int getPort() const
returns client port
virtual void onFrameReceived(stringstream *frame, const uint32_t messageId=0, const uint8_t retries=0)
Event, which will be called if frame has been received, defaults to worker thread pool.
const bool setKey(const string &key)
sets the clients identification key
const uint32_t getClientId()
Get client id.
UDPServer * getServer()
Returns server.
virtual ~UDPServerClient()
public destructor, should only be called implicitly by Reference::releaseReference()
void sendConnected()
Sends an connect message to client.
bool processSafeMessage(const uint32_t messageId)
Checks if message has already been processed and sends an acknowlegdement to client / safe client mes...
static const uint64_t MESSAGEACK_RESENDTIMES[MESSAGEACK_RESENDTIMES_TRIES]
Base class for network UDP servers.
Definition: UDPServer.h:39
static void initializeHeader(stringstream *frame)
Writes a empty header to message.
Definition: UDPServer.cpp:229
ServerWorkerThreadPool * workerThreadPool
Definition: UDPServer.h:208
void sendMessage(const UDPServerClient *client, stringstream *frame, const bool safe, const bool deleteFrame, const MessageType messageType, const uint32_t messageId=MESSAGE_ID_NONE)
pushes a message to be send, takes over ownership of frame
Definition: UDPServer.cpp:469
void unlock()
Unlocks this mutex.
Definition: Mutex.cpp:48
void lock()
Locks the mutex, additionally mutex locks will block until other locks have been unlocked.
Definition: Mutex.cpp:39
bool addElement(T *element, const bool declinable)
Adds an element to this queue, signals threads which waits for an element.
Definition: Queue.h:83
Console class.
Definition: Console.h:26
Integer class.
Definition: Integer.h:26
Run time type information utility class.
Definition: RTTI.h:14
void releaseReference()
releases a reference, thus decrementing the counter and delete it if reference counter is zero
Definition: Reference.cpp:20
void acquireReference()
acquires a reference, incrementing the counter
Definition: Reference.cpp:16
Time utility class.
Definition: Time.h:21