-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathchat.c
315 lines (277 loc) · 9.48 KB
/
chat.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
#include <pthread.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdbool.h>
#include <unistd.h>
#include <sys/socket.h>
#include <netdb.h>
#include <errno.h>
#include "chat.h"
#include "storage.h"
#include "server.h"
#include "display.h"
#include "terminal.h"
#include "main.h"
static pthread_mutex_t lock;
static struct message *first_message;
static struct message *last_message;
static char *chat_name;
static size_t message_length;
static char *username;
static int socket_descriptor;
static pthread_t listen_thread;
void parse_chat_log(char *buffer);
void parse_server_response(char **response);
void *listen_server(void *arg);
void append_message(char *username, char *content);
/**
* Initializes:
* - mutex
* - heap memory for file global variables
*/
static void initialize(){
if(pthread_mutex_init(&lock, NULL) != 0){
printf("Failed to create chat lock. Exiting...\n");
exit(1);
}
chat_name = calloc(MAX_LENGTH_CHAT_NAME, sizeof(char));
username = calloc(MAX_LENGTH_USERNAME, sizeof(char));
}
static void handle_socket_failure(char *details){
printf("Failed to connect to server: %s\n", details);
printf("Error %d: %s\n", errno, strerror(errno));
if(errno == 111){
// "Connection refused" error
printf("You are connecting to a nonexisting server. Make sure there is a server running\n");
}
exit(1);
}
/**
* Connect to the server hosting the chat.
* @param ip4_address IP address of the server.
*/
static void initialize_server_connection(char *ipv4_address){
socket_descriptor = socket(AF_INET, SOCK_STREAM, 0 );
if(socket_descriptor < 0){
handle_socket_failure("Failed to create socket\n");
}
struct addrinfo * hints, * results;
hints = (struct addrinfo *)calloc(1, sizeof(struct addrinfo));
results = calloc(1, sizeof(struct addrinfo));
hints->ai_family = AF_INET; //IPv4
hints->ai_socktype = SOCK_STREAM; //TCP socket
getaddrinfo(ipv4_address, PORT, hints, &results);
int state = connect(socket_descriptor, results->ai_addr, results->ai_addrlen);
if(state < 0){
free(hints);
freeaddrinfo(results);
handle_socket_failure("Failed to connect to server");
}
free(hints);
freeaddrinfo(results);
// initialize listening thread
pthread_create(&listen_thread, NULL, listen_server, NULL);
}
void initialize_new_chat(char *given_chat_name, char *given_username){
initialize();
strncpy(chat_name, given_chat_name, MAX_LENGTH_CHAT_NAME);
strncpy(username, given_username, MAX_LENGTH_USERNAME);
// IP of localhost since the one creating the chat is the one hosting it
initialize_server_connection("127.0.0.1");
}
void initialize_disk_chat(char *given_chat_name){
initialize();
strncpy(chat_name, given_chat_name, MAX_LENGTH_CHAT_NAME);
// read the file
int fd;
off_t size;
chat_file_descriptor(&fd, &size);
char buff[size];
read(fd, buff, size);
buff[size - 1] = '\0'; // replace the ending end of line character with a end of string character
// Get the username
username = calloc(MAX_LENGTH_USERNAME, sizeof(char));
int end_of_username_index = strchr(buff, '\n') - buff;
strncpy(username, buff, end_of_username_index);
parse_chat_log(buff + end_of_username_index + 1); // message content starts after end of line character from username))
initialize_server_connection("127.0.0.1");
}
void initialize_join_chat(char *given_username, char *ipv4_address){
initialize();
strncpy(username, given_username, MAX_LENGTH_USERNAME);
initialize_server_connection(ipv4_address);
// Server name and current chat log are received from the server.
}
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wmissing-noreturn"
void *listen_server(void *arg){
while(true){
char *buffer = force_read_message(socket_descriptor);
pthread_mutex_lock(&lock);
parse_server_response(&buffer);
free(buffer);
pthread_mutex_unlock(&lock);
display();
}
}
#pragma clang diagnostic pop
void send_message(struct message *new_message){
strcpy(new_message->username, username);
// send message to the server
char buffer[MESSAGE_SIZE] = {'\0'};
strcat(buffer, MESSAGE);
strcat(buffer, "\n");
strcat(buffer, username);
strcat(buffer, "\n");
strcat(buffer, new_message->content);
write(socket_descriptor, buffer, MESSAGE_SIZE);
}
void leave_connection(){
char buffer[MESSAGE_SIZE] = {'\0'};
strcat(buffer, LEAVE);
strcat(buffer, "\n");
strcat(buffer, username);
write(socket_descriptor, buffer, MESSAGE_SIZE);
}
void append_message(char *username_string, char *content){
struct message *new_message = calloc(1, sizeof(struct message));
strcpy(new_message->username, username_string);
strcpy(new_message->content, content);
new_message->next = NULL;
if(first_message == NULL){
// First message of the chat
first_message = new_message;
last_message = new_message;
}else{
// Not first message of chat, so it goes to end
last_message->next = new_message;
new_message->previous = last_message;
last_message = new_message;
}
message_length++;
}
/**
* Clears the chat locally. Used only when the user is exiting the chat (exiting the program)
*/
void clear_chat(){
pthread_mutex_lock(&lock);
message_length = 0;
struct message *current = first_message;
while(current != NULL){
struct message *next = current->next;
free(current);
current = next;
}
free(username);
free(chat_name);
close(socket_descriptor);
pthread_mutex_unlock(&lock);
}
/**
* Gain access to the entire chat log. Since the thread calling this now has ownership of the thread, it must also call
* release_message_lock when done working with the chat log. Accessing the chat log should be read only.
* @param first_message_buff
* @param last_message_buff
* @param message_length_buff
*/
void get_message_lock(struct message **first_message_buff, struct message **last_message_buff, size_t *message_length_buff){
pthread_mutex_lock(&lock);
*first_message_buff = first_message;
*last_message_buff = last_message;
*message_length_buff = message_length;
}
/**
* When you are done reading the chat log from get_message_lock, you must call this to relese the lock so other threads
* can access it.
*/
void release_message_lock(){
pthread_mutex_unlock(&lock);
}
/**
* Gets the number of messages sent.
* @return number of messages sent.
*/
size_t get_message_length(){
pthread_mutex_lock(&lock);
size_t copy = message_length;
pthread_mutex_unlock(&lock);
return copy;
}
char *get_chat_name(){
return chat_name;
}
char *get_username(){
return username;
}
/**
* Reads the content of a string containing the entire chat log into memory. See stringify_chat_log() to convert chat
* to a string.
*/
void parse_chat_log(char *buffer){
char **copy = &buffer;
char *username_buff;
while((username_buff = strsep(copy, "\n")) != NULL){
char *content = strsep(copy, "\n");
append_message(username_buff, content);
}
}
void parse_server_response(char **response){
char *command = strsep(response, "\n");
if(strcmp(command, MESSAGE) == 0){
// is a message
char *username_string = strsep(response, "\n");
char *content = strsep(response, "\n");
append_message(username_string, content);
}else if(strcmp(command, LEAVE) == 0){
// Someone left the chat
char system_message[MAX_LENGTH_USERNAME + 10]; // 10 for space for text " left."
char *username_who_left = strsep(response, "\n");
strcat(system_message, username_who_left);
strcat(system_message, " left.");
append_message("System", system_message);
}else if(strcmp(command, EXIT) == 0){
// Host left, so all client need to leave
if(!is_user_host()){
disable_raw_mode();
clear_terminal();
printf("Host left. Clients (you) must leave. A copy of the chat is not saved for clients. Bye!\n");
}
exit(0);
}else if(strcmp(command, FULL) == 0){
// Too many people connected to server, cannot join chat
disable_raw_mode();
clear_terminal();
printf("Too many people connected to the server. Failed to connect.\n");
exit(0);
}
}
/**
* Converts the entire chat log into a formatted string. To convert back to memory, use parse_chat_log(char *chat_log)
* @return String representing the entire chat log.
*/
char *stringify_chat_log(){
// Calculate size of string that will contain the chat log
size_t size = 1; // Start at 1 to reserve space for end of string character
// Store the current user's username
size += MAX_LENGTH_USERNAME + 1; // add one for new line character
struct message *current = first_message;
while(current != NULL){
size += strlen(current->username);
size += strlen(current->content);
size += 2; // Space for new line character delimiting the username, message, MessageType length
current = current->next;
}
char *string = calloc(size, sizeof(char));
strcat(string, username);
strcat(string, "\n");
current = first_message;
while(current != NULL){
strcat(string, current->username);
strcat(string, "\n");
strcat(string, current->content);
strcat(string, "\n");
current = current->next;
}
return string;
}