#include #include #include #include #include #include #include #include "commands.h" #define send(queue_id, msgbuf_ptr) \ msgsnd(queue_id, msgbuf_ptr, sizeof(*(msgbuf_ptr)) - sizeof(long), 0) static void heartbeat_loop(int server_queue_id, const char *client_id) { msgbuf_t msg = {.mtype = Hearbeat, .sender = ""}; strncpy(msg.sender, client_id, COMMAND_LENGTH - 1); while (1) { if (send(server_queue_id, &msg) == -1) { perror("msgsnd(heartbeat) failed"); break; } sleep(HEARTBEAT_INTERVAL); } } static void cleanup_queue(int qid) { if (qid != -1) { msgctl(qid, IPC_RMID, NULL); } } int main(int argc, char *argv[]) { if (argc < 2) { fprintf(stderr, "Usage: %s \n", argv[0]); return 1; } char *client_id = argv[1]; int client_queue_id = msgget(IPC_PRIVATE, IPC_CREAT | 0666); if (client_queue_id == -1) { perror("msgget(client) failed"); return 1; } int server_queue_id = -1; const char *id_path = "/home/piotr/server_queue_id"; int fd = open(id_path, O_RDONLY); if (fd == -1) { perror("Failed to open server queue id file"); cleanup_queue(client_queue_id); return 1; } if (read(fd, &server_queue_id, sizeof(int)) == -1) { fprintf(stderr, "Failed to read server queue id from %s\n", id_path); server_queue_id = -1; } close(fd); msgbuf_t msg = {.mtype = Login, .sender = ""}; strncpy(msg.sender, client_id, COMMAND_LENGTH - 1); snprintf(msg.command, COMMAND_LENGTH, "%d", client_queue_id); if (msgsnd(server_queue_id, &msg, sizeof(msg) - sizeof(long), 0) == -1) { perror("msgsnd(login) failed"); cleanup_queue(client_queue_id); return 1; } /* Wait for login response (server replies with mtype = Login) */ msgbuf_t resp; ssize_t r = msgrcv(client_queue_id, &resp, sizeof(resp) - sizeof(long), Login, 0); if (r == -1) { perror("msgrcv(login response) failed"); cleanup_queue(client_queue_id); return 1; } if (resp.stype != ACK_ACCEPTED) { fprintf(stderr, "Login rejected: code=%d\n", resp.stype); cleanup_queue(client_queue_id); return 1; } printf("Login accepted for id '%s'\n", client_id); if (fork() == 0) { heartbeat_loop(server_queue_id, client_id); return 0; } /* Send a test message to the server */ msgbuf_t test = {.mtype = Message, .sender = ""}; strncpy(test.sender, client_id, COMMAND_LENGTH - 1); snprintf(test.message, MESSAGE_LENGTH, "Hello from %s", client_id); if (msgsnd(server_queue_id, &test, sizeof(test) - sizeof(long), 0) == -1) { printf("DEBUG: test.mtype = %u\n", test.mtype); perror("failed while sending a test message"); cleanup_queue(client_queue_id); return 1; } printf("Sent test message to server: \"%s\"\n", test.message); /* Now wait for server forwarded message(s) and a signal ack. We'll wait until we've seen both a Message and a Signal ack or timeout. */ int seen_message = 0; int seen_ack = 0; const int MAX_ITER = 10; // for (int i = 0; i < MAX_ITER && !(seen_message && seen_ack); ++i) { while (1) { msgbuf_t incoming; ssize_t got = msgrcv(client_queue_id, &incoming, sizeof(incoming) - sizeof(long), 0, 0); perror("Got a message damn"); if (got == -1) { perror("msgrcv(incoming) failed"); if (errno == EINTR) { continue; } perror("msgrcv(incoming) failed"); break; } switch (incoming.mtype) { case Message: printf("Received Message: \"%s\"\n", incoming.message); seen_message = 1; break; case Signal: printf("Received Signal: code=%d\n", incoming.stype); if (incoming.stype == ACK_ACCEPTED || incoming.stype == ACK_DELIVERED) { seen_ack = 1; } break; default: printf("Received unknown mtype=%ld stype=%d\n", (long)incoming.mtype, incoming.stype); break; } } if (!seen_message) { fprintf(stderr, "Did not receive forwarded message from server\n"); } if (!seen_ack) { fprintf(stderr, "Did not receive ack from server\n"); } /* Logout/cleanup: remove our queue */ cleanup_queue(client_queue_id); printf("Client exiting\n"); return 0; }