socket_listener.cpp 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. // Project headers
  2. #include "headers/socket_listener.h"
  3. #include "headers/constants.h"
  4. #include "headers/listen_interface.h"
  5. // System libraries
  6. #include <arpa/inet.h>
  7. #include <netdb.h>
  8. #include <string.h>
  9. #include <sys/socket.h>
  10. #include <sys/types.h>
  11. #include <unistd.h>
  12. // C++ Libraries
  13. #include <atomic>
  14. #include <condition_variable>
  15. #include <functional>
  16. #include <iostream>
  17. #include <memory>
  18. #include <queue>
  19. #include <string>
  20. #include <thread>
  21. #include <vector>
  22. /**
  23. * Constructor
  24. * Initialize with ip_address, port and message_handler
  25. */
  26. SocketListener::SocketListener(std::string ip_address, int port)
  27. : m_ip_address(ip_address), m_port(port), accepting_tasks(true), shutdown_loop(false) {}
  28. /**
  29. * Destructor
  30. * TODO: Determine if we should make buffer a class member
  31. */
  32. SocketListener::~SocketListener() { cleanup(); }
  33. SocketListener::MessageHandler SocketListener::createMessageHandler(
  34. std::function<void()> cb) {
  35. return MessageHandler(cb);
  36. }
  37. /*
  38. **setMessageHandler *@method *Set the function to handle received
  39. messages * /
  40. // SocketListener::setMessageHandler(MessageHandler message_handler) {
  41. // m_message_handler = message_handler;
  42. // }
  43. /**
  44. * sendMessage
  45. * @method
  46. * Send a null-terminated array of characters, supplied as a const char
  47. * pointer, to a client socket described by its file descriptor
  48. */
  49. void SocketListener::sendMessage(int client_socket_fd,
  50. std::shared_ptr<char[]> s_ptr) {
  51. send(client_socket_fd, s_ptr.get(), static_cast<size_t>(MAX_BUFFER_SIZE) + 1,
  52. 0);
  53. }
  54. /**
  55. * init
  56. * TODO: Initialize buffer memory, if buffer is to be a class member
  57. */
  58. bool SocketListener::init() {
  59. std::cout << "Initializing socket listener" << std::endl;
  60. return true;
  61. }
  62. void SocketListener::push_to_queue(std::function<void()> fn) {
  63. std::unique_lock<std::mutex> lock(m_mutex_lock);
  64. task_queue.push(fn);
  65. lock.unlock();
  66. pool_condition.notify_one();
  67. }
  68. void SocketListener::handle_loop() {
  69. std::function<void()> fn;
  70. for (;;) {
  71. {
  72. std::unique_lock<std::mutex> lock(m_mutex_lock);
  73. pool_condition.wait(
  74. lock, [this]() { return !accepting_tasks || !task_queue.empty(); });
  75. if (!accepting_tasks && task_queue.empty()) {
  76. return; // we are done here
  77. }
  78. fn = task_queue.front();
  79. task_queue.pop();
  80. }
  81. fn();
  82. }
  83. }
  84. void SocketListener::loop_check() {
  85. if (!m_loop_switch) {
  86. m_loop_switch = true;
  87. m_loop_thread = std::thread(&SocketListener::handle_loop, this);
  88. }
  89. }
  90. void SocketListener::handle_client_socket(int client_socket_fd,
  91. SocketListener::MessageHandler message_handler,
  92. std::shared_ptr<char[]> buf) {
  93. // char buf[MAX_BUFFER_SIZE] =
  94. // {}; // Declare, define and initialize a character buffer
  95. // std::string buffer_string{}; // Initialize a string buffer
  96. while (true) {
  97. memset(buf.get(), 0, MAX_BUFFER_SIZE); // Zero the character buffer
  98. int bytes_received = 0;
  99. // Receive and write incoming data to buffer and return the number of
  100. // bytes received
  101. bytes_received =
  102. recv(client_socket_fd, buf.get(),
  103. MAX_BUFFER_SIZE - 2, // Leave room for null-termination
  104. 0);
  105. buf.get()[MAX_BUFFER_SIZE - 1] = 0; // Null-terminate the character buffer
  106. if (bytes_received > 0) {
  107. std::cout << "Bytes received: " << bytes_received
  108. << "\nData: " << buf.get() << std::endl;
  109. // Handle incoming message
  110. message_handler();
  111. } else {
  112. std::cout << "client disconnected" << std::endl;
  113. break;
  114. }
  115. }
  116. // Zero the buffer again before closing
  117. memset(buf.get(), 0, MAX_BUFFER_SIZE);
  118. // TODO: Determine if we should free memory, or handle as class member
  119. close(client_socket_fd); // Destroy client socket and deallocate its fd
  120. }
  121. /**
  122. * run
  123. * @method
  124. * Main message loop
  125. * TODO: Implement multithreading
  126. */
  127. void SocketListener::run() {
  128. // Begin listening loop
  129. while (true) {
  130. std::cout << "Begin" << std::endl;
  131. // Call system to open a listening socket, and return its file descriptor
  132. int listening_socket_fd = createSocket();
  133. if (listening_socket_fd == SOCKET_ERROR) {
  134. std::cout << "Socket error: shutting down server" << std::endl;
  135. break;
  136. }
  137. std::cout << "Attempting to wait for connection" << std::endl;
  138. // wait for a client connection and get its socket file descriptor
  139. int client_socket_fd = waitForConnection(listening_socket_fd);
  140. if (client_socket_fd != SOCKET_ERROR) {
  141. // Destroy listening socket and deallocate its file descriptor. Only use
  142. // the client socket now.
  143. close(listening_socket_fd);
  144. std::shared_ptr<char[]> s_ptr(new char[MAX_BUFFER_SIZE]);
  145. std::function<void()> message_send_fn = [this, client_socket_fd,
  146. s_ptr]() {
  147. this->sendMessage(client_socket_fd, s_ptr);
  148. };
  149. MessageHandler message_handler = createMessageHandler(message_send_fn);
  150. std::cout << "Pushing client to queue" << std::endl;
  151. push_to_queue(std::bind(&SocketListener::handle_client_socket, this, client_socket_fd,
  152. message_handler, s_ptr));
  153. loop_check();
  154. std::cout << "At the end" << std::endl;
  155. }
  156. }
  157. }
  158. /**
  159. * cleanUp
  160. * @method
  161. * TODO: Determine if we should be cleaning up buffer memory
  162. */
  163. void SocketListener::cleanup() { std::cout << "Cleaning up" << std::endl;
  164. if (m_loop_thread.joinable()) {
  165. m_loop_thread.join();
  166. }
  167. }
  168. /**
  169. * createSocket
  170. * Open a listening socket and return its file descriptor
  171. */
  172. int SocketListener::createSocket() {
  173. /* Call the system to open a socket passing arguments for
  174. ipv4 family, tcp type and no additional protocol info */
  175. int listening_socket_fd = socket(AF_INET, SOCK_STREAM, 0);
  176. if (listening_socket_fd != SOCKET_ERROR) {
  177. std::cout << "Created listening socket" << std::endl;
  178. // Create socket structure to hold address and type
  179. sockaddr_in socket_struct;
  180. socket_struct.sin_family = AF_INET; // ipv4
  181. socket_struct.sin_port =
  182. htons(m_port); // convert byte order of port value from host to network
  183. inet_pton(AF_INET, m_ip_address.c_str(), // convert address to binary
  184. &socket_struct.sin_addr);
  185. int socket_option = 1;
  186. setsockopt(listening_socket_fd, SOL_SOCKET, SO_REUSEADDR, &socket_option, sizeof(socket_option));
  187. // Bind local socket address to socket file descriptor
  188. int bind_result = bind(
  189. listening_socket_fd, // TODO: Use C++ cast on next line?
  190. (sockaddr *)&socket_struct, // cast socket_struct to more generic type
  191. sizeof(socket_struct));
  192. if (bind_result != SOCKET_ERROR) {
  193. // Listen for connections to socket and allow up to max number of
  194. // connections for queue
  195. int listen_result = listen(listening_socket_fd, SOMAXCONN);
  196. if (listen_result == SOCKET_ERROR) {
  197. return WAIT_SOCKET_FAILURE;
  198. }
  199. } else {
  200. return WAIT_SOCKET_FAILURE;
  201. }
  202. }
  203. return listening_socket_fd; // Return socket file descriptor
  204. }
  205. /**
  206. * waitForConnection
  207. * @method
  208. * Takes first connection on queue of pending connections, creates a new socket
  209. * and returns its file descriptor
  210. */
  211. int SocketListener::waitForConnection(int listening_socket) {
  212. int client_socket_fd = accept(listening_socket, NULL, NULL);
  213. return client_socket_fd;
  214. }