WHY2/src/chat/misc.c

1042 lines
32 KiB
C

/*
This is part of WHY2
Copyright (C) 2022 Václav Šmejkal
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
#include <why2/chat/misc.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <ctype.h>
#include <sys/socket.h>
#include <sys/ioctl.h>
#include <unistd.h>
#include <pthread.h>
#include <json-c/json.h>
#include <why2/chat/config.h>
#include <why2/chat/flags.h>
#include <why2/llist.h>
#include <why2/memory.h>
#include <why2/misc.h>
pthread_t getline_thread;
//LINKED LIST STUFF
typedef struct _connection_node
{
int connection;
pthread_t thread;
char *username;
unsigned long id;
} connection_node_t; //SINGLE LINKED LIST
why2_list_t connection_list = WHY2_LIST_EMPTY;
why2_list_t waiting_list = WHY2_LIST_EMPTY;
char *get_string_from_json(struct json_object *json, char *string)
{
struct json_object *object;
json_object_object_get_ex(json, string, &object);
return (char*) json_object_get_string(object);
}
char *get_string_from_json_string(char *json, char *string)
{
struct json_object *json_obj = json_tokener_parse(json);
if (json_obj == NULL) return NULL; //INVALID SYNTAX WAS SENT BY SOME FUCKING SCRIPT KIDDIE
char *returning = why2_strdup(get_string_from_json(json_obj, string));
//DEALLOCATION
json_object_put(json_obj);
//GET STRINGS
return returning;
}
int get_int_from_json_string(char *json, char *string)
{
char *value = get_string_from_json_string(json, string); //GET VALUE
int returning = atoi(value); //PARSE
why2_deallocate(value);
return returning;
}
int server_config_int(char *key)
{
char *value_str = why2_chat_server_config(key); //GET STRING
int returning = atoi(value_str); //PARSE INT
why2_toml_read_free(value_str);
return returning;
}
void send_to_all(char *json)
{
why2_node_t _first_node = (why2_node_t) { NULL, connection_list.head };
why2_node_t *node_buffer = &_first_node;
connection_node_t connection_buffer;
//PARSE
struct json_object *json_obj = json_tokener_parse(json);
char *message = get_string_from_json(json_obj, "message");
char *username = get_string_from_json(json_obj, "username");
if (json_obj == NULL) return; //EXIT IF INVALID SYNTAX WAS SENT
while (node_buffer -> next != NULL) //SEND TO ALL CONNECTIONS
{
node_buffer = node_buffer -> next;
connection_buffer = *(connection_node_t*) node_buffer -> value;
why2_send_socket(message, username, connection_buffer.connection); //SEND TO CLIENT
}
//DEALLOCATION
json_object_put(json_obj);
}
void append(char **destination, char *key, char *value)
{
char *output = why2_calloc(strlen(*destination) + strlen(key) + strlen(value) + 7, sizeof(char));
sprintf(output, "%s%s\"%s\":\"%s\"", *destination, strcmp(*destination, "") == 0 ? "" : ",", key, value);
why2_deallocate(*destination);
*destination = output;
}
void add_brackets(char **json)
{
char *output = why2_calloc(strlen(*json) + 4, sizeof(char));
sprintf(output, "{%s}%c", *json, '\0');
why2_deallocate(*json);
*json = output;
}
char *read_socket_raw(int socket)
{
if (socket == -1)
{
fprintf(stderr, "Reading socket failed.\n");
return NULL;
}
char *content_buffer = NULL;
int content_size;
why2_bool empty_buffer = 0; //WHETHER MSG_PEEK SHOULD BE USER OR NOT
do
{
//FIND THE SENT SIZE
content_size = 0;
if (ioctl(socket, FIONREAD, &content_size) < 0 || content_size <= 0) continue;
//ALLOCATE
content_buffer = why2_realloc(content_buffer, content_size + 1);
read_section:
//READ JSON MESSAGE
if (recv(socket, content_buffer, content_size, !empty_buffer ? MSG_PEEK : 0) != content_size) //READ THE MESSAGE BY CHARACTERS
{
fprintf(stderr, "Socket probably read wrongly!\n");
}
if (empty_buffer) goto return_section; //STOP LOOPING
} while (content_buffer == NULL || strncmp(content_buffer + (content_size - 2), "\"}", 2) != 0);
//REMOVE JUNK FROM BUFFER (CUZ THE MSG_PEEK FLAG)
empty_buffer = 1;
goto read_section; //TODO: remove the stupid goto
return_section:
content_buffer[content_size] = '\0';
return content_buffer;
}
void *read_socket_raw_thread(void *socket)
{
return read_socket_raw(*(int*) socket);
}
void remove_json_syntax_characters(char *text)
{
for (size_t i = 0; i < strlen(text); i++) //TODO: DO SOMETHING MORE
{
if (text[i] == '\"')
{
text[i] = '\'';
}
}
}
void lowercase(char *string)
{
for (unsigned long i = 0; i < strlen(string); i++)
{
string[i] = tolower(string[i]);
}
}
why2_bool username_equal(char *u1, char *u2)
{
char *one = why2_strdup(u1);
char *two = why2_strdup(u2);
//LOWERCASE
lowercase(one);
lowercase(two);
return strcmp(one, two) == 0;
}
why2_bool username_equal_deallocate(char *u1, char *u2) //SAME THING AS username_equal BUT IT DEALLOCATES u2
{
why2_bool returning = username_equal(u1, u2);
why2_toml_read_free(u2);
return returning;
}
why2_bool check_username(char *username)
{
if (username == NULL) return 0;
if (username_equal_deallocate(username, why2_chat_server_config("server_username"))) return 0; //DISABLE 'server' USERNAME
if (username_equal_deallocate(username, why2_chat_server_config("default_username"))) return 0; //DISABLE 'anon' USERNAME DUE TO ONE USERNAME PER SERVER
for (unsigned long i = 0; i < strlen(username); i++)
{
if (!((username[i] >= 48 && username[i] <= 57) ||
(username[i] >= 65 && username[i] <= 90) || //CHECK ONLY FOR a-Z & 0-9
(username[i] >= 97 && username[i] <= 122)))
{
return 0;
}
}
return 1;
}
void *stop_oldest_thread(void *id)
{
sleep(server_config_int("communication_time")); //yk wait
if (waiting_list.head == NULL) return NULL;
if (**(pthread_t**) waiting_list.head -> value == *(pthread_t*) id) //THREAD IS STILL ALIVE, I HOPE
{
//KILL THE THREAD
pthread_cancel(*(pthread_t*) id);
why2_list_remove(&waiting_list, why2_list_find(&waiting_list, id));
}
return NULL;
}
why2_node_t *find_request(void *thread) //COPIED FROM why2_list_find; using double pointers
{
why2_node_t *head = waiting_list.head;
if (head == NULL) return NULL; //EMPTY LIST
why2_node_t *buffer = head;
while (buffer -> next != NULL)
{
if (*(void**) buffer -> value == thread) return buffer;
buffer = buffer -> next;
}
if (thread != *(void**) buffer -> value) buffer = NULL; //PREVENT FROM RETURNING INVALID NODE
return buffer;
}
why2_node_t *find_connection(int connection)
{
why2_node_t *head = connection_list.head;
if (head == NULL) return NULL; //EMPTY LIST
why2_node_t *buffer = head;
while (buffer -> next != NULL)
{
if ((*(connection_node_t*) buffer -> value).connection == connection) return buffer;
buffer = buffer -> next;
}
if (connection != (*(connection_node_t*) buffer -> value).connection) buffer = NULL; //PREVENT FROM RETURNING INVALID NODE
return buffer;
}
why2_node_t *find_connection_by_id(unsigned long id)
{
why2_node_t *head = connection_list.head;
if (head == NULL) return NULL; //EMPTY LIST
why2_node_t *buffer = head;
while (buffer -> next != NULL)
{
if ((*(connection_node_t*) buffer -> value).id == id) return buffer;
buffer = buffer -> next;
}
if (id != (*(connection_node_t*) buffer -> value).id) buffer = NULL; //PREVENT FROM RETURNING INVALID NODE
return buffer;
}
char *read_user(int connection, void **raw_ptr)
{
//VARIABLES
void *buffer;
pthread_t thread_buffer;
pthread_t thread_deletion_buffer;
//RESET VARIABLES
*raw_ptr = NULL;
buffer = &thread_buffer;
//READ
pthread_create(&thread_buffer, NULL, read_socket_raw_thread, &connection);
why2_list_push(&waiting_list, &buffer, sizeof(buffer));
//RUN DELETION THREAD
pthread_create(&thread_deletion_buffer, NULL, stop_oldest_thread, &thread_buffer);
//WAIT FOR MESSAGE
pthread_join(thread_buffer, raw_ptr);
why2_list_remove(&waiting_list, find_request(&thread_buffer));
pthread_cancel(thread_deletion_buffer);
if (*raw_ptr == WHY2_INVALID_POINTER || *raw_ptr == NULL) return NULL; //QUIT COMMUNICATION IF INVALID PACKET WAS RECEIVED
return (char*) *raw_ptr;
}
char *get_username(int connection)
{
why2_node_t *node = find_connection(connection);
connection_node_t c_node = *(connection_node_t*) node -> value;
return c_node.username;
}
void send_socket_deallocate(char *text, char *username, int socket) //SAME AS why2_send_socket BUT IT DEALLOCATES username
{
why2_send_socket(text, username, socket);
why2_toml_read_free(username);
}
void send_socket(char *text, char *username, int socket, why2_bool welcome)
{
char *output = why2_strdup("");
size_t length_buffer = strlen(text);
char *text_copy = why2_calloc(length_buffer + 2, sizeof(char));
struct json_object *json = json_tokener_parse("{}");
//COPY text INTO text_copy (WITHOUT LAST CHARACTER WHEN NEWLINE IS AT THE END)
if (text[length_buffer - 1] == '\n') length_buffer--;
strncpy(text_copy, text, length_buffer);
//UNFUCK QUOTES FROM text_copy
remove_json_syntax_characters(text_copy);
//REMOVE NON ASCII CHARS
int j = 0;
for (int i = 0; text_copy[i] != '\0'; i++)
{
if ((20 <= text_copy[i] && text_copy[i] <= 126)) text_copy[i] = text_copy[j++] = text_copy[i];
}
text_copy[j] = '\0';
//ADD OBJECTS
json_object_object_add(json, "message", json_object_new_string(text_copy));
if (username != NULL) json_object_object_add(json, "username", json_object_new_string(username)); //WAS SENT FROM SERVER
if (welcome) //SENDING WELCOME MESSAGE TO USER
{
//GET FROM CONFIG
char *max_uname = why2_chat_server_config("max_username_length");
char *min_uname = why2_chat_server_config("min_username_length");
char *max_tries = why2_chat_server_config("max_username_tries");
//ADD THE INFO OBJS
json_object_object_add(json, "max_uname", json_object_new_string(max_uname));
json_object_object_add(json, "min_uname", json_object_new_string(min_uname));
json_object_object_add(json, "max_tries", json_object_new_string(max_tries));
//DEALLOCATION
why2_toml_read_free(max_uname);
why2_toml_read_free(min_uname);
why2_toml_read_free(max_tries);
}
//GENERATE JSON STRING
json_object_object_foreach(json, key, value)
{
append(&output, key, (char*) json_object_get_string(value));
}
add_brackets(&output);
why2_deallocate(text_copy);
json_object_put(json);
//SEND
send(socket, output, strlen(output), 0);
//DEALLOCATION
why2_deallocate(output);
}
void send_welcome_socket_deallocate(char *text, char *username, int socket) //SAME AS why2_send_socket BUT IT DEALLOCATES username
{
send_socket(text, username, socket, 1);
why2_toml_read_free(username);
}
void send_welcome_packet(int connection)
{
send_welcome_socket_deallocate(WHY2_CHAT_CODE_ACCEPT_MESSAGES, why2_chat_server_config("server_username"), connection);
}
unsigned long get_latest_id()
{
unsigned long returning = 1;
why2_node_t *buffer = connection_list.head;
connection_node_t value_buffer;
while (buffer != NULL)
{
value_buffer = *(connection_node_t*) buffer -> value;
if (value_buffer.id == returning)
{
returning++;
buffer = connection_list.head;
} else
{
buffer = buffer -> next;
}
}
return returning;
}
//GLOBAL
void why2_send_socket(char *text, char *username, int socket)
{
send_socket(text, username, socket, 0);
}
void *why2_communicate_thread(void *arg)
{
int connection = *(int*) arg;
printf("User connected.\t\t%d\n", connection);
send_welcome_packet(connection); //TELL USER ALL THE INFO THEY NEED
//GET USERNAME
char *config_username = why2_chat_server_config("user_pick_username");
char *raw = NULL;
void *raw_ptr = NULL;
char *raw_output = NULL;
why2_bool force_exiting = 0;
why2_bool invalid_username = 1;
why2_bool exiting = 0;
char *decoded_buffer = NULL;
char *username;
int usernames_n = 0;
struct json_object *json = json_tokener_parse("{}");
//GET DEFAULT USERNAME
char *default_username = why2_chat_server_config("default_username");
username = why2_strdup(default_username);
why2_toml_read_free(default_username);
if (config_username == NULL || strcmp(config_username, "true") == 0)
{
if (config_username == NULL) fprintf(stderr, "Your config doesn't contain 'user_pick_username'. Please update your configuration.\n");
send_socket_deallocate(WHY2_CHAT_CODE_PICK_USERNAME, why2_chat_server_config("server_username"), connection); //ASK USER FOR USERNAME
while (invalid_username)
{
why2_deallocate(username);
if (usernames_n++ == server_config_int("max_username_tries")) //ASKED CLIENT WAY TOO FUCKING MANY TIMES FOR USERNAME, KICK THEM
{
exiting = 1;
goto deallocation;
}
if ((raw = read_user(connection, &raw_ptr)) == NULL) //READ
{
force_exiting = 1; //FAILURE
goto deallocation;
}
decoded_buffer = get_string_from_json_string(raw, "message"); //DECODE
invalid_username = decoded_buffer == NULL || (strlen(decoded_buffer) > (unsigned long) server_config_int("max_username_length")) || (strlen(decoded_buffer) < (unsigned long) server_config_int("min_username_length")) || (!check_username(decoded_buffer)); //CHECK FOR USERNAMES LONGER THAN 20 CHARACTERS
username = decoded_buffer;
if (!invalid_username)
{
//GO TROUGH LIST AND CHECK FOR DUPLICITY
why2_node_t *buffer = connection_list.head;
while (buffer != NULL)
{
//GET USERNAME
connection_node_t *co_node = (connection_node_t*) buffer -> value;
if (strcmp(co_node -> username, decoded_buffer) == 0) //COMPARE
{
invalid_username = 1;
break;
}
buffer = buffer -> next; //NEXT
}
}
//DEALLOCATE STUFF HERE
why2_deallocate(raw);
why2_deallocate(raw_ptr);
if (invalid_username)
{
send_socket_deallocate(WHY2_CHAT_CODE_INVALID_USERNAME, why2_chat_server_config("server_username"), connection); //TELL THE USER THEY ARE DUMB AS FUCK
continue;
}
printf("User set username.\t%d\t%s\n", connection, decoded_buffer);
}
}
why2_toml_read_free(config_username);
unsigned long connection_id = get_latest_id();
connection_node_t node = (connection_node_t)
{
connection,
pthread_self(),
why2_strdup(username),
connection_id
};
why2_list_push(&connection_list, &node, sizeof(node)); //ADD TO LIST
raw = why2_strdup("");
char *connection_message = why2_malloc(strlen(username) + 11);
char *server_username = why2_chat_server_config("server_username");
//BUILD connection_message
sprintf(connection_message, "%s connected", username);
//SEND CONNECTION MESSAGE
json_object_object_add(json, "message", json_object_new_string(connection_message));
json_object_object_add(json, "username", json_object_new_string(server_username));
json_object_object_foreach(json, key, value) //GENERATE JSON STRING
{
append(&raw, key, (char*) json_object_get_string(value));
}
add_brackets(&raw);
json_object_put(json);
send_to_all(raw); //FUCKING SEND TO ALL YOU TWAT
why2_deallocate(raw);
why2_deallocate(connection_message);
why2_deallocate(username);
why2_toml_read_free(server_username);
while (!(exiting || force_exiting)) //KEEP COMMUNICATION ALIVE FOR 5 MINUTES [RESET TIMER AT MESSAGE SENT]
{
if ((raw = read_user(connection, &raw_ptr)) == NULL) break; //READ
json = json_tokener_parse("{}");
raw_output = why2_strdup("");
//REMOVE CONTROL CHARACTERS FROM raw
for (size_t i = 0; i < strlen(raw); i++)
{
if (raw[i] == '\\') raw[i] = '/';
}
decoded_buffer = get_string_from_json_string(raw, "message"); //DECODE
//TRIM MESSAGE
why2_trim_string(&decoded_buffer);
if (decoded_buffer != NULL && strlen(decoded_buffer) != 0)
{
if (strncmp(decoded_buffer, "code", 4) == 0) //CODES FROM CLIENT
{
if (strcmp(decoded_buffer, WHY2_CHAT_CODE_EXIT) == 0) //USER REQUESTED EXIT
{
exiting = 1;
} else if (strcmp(decoded_buffer, WHY2_CHAT_CODE_LIST) == 0) //USER REQUESTED LIST OF USERS
{
why2_node_t *head = connection_list.head;
if (head == NULL) goto deallocation; //TODO: Send no users code
//BUFFERS
why2_node_t *buffer = head;
connection_node_t value_buffer;
unsigned long alloc_size = 0;
char *append_buffer;
//COUNT REQUIRED MESSAGE ALLOCATION SIZE
do
{
value_buffer = *(connection_node_t*) buffer -> value;
alloc_size += strlen(value_buffer.username) + why2_count_int_length(value_buffer.id) + 2;
buffer = buffer -> next; //ITER
} while (buffer != NULL);
char *message = why2_calloc(strlen(WHY2_CHAT_CODE_LIST_SERVER) + alloc_size + 2, sizeof(char));
buffer = head; //RESET
sprintf(message, WHY2_CHAT_CODE_LIST_SERVER); //SET CODE
//FILL THE message
do
{
value_buffer = *(connection_node_t*) buffer -> value;
append_buffer = why2_malloc(strlen(value_buffer.username) + why2_count_int_length(value_buffer.id) + 3); //ALLOCATE THE BUFFER
sprintf(append_buffer, ";%s;%ld", value_buffer.username, value_buffer.id); //FILL THE BUFFER
strcat(message, append_buffer); //JOIN THE BUFFER TO OUTPUT message
why2_deallocate(append_buffer); //DEALLOCATION
buffer = buffer -> next; //ITER
} while (buffer != NULL);
strcat(message, ";");
//SEND
send_socket_deallocate(message, why2_chat_server_config("server_username"), connection);
//DEALLOCATION
why2_deallocate(message);
} else if (strcmp(decoded_buffer, WHY2_CHAT_CODE_VERSION) == 0)
{
//GET VERSION STRING FROM THE VERSIONS JSON
char *message = why2_malloc(strlen(WHY2_CHAT_CODE_VERSION_SERVER) + strlen(WHY2_VERSION) + 2); //ALLOCATE MESSAGE FOR CLIENT
sprintf(message, WHY2_CHAT_CODE_VERSION_SERVER ";%s%c", WHY2_VERSION, '\0'); //CREATE THE MESSAGE
//SEND
send_socket_deallocate(message, why2_chat_server_config("server_username"), connection);
//DEALLOCATION
why2_deallocate(message);
} else if (strncmp(decoded_buffer, WHY2_CHAT_CODE_PM, strlen(WHY2_CHAT_CODE_PM)) == 0) //PM
{
char *input = decoded_buffer + strlen(WHY2_CHAT_CODE_PM) + 1;
char *id = NULL; //RECEIVER
WHY2_UNUSED char *msg;
//CHECK ARGS VALIDITY
why2_bool valid_param = strlen(input) >= 3;
if (valid_param)
{
valid_param = 0;
for (unsigned long i = 1; i < strlen(input); i++)
{
if (input[i] == ';')
{
valid_param = 1;
//EXTRACT FIRST ARG (ID)
id = why2_malloc(i + 1);
for (unsigned long j = 0; j < i; j++)
{
id[j] = input[j];
}
id[i] = '\0';
//EXTRACT MESSAGE
msg = input + i + 1;
break;
}
}
}
//FIND CONNECTION
why2_node_t *pm_connection = id == NULL ? NULL : find_connection_by_id(atoi(id));
if (pm_connection == NULL) valid_param = 0;
if (valid_param) //IGNORE INVALID ARGS
{
//ALLOCATE MESSAGE TO SEND TO RECEIVER
char *private_msg = why2_malloc(strlen(WHY2_CHAT_CODE_PM_SERVER) + why2_count_int_length((int) connection_id) + strlen(msg) + 4);
//CONSTRUCT DA MESSAGE
sprintf(private_msg, WHY2_CHAT_CODE_PM_SERVER ";%lu;%s;%c", connection_id, msg, '\0');
//SEND YOU DUMB FUCK
send_socket_deallocate(private_msg, why2_chat_server_config("server_username"), (*(connection_node_t*) pm_connection -> value).connection);
why2_deallocate(private_msg);
}
//DEALLOCATION
why2_deallocate(id);
}
//IGNORE INVALID CODES, THE USER JUST GOT THEIR LOBOTOMY DONE
} else if (strncmp(decoded_buffer, WHY2_CHAT_COMMAND_PREFIX, strlen(WHY2_CHAT_COMMAND_PREFIX)) != 0) //IGNORE MESSAGES BEGINNING WITH '!'
{
//REBUILD MESSAGE WITH USERNAME
json_object_object_add(json, "message", json_object_new_string(decoded_buffer));
json_object_object_add(json, "username", json_object_new_string(get_username(connection)));
json_object_object_foreach(json, key, value) //GENERATE JSON STRING
{
append(&raw_output, key, (char*) json_object_get_string(value));
}
add_brackets(&raw_output);
send_to_all(raw_output); //e
}
}
deallocation:
//DEALLOCATION
why2_deallocate(raw);
why2_deallocate(raw_ptr);
why2_deallocate(raw_output);
why2_deallocate(decoded_buffer);
json_object_put(json);
}
if (exiting) send_socket_deallocate(WHY2_CHAT_CODE_SSQC, why2_chat_server_config("server_username"), connection);
printf("User disconnected.\t%d\n", connection);
//DEALLOCATION
close(connection);
why2_deallocate(node.username);
why2_list_remove(&connection_list, find_connection(connection));
return NULL;
}
void *why2_accept_thread(void *socket)
{
int accepted;
pthread_t thread;
//LOOP ACCEPT
for (;;)
{
accepted = accept(*((int*) socket), (WHY2_SA *) NULL, NULL); //ACCEPT NEW SOCKET
if (accepted == -1) continue;
pthread_create(&thread, NULL, why2_communicate_thread, &accepted);
}
return NULL;
}
void why2_clean_connections(void)
{
why2_node_t *head = connection_list.head;
if (head == NULL) return; //EMPTY LIST
why2_node_t *node_buffer = head;
why2_node_t *node_buffer_2;
connection_node_t connection_buffer;
do //GO TROUGH LIST
{
node_buffer_2 = node_buffer;
node_buffer = node_buffer -> next;
connection_buffer = *(connection_node_t*) node_buffer_2 -> value;
send_socket_deallocate(WHY2_CHAT_CODE_SSQC, why2_chat_server_config("server_username"), connection_buffer.connection);
close(connection_buffer.connection);
why2_list_remove(&connection_list, node_buffer_2); //REMOVE
} while (node_buffer != NULL);
}
void why2_clean_threads(void)
{
why2_node_t *head = waiting_list.head;
if (head == NULL) return; //EMPTY LIST
why2_node_t *node_buffer = head;
why2_node_t *node_buffer_2;
do //GO TROUGH LIST
{
node_buffer_2 = node_buffer;
node_buffer = node_buffer -> next;
pthread_cancel(**(pthread_t**)(node_buffer_2 -> value));
why2_list_remove(&waiting_list, node_buffer_2); //REMOVE
} while (node_buffer != NULL);
}
void *why2_listen_server(void *socket)
{
//STUFF
char *read = NULL;
why2_bool exiting = 0;
why2_bool continuing;
unsigned char asking_username = 0;
char *server_uname = NULL;
//CONTENT
char *username = NULL;
char *message = NULL;
//SERVER SETTINGS
int max_uname = -1;
int min_uname = -1;
int max_tries = -1;
printf(">>> ");
fflush(stdout);
while (!exiting)
{
continuing = 0;
read = read_socket_raw(*((int*) socket));
if (read == NULL) continue;
//GET CONTENT
username = get_string_from_json_string(read, "username");
message = get_string_from_json_string(read, "message");
if (server_uname == NULL) //GET SERVER USERNAME
{
server_uname = why2_strdup(username); //GET USERNAME
//GET INFO
max_uname = get_int_from_json_string(read, "max_uname");
min_uname = get_int_from_json_string(read, "min_uname");
max_tries = get_int_from_json_string(read, "max_tries");
continuing = 1;
}
if ((strcmp(username, server_uname) == 0 && strncmp(message, "code", 4) == 0) && !continuing) //CODE WAS SENT
{
if (strcmp(message, WHY2_CHAT_CODE_SSQC) == 0) //SERVER BROKE UP WITH YOU
{
printf("%s%s%s\nServer closed the connection.\n", asking_username > max_tries ? WHY2_CLEAR_AND_GO_UP : "", WHY2_CLEAR_AND_GO_UP WHY2_CLEAR_AND_GO_UP, (asking_username == 0 ? "\n": ""));
fflush(stdout);
pthread_cancel(getline_thread); //CANCEL CLIENT getline
exiting = 1; //EXIT THIS THREAD
} else if (strcmp(message, WHY2_CHAT_CODE_PICK_USERNAME) == 0 || strcmp(message, WHY2_CHAT_CODE_INVALID_USERNAME) == 0) //PICK USERNAME (COULD BE CAUSE BY INVALID USERNAME)
{
if (strcmp(message, WHY2_CHAT_CODE_INVALID_USERNAME) == 0) //INVALID USERNAME
{
printf(WHY2_CLEAR_AND_GO_UP WHY2_CLEAR_AND_GO_UP "%s\nInvalid username!\n\n\n", asking_username > 1 ? WHY2_CLEAR_AND_GO_UP : "");
fflush(stdout);
}
printf("%s%sEnter username (a-Z, 0-9; %d-%d characters):\n", asking_username++ > 0 ? WHY2_CLEAR_AND_GO_UP : "", WHY2_CLEAR_AND_GO_UP, min_uname, max_uname);
fflush(stdout);
} else if (strncmp(message, WHY2_CHAT_CODE_LIST_SERVER, strlen(WHY2_CHAT_CODE_LIST_SERVER)) == 0) //LIST USERS
{
why2_bool printing_id = 0;
printf("\nList:\n-----\n");
//ITER TROUGH LIST OF USERS FROM SERVER
for (unsigned long i = strlen(WHY2_CHAT_CODE_LIST_SERVER) + 1; i < strlen(message); i++)
{
if (message[i] == ';')
{
printf((printing_id = !printing_id) ? " (" : ")\n");
continue;
}
printf("%c", message[i]);
}
printf("\n");
fflush(stdout);
} else if (strncmp(message, WHY2_CHAT_CODE_VERSION_SERVER, strlen(WHY2_CHAT_CODE_VERSION_SERVER)) == 0)
{
char *server_version = message + strlen(WHY2_CHAT_CODE_VERSION_SERVER) + 1;
//INFO
printf("\nServer Version: %s\nClient Version: %s\n\n", server_version, WHY2_VERSION);
//SERVER IS OUTDATED
if (atoi(server_version + 1) < atoi(WHY2_VERSION + 1))
{
printf("Server is outdated. Some new features may not work correctly.\n\n");
}
} else if (strncmp(message, WHY2_CHAT_CODE_PM_SERVER, strlen(WHY2_CHAT_CODE_PM_SERVER)) == 0)
{
printf("\n%s\n\n", message);
}
} else if (!continuing)
{
printf(WHY2_CLEAR_AND_GO_UP WHY2_CLEAR_AND_GO_UP); //do not fucking ask me how the fucking formatting fucking works, i dont fucking know
if (asking_username > 1)
{
asking_username = 0;
printf(WHY2_CLEAR_AND_GO_UP);
}
printf("\n\n%s%s: %s\n\n", WHY2_CLEAR_AND_GO_UP, username, message);
}
if (!exiting && !continuing)
{
printf(">>> ");
fflush(stdout);
}
//DEALLOCATION
why2_deallocate(read);
why2_deallocate(username);
why2_deallocate(message);
}
why2_deallocate(server_uname);
return NULL;
}
void *why2_getline_thread(WHY2_UNUSED void* arg)
{
getline_thread = pthread_self();
char *line = NULL;
size_t line_length = 0;
if (getline(&line, &line_length, stdin) == -1) why2_die("Reading input failed.");
return line;
}
void why2_trim_string(char **s)
{
//GET CORRECT ALLOCATION FNS
void* (*allocate)(unsigned long);
void (*deallocate)(void*);
if (why2_allocated(*s))
{
allocate = why2_malloc;
deallocate = why2_deallocate;
} else
{
allocate = malloc;
deallocate = free;
}
//BUFFERS
unsigned long start_spaces = 0;
unsigned long end_spaces = 0;
unsigned long actual_length;
//COUNT start_spaces (HOW MANY SPACES ARE IN THE START)
for (unsigned long i = 0; i < strlen(*s); i++)
{
if ((*s)[i] != ' ') break;
start_spaces++;
}
//COUNT end_spaces
for (long long i = (long long) strlen(*s) - 1; i >= 0; i--)
{
if ((*s)[i] != ' ') break;
end_spaces++;
}
//USER'S HEAD HAS FELL ON THE SPACEBAR
if (start_spaces + end_spaces > strlen(*s))
{
deallocate(*s);
*s = NULL;
return;
}
//COUNT actual_length
actual_length = strlen(*s) - (end_spaces + start_spaces);
if (actual_length == strlen(*s)) return; //NO SPACES TO REMOVE
char *st = allocate(actual_length + 2); //TRIMMED s
for (unsigned long i = start_spaces; i < (start_spaces + actual_length); i++)
{
st[i - start_spaces] = (*s)[i];
}
st[actual_length] = '\0';
//DEALLOCATE UNTRIMMED s
deallocate(*s);
//SET NEW s
*s = st;
}