You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

ptsocket.cpp 8.0KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323
  1. #include "ptsocket.h"
  2. PTSocket::PTSocket(QObject* p) : QTcpSocket(p)
  3. {
  4. #ifdef PT_DEBUG
  5. m_willPing = true;
  6. dbgSetWillHandshake(true);
  7. #endif
  8. setSocketOption(QAbstractSocket::LowDelayOption, 1);
  9. m_packetSize = 0;
  10. m_timeoutTimer.setSingleShot(true);
  11. setTimeout(5000);
  12. m_pingTimer.setSingleShot(true);
  13. setPingInterval(1000 * 7);
  14. m_state = Disconnected;
  15. m_isServerSide = false;
  16. connect(&m_timeoutTimer, SIGNAL(timeout()), this, SLOT(m_timedout()));
  17. connect(&m_pingTimer, SIGNAL(timeout()), this, SLOT(m_ping()));
  18. connect(this, SIGNAL(stateChanged(QAbstractSocket::SocketState)), this, SLOT(m_stateChanged(QAbstractSocket::SocketState)));
  19. connect(this, SIGNAL(error(QAbstractSocket::SocketError)), this, SLOT(m_socketError(QAbstractSocket::SocketError)));
  20. connect(this, SIGNAL(readyRead()), this, SLOT(m_readyRead()));
  21. }
  22. int PTSocket::getTimeout() const
  23. {
  24. return m_timeout;
  25. }
  26. int PTSocket::getPingInterval() const
  27. {
  28. return m_pingInterval;
  29. }
  30. bool PTSocket::isServerSide() const
  31. {
  32. return m_isServerSide;
  33. }
  34. QString PTSocket::getError() const
  35. {
  36. return (m_error.isEmpty()? errorString() : m_error);
  37. }
  38. QByteArray PTSocket::handshakeData()
  39. {
  40. return "LibPTSocketv1";
  41. }
  42. void PTSocket::setTimeout(int t)
  43. {
  44. m_timeout = t;
  45. if(m_timeoutTimer.isActive())
  46. m_timeoutTimer.start(m_timeout);
  47. else
  48. m_timeoutTimer.setInterval(m_timeout);
  49. }
  50. void PTSocket::setPingInterval(int p)
  51. {
  52. m_pingInterval = p;
  53. if(m_pingTimer.isActive())
  54. m_pingTimer.start(m_pingInterval);
  55. else
  56. m_pingTimer.setInterval(m_pingInterval);
  57. }
  58. void PTSocket::setIsServerSide(bool s)
  59. {
  60. m_isServerSide = s;
  61. }
  62. void PTSocket::send(int pktType, QByteArray data)
  63. {
  64. send(false, pktType, data);
  65. }
  66. void PTSocket::disconnectFromServer()
  67. {
  68. disconnectFromHost();
  69. }
  70. void PTSocket::dbgSetWillPing(bool w)
  71. {
  72. #ifdef PT_DEBUG
  73. m_willPing = w;
  74. #endif
  75. }
  76. void PTSocket::dbgSetWillHandshake(bool w)
  77. {
  78. #ifdef PT_DEBUG
  79. m_willHandshake = w;
  80. m_willHandshakeFake1 = false;
  81. m_willHandshakeFake2 = false;
  82. #endif
  83. }
  84. void PTSocket::dbgSetWillHandshakeFake1(bool w)
  85. {
  86. #ifdef PT_DEBUG
  87. m_willHandshakeFake1 = w;
  88. if(!w)
  89. m_willHandshake = true;
  90. else
  91. m_willHandshake = false;
  92. m_willHandshakeFake2 = false;
  93. #endif
  94. }
  95. void PTSocket::dbgSetWillHandshakeFake2(bool w)
  96. {
  97. #ifdef PT_DEBUG
  98. m_willHandshakeFake2 = w;
  99. if(!w)
  100. m_willHandshake = true;
  101. else
  102. m_willHandshake = false;
  103. m_willHandshakeFake1 = false;
  104. #endif
  105. }
  106. void PTSocket::send(bool sys, int pktType, QByteArray data)
  107. {
  108. if(data.isEmpty())
  109. data = "\0";
  110. data.prepend(intToByteArray(pktType));
  111. data.prepend((char)sys);
  112. data.prepend(intToByteArray(data.size()));
  113. write(data);
  114. }
  115. void PTSocket::m_ping()
  116. {
  117. m_pingTimer.stop();
  118. if(m_timeoutTimer.isActive())
  119. return;
  120. #ifdef PT_DEBUG
  121. if(m_willPing)
  122. send(true, PingRequest);
  123. #else
  124. send(true, PingRequest);
  125. #endif
  126. m_timeoutTimer.start();
  127. }
  128. void PTSocket::m_readyRead()
  129. {
  130. m_timeoutTimer.stop();
  131. m_pingTimer.start();
  132. if(m_state == Handshaked)
  133. {
  134. if(m_packetSize == 0)
  135. {
  136. if((unsigned)bytesAvailable() >= sizeof(int))
  137. m_packetSize = byteArrayToInt(read(sizeof(int)));
  138. else if(bytesAvailable() != 0)
  139. m_timeoutTimer.start();
  140. }
  141. if(m_packetSize != 0)
  142. {
  143. if((unsigned)bytesAvailable() >= m_packetSize)
  144. {
  145. bool sys = (bool)read(1).at(0);
  146. QByteArray p = read(sizeof(int));
  147. int pktType = byteArrayToInt(p);
  148. QByteArray data = read(m_packetSize - 1 - p.size());
  149. if(sys)
  150. {
  151. if(pktType == PingRequest)
  152. {
  153. #ifdef PT_DEBUG
  154. if(m_willPing)
  155. send(true, PingAnswer);
  156. #else
  157. send(true, PingAnswer);
  158. #endif
  159. }
  160. //else if(pktType == PingAnswer)
  161. }
  162. else
  163. emit packetReceived(pktType, data);
  164. m_packetSize = 0;
  165. m_readyRead();
  166. }
  167. else
  168. m_timeoutTimer.start();
  169. }
  170. }
  171. else
  172. {
  173. QByteArray handshake = handshakeData();
  174. if(bytesAvailable() >= handshake.size())
  175. {
  176. QByteArray data = read(handshake.size());
  177. if(data == handshake)
  178. {
  179. setState(Handshaked);
  180. m_readyRead();
  181. }
  182. else
  183. m_handshakeError();
  184. }
  185. else
  186. m_timeoutTimer.start();
  187. }
  188. }
  189. void PTSocket::m_timedout()
  190. {
  191. if(m_state == Disconnected)
  192. return;
  193. if(m_state == HostLookUp)
  194. setError("Unable to find the host " + peerName() + " in the time (" + QString::number(m_timeout) + " ms)");
  195. else if(m_state == Connecting)
  196. setError("The " + QString(isServerSide()? "client " + peerAddress().toString() : "server " + peerName()) + " didn't answer to the connection request in the time (" + QString::number(m_timeout) + " ms)");
  197. else if(m_state == Connected)
  198. setError("The " + QString(isServerSide()? "client " + peerAddress().toString() : "server " + peerName()) + " didn't answer to the handshake in the time (" + QString::number(m_timeout) + " ms)");
  199. else if(m_state == Handshaked)
  200. setError("The " + QString(isServerSide()? "client " + peerAddress().toString() : "server " + peerName()) + " timed out (" + QString::number(m_timeout) + " ms)");
  201. else if(m_state == Disconnecting)
  202. abort();
  203. }
  204. void PTSocket::m_handshakeError()
  205. {
  206. setError("The " + QString(isServerSide()? "client " + peerAddress().toString() : "server " + peerName()) + " didn't answer to the handshake correclty");
  207. }
  208. void PTSocket::m_stateChanged(QAbstractSocket::SocketState s)
  209. {
  210. if(s == QAbstractSocket::HostLookupState)
  211. setState(HostLookUp);
  212. else if(s == QAbstractSocket::ConnectingState)
  213. setState(Connecting);
  214. else if(s == QAbstractSocket::ConnectedState)
  215. {
  216. m_packetSize = 0;
  217. setState(Connected);
  218. #ifdef PT_DEBUG
  219. if(m_willHandshakeFake1)
  220. write("LibPTSocketv");
  221. else if(m_willHandshakeFake2)
  222. write("42LibPTSocketv4242");
  223. else if(m_willHandshake)
  224. write(handshakeData());
  225. #else
  226. write(handshakeData());
  227. #endif
  228. }
  229. else if(s == QAbstractSocket::ClosingState)
  230. setState(Disconnecting);
  231. m_timeoutTimer.start();
  232. if(s == QAbstractSocket::UnconnectedState)
  233. {
  234. m_timeoutTimer.stop();
  235. setState(Disconnected);
  236. }
  237. }
  238. void PTSocket::m_socketError(QAbstractSocket::SocketError s)
  239. {
  240. if(s != QAbstractSocket::RemoteHostClosedError)
  241. {
  242. qDebug()<<"Socket error:"<<s;
  243. setState(Error);
  244. }
  245. }
  246. void PTSocket::setState(PTSocket::State s)
  247. {
  248. if(s == m_state)
  249. return;
  250. if(s == Error && (m_state == Disconnected || m_state == Disconnecting))
  251. return;
  252. if(m_state == Error && (s == Disconnected || s == Disconnecting))
  253. return;
  254. m_state = s;
  255. if(s == Error)
  256. abort();
  257. else
  258. m_error.clear();
  259. if(s == Handshaked)
  260. m_pingTimer.start();
  261. else
  262. m_pingTimer.stop();
  263. emit stateChanged(m_state);
  264. }
  265. void PTSocket::setError(QString e)
  266. {
  267. if(m_state == Error || m_state == Disconnected || m_state == Disconnecting)
  268. return;
  269. m_error = e;
  270. setState(Error);
  271. }
  272. QByteArray PTSocket::intToByteArray(int a)
  273. {
  274. QByteArray data;
  275. for(unsigned char i = 0; i < sizeof(a); ++i)
  276. data.append((char)(a >> 8 * i));
  277. return data;
  278. }
  279. int PTSocket::byteArrayToInt(QByteArray d)
  280. {
  281. int a = 0;
  282. for(unsigned char i = 0; i < d.size(); ++i)
  283. a |= (unsigned int)((unsigned char)d.at(i) << i * 8);
  284. return a;
  285. }
  286. QDebug operator <<(QDebug dbg, const PTSocket::State s)
  287. {
  288. const QMetaObject & mo = PTSocket::staticMetaObject;
  289. QMetaEnum me = mo.enumerator(mo.indexOfEnumerator("State"));
  290. return dbg<<me.valueToKey(s);
  291. }