client.cpp 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  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 <iostream>
  13. #include <vector>
  14. #include <future>
  15. #include <headers/kmessage_codec.hpp>
  16. #include <headers/json.hpp>
  17. #include <headers/util.hpp>
  18. using namespace KData;
  19. using json = nlohmann::json;
  20. static const int MAX_BUFFER_SIZE = 2048;
  21. flatbuffers::FlatBufferBuilder builder(1024);
  22. /**
  23. * @brief Client::createMessageHandler
  24. * @param cb
  25. * @return
  26. */
  27. Client::MessageHandler Client::createMessageHandler(
  28. std::function<void()> cb) {
  29. return MessageHandler(cb);
  30. }
  31. /**
  32. * @brief Client::Client
  33. * @constructor
  34. * @param parent
  35. * @param count
  36. * @param arguments
  37. */
  38. Client::Client(QWidget *parent, int count, char** arguments) : QDialog(parent), argc(count), argv(arguments), m_client_socket_fd(-1), m_commands({}), executing(false) {
  39. qRegisterMetaType<QVector<QString>>("QVector<QString>");
  40. }
  41. /**
  42. * @brief Client::~Client
  43. * @destructor
  44. */
  45. Client::~Client() {
  46. closeConnection();
  47. }
  48. /**
  49. * @brief Client::handleMessages
  50. */
  51. void Client::handleMessages() {
  52. uint8_t receive_buffer[2048];
  53. for (;;) {
  54. memset(receive_buffer, 0, 2048);
  55. ssize_t bytes_received = 0;
  56. bytes_received = recv(m_client_socket_fd, receive_buffer, 2048 - 2, 0);
  57. receive_buffer[2047] = 0;
  58. if (bytes_received == 0) {
  59. break;
  60. }
  61. size_t end_idx = findNullIndex(receive_buffer);
  62. std::string data_string{receive_buffer, receive_buffer + end_idx};
  63. StringVec s_v{};
  64. if (isNewSession(data_string.c_str())) {
  65. m_commands = getArgMap(data_string.c_str());
  66. for (const auto& [k, v] : m_commands) {
  67. s_v.push_back(v.data());
  68. }
  69. emit Client::messageReceived(COMMANDS_UPDATE_TYPE, "", s_v);
  70. }
  71. std::string formatted_json = getJsonString(data_string);
  72. emit Client::messageReceived(MESSAGE_UPDATE_TYPE, QString::fromUtf8(formatted_json.data(), formatted_json.size()), {});
  73. }
  74. memset(receive_buffer, 0, 2048);
  75. ::close(m_client_socket_fd);
  76. }
  77. /**
  78. * @brief Client::start
  79. * @return A meaningless integer
  80. */
  81. void Client::start() {
  82. if (m_client_socket_fd == -1) {
  83. m_client_socket_fd = socket(AF_INET, SOCK_STREAM, 0);
  84. if (m_client_socket_fd != -1) {
  85. sockaddr_in server_socket;
  86. char* end;
  87. server_socket.sin_family = AF_INET;
  88. auto port_value = strtol(argv[2], &end, 10);
  89. if (port_value < 0 || end == argv[2]) {
  90. return;
  91. }
  92. int socket_option = 1;
  93. // Free up the port to begin listening again
  94. setsockopt(m_client_socket_fd, SOL_SOCKET, SO_REUSEADDR, &socket_option,
  95. sizeof(socket_option));
  96. server_socket.sin_port = htons(port_value);
  97. inet_pton(AF_INET, argv[1], &server_socket.sin_addr.s_addr);
  98. if (::connect(m_client_socket_fd, reinterpret_cast<sockaddr*>(&server_socket),
  99. sizeof(server_socket)) != -1) {
  100. std::string start_operation_string = createOperation("start", {});
  101. // Send operation as an encoded message
  102. sendEncoded(start_operation_string);
  103. // Delegate message handling to its own thread
  104. std::function<void()> message_send_fn = [this]() {
  105. this->handleMessages();
  106. };
  107. MessageHandler message_handler = createMessageHandler(message_send_fn);
  108. // Handle received messages on separate thread
  109. std::thread (message_handler).detach();
  110. } else {
  111. qDebug() << errno;
  112. ::close(m_client_socket_fd);
  113. }
  114. } else {
  115. qDebug() << "Failed to create new connection";
  116. }
  117. } else {
  118. qDebug() << "Connection already in progress";
  119. }
  120. }
  121. /**
  122. * @brief Client::sendMessage
  123. * @param s[in] <const QString&> The message to send
  124. */
  125. void Client::sendMessage(const QString& s) {
  126. if (m_client_socket_fd != -1) {
  127. std::string json_string {"{\"type\":\"custom\", \"message\": \""};
  128. json_string += s.toUtf8().data();
  129. json_string += "\", \"args\":\"placeholder\"}";
  130. // Send custom message as an encoded message
  131. sendEncoded(json_string);
  132. } else {
  133. qDebug() << "You must first open a connection";
  134. }
  135. }
  136. void Client::sendEncoded(std::string message) {
  137. std::vector<uint8_t> fb_byte_vector{message.begin(), message.end()};
  138. auto byte_vector = builder.CreateVector(fb_byte_vector);
  139. auto k_message = CreateMessage(builder, 69, byte_vector);
  140. builder.Finish(k_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 start operation
  158. ::send(m_client_socket_fd, send_buffer, size + 4, 0);
  159. builder.Clear();
  160. }
  161. void Client::closeConnection() {
  162. if (m_client_socket_fd != -1) {
  163. std::string stop_operation_string = createOperation("stop", {});
  164. // Send operation as an encoded message
  165. sendEncoded(stop_operation_string);
  166. // Clean up socket file descriptor
  167. ::shutdown(m_client_socket_fd, SHUT_RDWR);
  168. ::close(m_client_socket_fd);
  169. m_client_socket_fd = -1;
  170. return;
  171. }
  172. qDebug() << "There is no active connection to close";
  173. }
  174. void Client::setSelectedApp(std::vector<QString> app_names) {
  175. selected_commands.clear();
  176. for (const auto& name : app_names) {
  177. qDebug() << "Matching mask to " << name;
  178. for (const auto& command : m_commands) {
  179. if (command.second.c_str() == name.toUtf8()) {
  180. selected_commands.push_back(command.first);
  181. }
  182. }
  183. }
  184. }
  185. void Client::execute() {
  186. if (!selected_commands.empty()) {
  187. executing = true;
  188. for (const auto& command : selected_commands) {
  189. std::string execute_operation = createOperation("Execute", {std::to_string(command)});
  190. sendEncoded(execute_operation);
  191. }
  192. }
  193. }