socket_listener.cpp 7.1 KB

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