source: network-game/common/MessageProcessor.cpp@ 2ee386d

Last change on this file since 2ee386d was f9cb9fb, checked in by dportnoy <dmp1488@…>, 11 years ago

The ackedMessages list in MessageProcessor now stores ack records based on sender address and message id so that messages with the same id, but from different senders are distinguishable

  • Property mode set to 100644
File size: 4.8 KB
Line 
1#include "MessageProcessor.h"
2
3#include <iostream>
4#include <fstream>
5
6#include "Common.h"
7
8MessageProcessor::MessageProcessor() {
9 lastUsedId = 0;
10}
11
12MessageProcessor::~MessageProcessor() {
13}
14
15int MessageProcessor::sendMessage(NETWORK_MSG *msg, int sock, struct sockaddr_in *dest, ofstream* outputLog) {
16 cout << "Sending message of type " << msg->type << endl;
17
18 msg->id = ++lastUsedId;
19 MessageContainer message(*msg, *dest);
20
21 if (outputLog)
22 (*outputLog) << "Sending message (id " << msg->id << ") of type " << MessageContainer::getMsgTypeString(msg->type) << endl;
23
24 sentMessages[msg->id][dest->sin_addr.s_addr] = message;
25
26 int ret = sendto(sock, (char*)msg, sizeof(NETWORK_MSG), 0, (struct sockaddr *)dest, sizeof(struct sockaddr_in));
27
28 return ret;
29}
30
31int MessageProcessor::receiveMessage(NETWORK_MSG *msg, int sock, struct sockaddr_in *source, ofstream* outputLog) {
32 socklen_t socklen = sizeof(struct sockaddr_in);
33
34 // assume we don't care about the value of socklen
35 int ret = recvfrom(sock, (char*)msg, sizeof(NETWORK_MSG), 0, (struct sockaddr *)source, &socklen);
36
37 if (ret == -1)
38 return ret;
39
40 // add id to the NETWORK_MSG struct
41 if (msg->type == MSG_TYPE_ACK) {
42 if (!sentMessages[msg->id][source->sin_addr.s_addr].getAcked()) {
43 sentMessages[msg->id][source->sin_addr.s_addr].setAcked(true);
44 sentMessages[msg->id][source->sin_addr.s_addr].setTimeAcked(getCurrentMillis());
45 if (outputLog)
46 (*outputLog) << "Received ack for message id " << msg->id << endl;
47 }
48
49 return -1; // don't do any further processing
50 }else {
51 bool isDuplicate = false;
52 map<unsigned int, unsigned long long>& ackedPlayerMessages = ackedMessages[source->sin_addr.s_addr];
53
54 if (ackedPlayerMessages.find(msg->id) != ackedPlayerMessages.end()) {
55 isDuplicate = true;
56 cout << "Got duplicate of type " << msg->type << endl;
57 if (outputLog)
58 (*outputLog) << "Received duplicate (id " << msg->id << ") of type " << MessageContainer::getMsgTypeString(msg->type) << endl;
59 }else {
60 cout << "Got message of type " << msg->type << endl;
61 if (outputLog)
62 (*outputLog) << "Received message (id " << msg->id << ") of type " << MessageContainer::getMsgTypeString(msg->type) << endl;
63 }
64
65 ackedPlayerMessages[msg->id] = getCurrentMillis();
66
67 NETWORK_MSG ack;
68 ack.id = msg->id;
69 ack.type = MSG_TYPE_ACK;
70
71 sendto(sock, (char*)&ack, sizeof(NETWORK_MSG), 0, (struct sockaddr *)source, sizeof(struct sockaddr_in));
72
73 if (isDuplicate)
74 return -1;
75 }
76
77 return ret;
78}
79
80void MessageProcessor::resendUnackedMessages(int sock, ofstream* outputLog) {
81 map<unsigned int, map<unsigned long, MessageContainer> >::iterator it;
82 map<unsigned long, MessageContainer>::iterator it2;
83 map<unsigned long, MessageContainer> sentMsg;
84
85 for (it = sentMessages.begin(); it != sentMessages.end(); it++) {
86 sentMsg = it->second;
87 for (it2 = sentMsg.begin(); it2 != sentMsg.end(); it2++) {
88 if (!(it2->second.getAcked())) {
89 sendto(sock, (const char*)it2->second.getMessage(), sizeof(NETWORK_MSG), 0, (struct sockaddr *)&it2->first, sizeof(struct sockaddr_in));
90 }
91 }
92 }
93}
94
95void MessageProcessor::cleanAckedMessages(ofstream* outputLog) {
96 map<unsigned int, map<unsigned long, MessageContainer> >::iterator it = sentMessages.begin();
97 map<unsigned long, MessageContainer>::iterator it2;
98
99 while (it != sentMessages.end()) {
100 it2 = it->second.begin();
101 while (it2 != it->second.end()) {
102 if (it2->second.getAcked()) {
103 if ((getCurrentMillis() - it2->second.getTimeAcked()) > 1000) {
104 if (outputLog)
105 (*outputLog) << "Removing id " << it2->second.getMessage()->id << " from the acked record" << endl;
106 it->second.erase(it2++);
107 }else
108 it2++;
109 }else
110 it2++;
111 }
112
113 if (it->second.size() == 0)
114 sentMessages.erase(it++);
115 else
116 it++;
117 }
118
119 map<unsigned long, map<unsigned int, unsigned long long> >::iterator it3 = ackedMessages.begin();
120 map<unsigned int, unsigned long long>::iterator it4;
121
122 // somehow want to delete the inner map once that player logs out
123 while (it3 != ackedMessages.end()) {
124 it4 = it3->second.begin();
125 while (it4 != it3->second.end()) {
126 if ((getCurrentMillis() - it4->second) > 500)
127 it3->second.erase(it4++);
128 else
129 it4++;
130 }
131 it3++;
132 }
133}
134
135map<unsigned int, map<unsigned long, MessageContainer> >& MessageProcessor::getSentMessages() {
136 return this->sentMessages;
137}
138
139map<unsigned long, map<unsigned int, unsigned long long> >& MessageProcessor::getAckedMessages() {
140 return this->ackedMessages;
141}
Note: See TracBrowser for help on using the repository browser.