client.cpp 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222
  1. #include <client.hpp>
  2. #include <arpa/inet.h>
  3. #include <netdb.h>
  4. #include <string.h>
  5. #include <sys/socket.h>
  6. #include <sys/types.h>
  7. #include <unistd.h>
  8. #include <functional>
  9. #include <cstring>
  10. #include <QDebug>
  11. #include <QByteArray>
  12. #include <string>
  13. #include <iostream>
  14. #include <vector>
  15. #include <future>
  16. #include <headers/kmessage_codec.hpp>
  17. #include <headers/json.hpp>
  18. #include <headers/util.hpp>
  19. using namespace KData;
  20. using json = nlohmann::json;
  21. static const int MAX_BUFFER_SIZE = 2048;
  22. flatbuffers::FlatBufferBuilder builder(1024);
  23. /**
  24. * @brief Client::createMessageHandler
  25. * @param cb
  26. * @return
  27. */
  28. Client::MessageHandler Client::createMessageHandler(
  29. std::function<void()> cb) {
  30. return MessageHandler(cb);
  31. }
  32. /**
  33. * @brief Client::Client
  34. * @constructor
  35. * @param parent
  36. * @param count
  37. * @param arguments
  38. */
  39. Client::Client(QWidget *parent, int count, char** arguments) : QDialog(parent), argc(count), argv(arguments), m_client_socket_fd(-1) {}
  40. /**
  41. * @brief Client::~Client
  42. * @destructor
  43. */
  44. Client::~Client() {
  45. closeConnection();
  46. }
  47. /**
  48. * @brief Client::handleMessages
  49. */
  50. void Client::handleMessages() {
  51. char receive_buffer[2048];
  52. for (;;) {
  53. memset(receive_buffer, 0, 2048);
  54. ssize_t bytes_received = 0;
  55. bytes_received = recv(m_client_socket_fd, receive_buffer, 2048 - 2, 0);
  56. receive_buffer[2047] = 0;
  57. if (bytes_received == 0) {
  58. break;
  59. }
  60. emit Client::messageReceived(QString::fromUtf8(receive_buffer, 2048));
  61. }
  62. memset(receive_buffer, 0, 2048);
  63. ::close(m_client_socket_fd);
  64. }
  65. /**
  66. * @brief Client::start
  67. * @return A meaningless integer
  68. */
  69. void Client::start() {
  70. if (m_client_socket_fd == -1) {
  71. m_client_socket_fd = socket(AF_INET, SOCK_STREAM, 0);
  72. if (m_client_socket_fd != -1) {
  73. sockaddr_in server_socket;
  74. char* end;
  75. server_socket.sin_family = AF_INET;
  76. auto port_value = strtol(argv[2], &end, 10);
  77. if (port_value < 0 || end == argv[2]) {
  78. return;
  79. }
  80. int socket_option = 1;
  81. // Free up the port to begin listening again
  82. setsockopt(m_client_socket_fd, SOL_SOCKET, SO_REUSEADDR, &socket_option,
  83. sizeof(socket_option));
  84. server_socket.sin_port = htons(port_value);
  85. inet_pton(AF_INET, argv[1], &server_socket.sin_addr.s_addr);
  86. if (::connect(m_client_socket_fd, reinterpret_cast<sockaddr*>(&server_socket),
  87. sizeof(server_socket)) != -1) {
  88. std::string start_operation_string = createOperation("start", {});
  89. std::vector<uint8_t> fb_byte_vector{start_operation_string.begin(), start_operation_string.end()};
  90. auto byte_vector = builder.CreateVector(fb_byte_vector);
  91. auto message = CreateMessage(builder, 69, byte_vector);
  92. builder.Finish(message);
  93. uint8_t* encoded_message_buffer = builder.GetBufferPointer();
  94. uint32_t size = builder.GetSize();
  95. qDebug() << "Size is " << size;
  96. uint8_t send_buffer[MAX_BUFFER_SIZE];
  97. memset(send_buffer, 0, MAX_BUFFER_SIZE);
  98. send_buffer[0] = (size & 0xFF) >> 24;
  99. send_buffer[1] = (size & 0xFF) >> 16;
  100. send_buffer[2] = (size & 0xFF) >> 8;
  101. send_buffer[3] = (size & 0xFF);
  102. std::memcpy(send_buffer + 4, encoded_message_buffer, size);
  103. qDebug() << "Ready to send:";
  104. std::string message_to_send{};
  105. for (unsigned int i = 0; i < (size + 4); i++) {
  106. message_to_send += (char)*(send_buffer + i);
  107. }
  108. qDebug() << message_to_send.c_str();
  109. // Send start operation
  110. ::send(m_client_socket_fd, send_buffer, size + 4, 0);
  111. builder.Clear();
  112. // Delegate message handling to its own thread
  113. std::function<void()> message_send_fn = [this]() {
  114. this->handleMessages();
  115. };
  116. MessageHandler message_handler = createMessageHandler(message_send_fn);
  117. // Handle received messages on separate thread
  118. std::thread (message_handler).detach();
  119. } else {
  120. qDebug() << errno;
  121. ::close(m_client_socket_fd);
  122. }
  123. } else {
  124. qDebug() << "Failed to create new connection";
  125. }
  126. } else {
  127. qDebug() << "Connection already in progress";
  128. }
  129. }
  130. /**
  131. * @brief Client::sendMessage
  132. * @param s[in] <const QString&> The message to send
  133. */
  134. void Client::sendMessage(const QString& s) {
  135. if (m_client_socket_fd != -1) {
  136. std::string json_string = createMessage(s.toUtf8().data());
  137. std::vector<uint8_t> fb_byte_vector{json_string.begin(), json_string.end()};
  138. auto byte_vector = builder.CreateVector(fb_byte_vector);
  139. auto message = CreateMessage(builder, 69, byte_vector);
  140. builder.Finish(message);
  141. uint8_t* encoded_message_buffer = builder.GetBufferPointer();
  142. uint32_t size = builder.GetSize();
  143. qDebug() << "Size is " << size;
  144. uint8_t send_buffer[MAX_BUFFER_SIZE];
  145. memset(send_buffer, 0, MAX_BUFFER_SIZE);
  146. send_buffer[0] = (size & 0xFF) >> 24;
  147. send_buffer[1] = (size & 0xFF) >> 16;
  148. send_buffer[2] = (size & 0xFF) >> 8;
  149. send_buffer[3] = (size & 0xFF);
  150. std::memcpy(send_buffer + 4, encoded_message_buffer, size);
  151. qDebug() << "Ready to send:";
  152. std::string message_to_send{};
  153. for (unsigned int i = 0; i < (size + 4); i++) {
  154. message_to_send += (char)*(send_buffer + i);
  155. }
  156. qDebug() << message_to_send.c_str();
  157. ::send(m_client_socket_fd, send_buffer, size + 4, 0);
  158. builder.Clear();
  159. } else {
  160. qDebug() << "You must first open a connection";
  161. }
  162. }
  163. void Client::closeConnection() {
  164. if (m_client_socket_fd != -1) {
  165. std::string stop_operation_string = createOperation("stop", {});
  166. std::vector<uint8_t> fb_byte_vector{stop_operation_string.begin(), stop_operation_string.end()};
  167. auto byte_vector = builder.CreateVector(fb_byte_vector);
  168. auto message = CreateMessage(builder, 69, byte_vector);
  169. builder.Finish(message);
  170. uint8_t* encoded_message_buffer = builder.GetBufferPointer();
  171. uint32_t size = builder.GetSize();
  172. qDebug() << "Size is " << size;
  173. uint8_t send_buffer[MAX_BUFFER_SIZE];
  174. memset(send_buffer, 0, MAX_BUFFER_SIZE);
  175. send_buffer[0] = (size & 0xFF) >> 24;
  176. send_buffer[1] = (size & 0xFF) >> 16;
  177. send_buffer[2] = (size & 0xFF) >> 8;
  178. send_buffer[3] = (size & 0xFF);
  179. std::memcpy(send_buffer + 4, encoded_message_buffer, size);
  180. qDebug() << "Ready to send:";
  181. std::string message_to_send{};
  182. for (unsigned int i = 0; i < (size + 4); i++) {
  183. message_to_send += (char)*(send_buffer + i);
  184. }
  185. qDebug() << message_to_send.c_str();
  186. // Send stop operation
  187. ::send(m_client_socket_fd, send_buffer, size + 4, 0);
  188. builder.Clear();
  189. // Clean up socket file descriptor
  190. ::shutdown(m_client_socket_fd, SHUT_RDWR);
  191. ::close(m_client_socket_fd);
  192. m_client_socket_fd = -1;
  193. }
  194. }