Nevar pievienot vairāk kā 25 tēmas Tēmai ir jāsākas ar burtu vai ciparu, tā var saturēt domu zīmes ('-') un var būt līdz 35 simboliem gara.

ptsocket.cpp 6.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. #include "ptsocket.h"
  2. PTSocket::PTSocket(QObject* p) : QTcpSocket(p)
  3. {
  4. setSocketOption(QAbstractSocket::LowDelayOption, 1);
  5. m_packetSize = 0;
  6. setTimeout(5000);
  7. setPingInterval(1000 * 7);
  8. m_state = Disconnected;
  9. m_isServerSide = false;
  10. connect(&m_timeoutTimer, SIGNAL(timeout()), this, SLOT(m_timedout()));
  11. connect(&m_pingTimer, SIGNAL(timeout()), this, SLOT(m_ping()));
  12. connect(this, SIGNAL(stateChanged(QAbstractSocket::SocketState)), this, SLOT(m_stateChanged(QAbstractSocket::SocketState)));
  13. connect(this, SIGNAL(error(QAbstractSocket::SocketError)), this, SLOT(m_socketError(QAbstractSocket::SocketError)));
  14. connect(this, SIGNAL(readyRead()), this, SLOT(m_readyRead()));
  15. }
  16. int PTSocket::getTimeout() const
  17. {
  18. return m_timeout;
  19. }
  20. int PTSocket::getPingInterval() const
  21. {
  22. return m_pingInterval;
  23. }
  24. bool PTSocket::isServerSide() const
  25. {
  26. return m_isServerSide;
  27. }
  28. QByteArray PTSocket::handshakeData()
  29. {
  30. return "LibPTSocketv1";
  31. }
  32. void PTSocket::setTimeout(int t)
  33. {
  34. m_timeout = t;
  35. if(m_timeoutTimer.isActive())
  36. m_timeoutTimer.start(m_timeout);
  37. else
  38. m_timeoutTimer.setInterval(m_timeout);
  39. }
  40. void PTSocket::setPingInterval(int p)
  41. {
  42. m_pingInterval = p;
  43. if(m_pingTimer.isActive())
  44. m_pingTimer.start(m_pingInterval);
  45. else
  46. m_pingTimer.setInterval(m_pingInterval);
  47. }
  48. void PTSocket::setIsServerSide(bool s)
  49. {
  50. m_isServerSide = s;
  51. }
  52. void PTSocket::send(int pktType, QByteArray data)
  53. {
  54. send(false, pktType, data);
  55. }
  56. void PTSocket::send(bool sys, int pktType, QByteArray data)
  57. {
  58. if(data.isEmpty())
  59. data = "\0";
  60. data.prepend(intToByteArray(pktType));
  61. data.prepend((char)sys);
  62. data.prepend(intToByteArray(data.size()));
  63. write(data);
  64. }
  65. void PTSocket::m_ping()
  66. {
  67. m_pingTimer.stop();
  68. if(m_timeoutTimer.isActive())
  69. return;
  70. send(true, PingRequest);
  71. m_timeoutTimer.start();
  72. }
  73. void PTSocket::m_readyRead()
  74. {
  75. m_timeoutTimer.stop();
  76. m_pingTimer.start();
  77. if(m_state == Handshaked)
  78. {
  79. if(m_packetSize == 0)
  80. {
  81. if((unsigned)bytesAvailable() >= sizeof(int))
  82. m_packetSize = byteArrayToInt(read(sizeof(int)));
  83. else if(bytesAvailable() != 0)
  84. m_timeoutTimer.start();
  85. }
  86. if(m_packetSize != 0)
  87. {
  88. if((unsigned)bytesAvailable() >= m_packetSize)
  89. {
  90. bool sys = (bool)read(1).at(0);
  91. QByteArray p = read(sizeof(int));
  92. int pktType = byteArrayToInt(p);
  93. QByteArray data = read(m_packetSize - 1 - p.size());
  94. if(sys)
  95. {
  96. if(pktType == PingRequest)
  97. send(true, PingAnswer);
  98. //else if(pktType == PingAnswer)
  99. }
  100. else
  101. emit packetReceived(pktType, data);
  102. m_packetSize = 0;
  103. m_readyRead();
  104. }
  105. else
  106. m_timeoutTimer.start();
  107. }
  108. }
  109. else
  110. {
  111. QByteArray handshake = handshakeData();
  112. if(bytesAvailable() >= handshake.size())
  113. {
  114. QByteArray data = read(handshake.size());
  115. if(data == handshake)
  116. {
  117. setState(Handshaked);
  118. m_readyRead();
  119. }
  120. else
  121. m_handshakeError();
  122. }
  123. else
  124. m_timeoutTimer.start();
  125. }
  126. }
  127. void PTSocket::m_timedout()
  128. {
  129. if(m_state == Disconnected)
  130. return;
  131. if(m_state == HostLookUp)
  132. setError("Unable to find the host " + peerName() + " in the time (" + QString::number(m_timeout) + " ms)");
  133. else if(m_state == Connecting)
  134. setError("The " + QString(isServerSide()? "client " + peerAddress().toString() : "server " + peerName()) + " didn't answer to the connection request in the time (" + QString::number(m_timeout) + " ms)");
  135. else if(m_state == Connected)
  136. setError("The " + QString(isServerSide()? "client " + peerAddress().toString() : "server " + peerName()) + " didn't answer to the handshake in the time (" + QString::number(m_timeout) + " ms)");
  137. else if(m_state == Handshaked)
  138. setError("The " + QString(isServerSide()? "client " + peerAddress().toString() : "server " + peerName()) + " timed out (" + QString::number(m_timeout) + " ms)");
  139. else if(m_state == Disconnecting)
  140. abort();
  141. }
  142. void PTSocket::m_handshakeError()
  143. {
  144. setError("The " + QString(isServerSide()? "client " + peerAddress().toString() : "server " + peerName()) + " didn't answer to the handshake correclty");
  145. }
  146. void PTSocket::m_stateChanged(QAbstractSocket::SocketState s)
  147. {
  148. if(s == QAbstractSocket::HostLookupState)
  149. setState(HostLookUp);
  150. else if(s == QAbstractSocket::ConnectingState)
  151. setState(Connecting);
  152. else if(s == QAbstractSocket::ConnectedState)
  153. {
  154. m_packetSize = 0;
  155. setState(Connected);
  156. write(handshakeData());
  157. }
  158. else if(s == QAbstractSocket::ClosingState)
  159. setState(Disconnecting);
  160. m_timeoutTimer.start();
  161. if(s == QAbstractSocket::UnconnectedState)
  162. {
  163. m_timeoutTimer.stop();
  164. setState(Disconnected);
  165. }
  166. }
  167. void PTSocket::m_socketError(QAbstractSocket::SocketError)
  168. {
  169. setState(Error);
  170. }
  171. void PTSocket::setState(PTSocket::State s)
  172. {
  173. if(s == m_state || (s == Error && (m_state == Disconnected || m_state == Disconnecting)))
  174. return;
  175. if(s == Error)
  176. abort();
  177. if(s == Handshaked)
  178. m_pingTimer.start();
  179. else
  180. m_pingTimer.stop();
  181. if(m_state == Error && (s != HostLookUp && s != Connecting))
  182. return;
  183. m_state = s;
  184. emit stateChanged(m_state);
  185. }
  186. void PTSocket::setError(QString e)
  187. {
  188. if(m_state == Error)
  189. return;
  190. setErrorString(e);
  191. setState(Error);
  192. }
  193. QByteArray PTSocket::intToByteArray(int a)
  194. {
  195. QByteArray data;
  196. for(unsigned char i = 0; i < sizeof(a); ++i)
  197. data.append((char)(a >> 8 * i));
  198. return data;
  199. }
  200. int PTSocket::byteArrayToInt(QByteArray d)
  201. {
  202. int a = 0;
  203. for(unsigned char i = 0; i < d.size(); ++i)
  204. a |= (unsigned int)((unsigned char)d.at(i) << i * 8);
  205. return a;
  206. }