|
@@ -5,92 +5,104 @@
|
|
|
#include <sys/socket.h>
|
|
|
#include <sys/types.h>
|
|
|
#include <unistd.h>
|
|
|
-
|
|
|
-#include <thread>
|
|
|
+#include <functional>
|
|
|
#include <cstring>
|
|
|
#include <QDebug>
|
|
|
#include <QByteArray>
|
|
|
#include <string>
|
|
|
#include <iostream>
|
|
|
|
|
|
-void input_loop(char buffer[]) {
|
|
|
- qDebug() << "Enter message: ";
|
|
|
- for (;;) {
|
|
|
- std::cin >> buffer;
|
|
|
- if (buffer[0] != 0) {
|
|
|
- return;
|
|
|
- }
|
|
|
- }
|
|
|
+/**
|
|
|
+ * @brief Client::createMessageHandler
|
|
|
+ * @param cb
|
|
|
+ * @return
|
|
|
+ */
|
|
|
+Client::MessageHandler Client::createMessageHandler(
|
|
|
+ std::function<void()> cb) {
|
|
|
+ return MessageHandler(cb);
|
|
|
}
|
|
|
-//hostCombo(new QComboBox),
|
|
|
-// portLineEdit(new QLineEdit),
|
|
|
-// getFortuneButton(new QPushButton(tr("Get Fortune"))),
|
|
|
-// tcpSocket(new QTcpSocket(this)) {
|
|
|
-// setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint);
|
|
|
-
|
|
|
-// in.setDevice(tcpSocket);
|
|
|
-// in.setVersion(QDataStream::Qt_4_0);
|
|
|
-
|
|
|
-// QPushButton *button = new QPushButton("Push");
|
|
|
-// parent->layout()->addWidget(button);
|
|
|
-// connect(button, &QPushButton::clicked, this, &Client::clicked);
|
|
|
-// connect(tcpSocket, &QIODevice::readyRead, this, &Client::readFortune);
|
|
|
-//}
|
|
|
-
|
|
|
|
|
|
+/**
|
|
|
+ * @brief Client::Client
|
|
|
+ * @constructor
|
|
|
+ * @param parent
|
|
|
+ * @param count
|
|
|
+ * @param arguments
|
|
|
+ */
|
|
|
Client::Client(QWidget *parent, int count, char** arguments) : QDialog(parent), argc(count), argv(arguments), m_client_socket_fd(-1) {}
|
|
|
|
|
|
- void Client::handleMessages() {
|
|
|
- char receive_buffer[2048];
|
|
|
- for (;;) {
|
|
|
- memset(receive_buffer, 0, 2048);
|
|
|
- int bytes_received = 0;
|
|
|
- bytes_received = recv(m_client_socket_fd, receive_buffer, 2048 - 2, 0);
|
|
|
- receive_buffer[2047] = 0;
|
|
|
- if (bytes_received == 0) {
|
|
|
- break;
|
|
|
- }
|
|
|
- emit Client::messageReceived(QString::fromUtf8(receive_buffer, 2048));
|
|
|
- }
|
|
|
+/**
|
|
|
+ * @brief Client::~Client
|
|
|
+ * @destructor
|
|
|
+ */
|
|
|
+Client::~Client() {
|
|
|
+ closeConnection();
|
|
|
+}
|
|
|
+
|
|
|
+ /**
|
|
|
+ * @brief Client::handleMessages
|
|
|
+ */
|
|
|
+void Client::handleMessages() {
|
|
|
+ char receive_buffer[2048];
|
|
|
+ for (;;) {
|
|
|
memset(receive_buffer, 0, 2048);
|
|
|
- ::close(m_client_socket_fd);
|
|
|
+ int bytes_received = 0;
|
|
|
+ bytes_received = recv(m_client_socket_fd, receive_buffer, 2048 - 2, 0);
|
|
|
+ receive_buffer[2047] = 0;
|
|
|
+ if (bytes_received == 0) {
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ emit Client::messageReceived(QString::fromUtf8(receive_buffer, 2048));
|
|
|
}
|
|
|
+ memset(receive_buffer, 0, 2048);
|
|
|
+ ::close(m_client_socket_fd);
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+ * @brief Client::start
|
|
|
+ * @return A meaningless integer
|
|
|
+ */
|
|
|
+void Client::start() {
|
|
|
+ m_client_socket_fd = socket(AF_INET, SOCK_STREAM, 0);
|
|
|
+ if (m_client_socket_fd != -1) {
|
|
|
+ sockaddr_in server_socket;
|
|
|
+ char* end;
|
|
|
+ server_socket.sin_family = AF_INET;
|
|
|
+ auto port_value = strtol(argv[2], &end, 10);
|
|
|
+ if (port_value < 0 || end == argv[2]) {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ int socket_option = 1;
|
|
|
+ // Free up the port to begin listening again
|
|
|
+ setsockopt(m_client_socket_fd, SOL_SOCKET, SO_REUSEADDR, &socket_option,
|
|
|
+ sizeof(socket_option));
|
|
|
|
|
|
- int Client::start() {
|
|
|
- m_client_socket_fd = socket(AF_INET, SOCK_STREAM, 0);
|
|
|
- if (m_client_socket_fd != -1) {
|
|
|
- sockaddr_in server_socket;
|
|
|
- char* end;
|
|
|
- server_socket.sin_family = AF_INET;
|
|
|
- auto port_value = strtol(argv[2], &end, 10);
|
|
|
- if (port_value < 0 || end == argv[2]) {
|
|
|
- return 1;
|
|
|
- }
|
|
|
+ server_socket.sin_port = htons(port_value);
|
|
|
+ inet_pton(AF_INET, argv[1], &server_socket.sin_addr.s_addr);
|
|
|
|
|
|
- int socket_option = 1;
|
|
|
- // Free up the port to begin listening again
|
|
|
- setsockopt(m_client_socket_fd, SOL_SOCKET, SO_REUSEADDR, &socket_option,
|
|
|
- sizeof(socket_option));
|
|
|
+ if (::connect(m_client_socket_fd, (sockaddr*)&server_socket,
|
|
|
+ sizeof(server_socket)) != -1) {
|
|
|
|
|
|
- server_socket.sin_port = htons(port_value);
|
|
|
- inet_pton(AF_INET, argv[1], &server_socket.sin_addr.s_addr);
|
|
|
+ std::function<void()> message_send_fn = [this]() {
|
|
|
+ this->handleMessages();
|
|
|
+ };
|
|
|
+ MessageHandler message_handler = createMessageHandler(message_send_fn);
|
|
|
+ // Handle received messages on separate thread
|
|
|
+ std::thread m_message_handling_thread(message_handler);
|
|
|
+ m_message_handling_thread.detach();
|
|
|
|
|
|
- if (::connect(m_client_socket_fd, (sockaddr*)&server_socket,
|
|
|
- sizeof(server_socket)) != -1) {
|
|
|
-// std::thread t(handleMessages);
|
|
|
- } else {
|
|
|
- qDebug() << errno;
|
|
|
- ::close(m_client_socket_fd);
|
|
|
- }
|
|
|
- return 0;
|
|
|
+ } else {
|
|
|
+ qDebug() << errno;
|
|
|
+ ::close(m_client_socket_fd);
|
|
|
}
|
|
|
}
|
|
|
-
|
|
|
-void Client::requestNewFortune() {
|
|
|
- QMessageBox::information(this, tr("Jigga Client"),
|
|
|
- tr("Gibs me dat fortune, punk!"));
|
|
|
}
|
|
|
|
|
|
+/**
|
|
|
+ * @brief Client::sendMessage
|
|
|
+ * @param s[in] <const QString&> The message to send
|
|
|
+ */
|
|
|
void Client::sendMessage(const QString& s) {
|
|
|
if (m_client_socket_fd != -1) {
|
|
|
QByteArray byte_array = s.toLocal8Bit();
|
|
@@ -99,3 +111,9 @@ void Client::sendMessage(const QString& s) {
|
|
|
qDebug() << "You must first open a connection";
|
|
|
}
|
|
|
}
|
|
|
+
|
|
|
+void Client::closeConnection() {
|
|
|
+ if (m_client_socket_fd != -1) {
|
|
|
+ ::close(m_client_socket_fd);
|
|
|
+ }
|
|
|
+}
|