fileTransfer_ClientServer/serwer.c

398 lines
15 KiB
C
Raw Normal View History

2019-04-30 06:45:20 +02:00
/**
* Implementacja wielowątkowe serwera obsługującego wiele klietów
* protokół TCP
* gniazda BSD
* system Unix/Linux
* testowany na serwerze lts.wmi***
*
* Tworzenie obiektu socketa dla serwera
* Tworzenie obiektu gniazd dla klieta
* 1. Wątki - obsługa polaczen od wielu klientow
* 2. Pokazywanie listy dostępnych dla klienta plików
* 3. Przesylanie za pomocą socketów wybranego przez klienta pliku
*
*
* gcc serwer.c -lpthread -o S
*/
#include <sys/types.h>
#include <sys/socket.h>
#include <stdio.h>
#include <stdlib.h>
#include <netinet/in.h>
#include <string.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stddef.h>
#include <netdb.h>
#include <dirent.h> // do listowania plikow w folderze
#include<pthread.h> // do watkow; uwaga na komende klompilacji! [gcc serwer.c -lpthread -o S]
#include <fcntl.h> // for open
#include <unistd.h> // for close
#define SERVER_PORT_NUMBER 12345
#define BACKLOG 10 // ilosc dozwolonych połczeńw kolejce przychodzących
#define BUFFER_SIZE 10000
#define SOCKET_DATA_LENGTH 1024 // do obierania i wysylania danych recv() i send()
char BUF[BUFFER_SIZE]; //globalna zmienna na bufor
pthread_mutex_t lock = PTHREAD_MUTEX_INITIALIZER; //semafor do blokowania
/********************* Deklaracje ****************************/
int get_message_from_client(int client_socket);
void * server_handler(void * client_socket);
int accept_connection(int server_socket, struct sockaddr_in server_struct);
void end_client_connection(int client_socket);
/*************************** Definicje /***************************/
struct sockaddr_in create_server_struct(int server_port){
/* Zwraca obiekt gniazda serwera
struct sockaddr_in { // równoważna sockaddr stworzona przez programistów do łatwijeszej pracy
short int sin_family; //rodzina adresów -- odpowiada sa_data z struct sockaddr
unsigned short int sin_port; // numer portu; kolejnosć sieciowa
struct in_addr sin_addr; //adres internetowy; sieciowa kolejność
unsigned char siz_zero[8]; // - zerować memssetem dla zachowania rozmiaru struct sockaddr
}
@server_address
@server_port (int): numer portu serwera
*/
struct sockaddr_in server_struct;
server_struct.sin_family = AF_INET;
server_struct.sin_port = htons(server_port); // używam wybranego przez siebie nr portu - ustawiajac na 0 pozwalam funkcji bind() wybrac losowo pot za mnie
server_struct.sin_addr.s_addr = htonl(INADDR_ANY); // puryście mówią żeby owijać INADDR_ANY w funkcję htonl - Host-to-Network-Long dla pełnej przenośności kodu; INADDR_ANY - używa mojego adresu IP
memset(&(server_struct.sin_zero), '\0', 8 ); //wyzerowuje reszte struktury
return server_struct;
}
int create_socket_TCP(){
/* buduje gniazdo TCP
socket(int domain, int type, int protocol) - zwraca deskryptor pliku, bo w Liuksie wszystko jest plikiem
gdzie: domain - doemna powinna być ustawiona na AF_INET
type - mówi jądru jaki jest typ gniazda: SOCK_STREAM dla TCP / SOCK_DGRAM dla UDP
protocol - ustawić na 0, żeby funckja socket() wybrała sama protokół bazują na parametrze type
@return int socket - desktyptor pliku
*/
return socket(AF_INET, SOCK_STREAM, 0);
}
void bind_server_socket(int sdsocket, struct sockaddr_in myaddr){
/* binduje gniazdo; UWAGA bind jest tylko z moją strukturą gniazda; nie z klienta łaczącego się zdalnie
struktura sockaddr jest i sockaddr_in (in od internet) równorzędne;
ta druga powstała przez programitów aby ułatwić sobie pracę
bind(int sockfd, struct sockaddr *myaddr, int addrlen);
gdzie: int sockfd to gniado - deskrypor pliku zwrócony przez socet(); bo w Linuxie wszystko jest plikiem (zwykły int)
struct sockaddr *myaddr - wskaźnikna strukturę która przechowuje informacje (dokłądniej porcie i adresie IP) o adresie gniazda dla różnych typów gniazda
int addrlen - może być ustawiony na rozmiar structury typu sockaddr (jak powyżej): sizeof(struct sockaddr)
bind() w przypadku błędu zwraca -1 i ustawia odpowiednio `errno` na wartośc błędu
nie jest totalnie potrzbeny gdy łączymy się ze zdalnym serwerem np telnetem wtedy sywtarczy tylko `connect`
struct sockaddr {
unsigned short sa_family; //rodzina adresów, AF_xxx
char sa_data[14]; // 14-bajtowy adres protokołowy
};
struct sockaddr_in { // równoważna sockaddr stworzona przez programistów do łatwijeszej pracy
short int sin_family; //rodzina adresów -- odpowiada sa_data z struct sockaddr
unsigned short int sin_port; // numer portu; kolejnosć sieciowa
struct in_addr sin_addr; //adres internetowy; sieciowa kolejność
unsigned char siz_zero[8]; // - zerować memssetem dla zachowania rozmiaru struct sockaddr
}
@note: Używac tylko portów z przedziału [1024; 65535> jeśli nie sa zajęte przez inny program; chyba że jest się superuserem
*/
int addrlen = sizeof(struct sockaddr_in);
if(bind(sdsocket, (struct sockaddr* ) &myaddr, addrlen) < 0){
//bind() w przypadku błedu zwraca -1
printf("[ERROR] Bind sie nie pwoiodl :<\n");
exit(-1);
}
printf("Bind sie udał\n");
}
void start_listen(int server_socket){
/*Serwer rzpoczyna nasłuchiwanie
listen(int sockfd, int backlog);
gdzie: sockfd - deskryptor gniazda serwera zwrócony rzez socket()
backlog - ilosć dozwolonych połaczęń w kolejsce przychodzacych
listen() zwraca -1 w przypadku błedu i ustawia odpowiednio zmienna errno
*/
if (listen(server_socket, BACKLOG) == -1){
printf("[Error] Listen sie nie powiodl\n");
exit(-1);
}
printf("Listen sie powiodl\n");
}
int transfer_file_to_client(int client_socket, char* filename ){
/*Przesyła wybrany plik do klienta*/
char file_data[SOCKET_DATA_LENGTH];
FILE *fs = fopen(filename, "r");
if(fs == NULL)
{
printf("[ERROR] Brak pliku '%s'. :< \n", filename);
return -2;
}
bzero(file_data, SOCKET_DATA_LENGTH);
int fs_block_sz;
while((fs_block_sz = fread(file_data, sizeof(char), SOCKET_DATA_LENGTH, fs))>0)
{
if(send(client_socket, file_data, fs_block_sz, 0) < 0)
{
printf("[ERROR] Send nie wysłał pliku '%s' :<.\n", filename);
return -1;
}
bzero(file_data, SOCKET_DATA_LENGTH);
}
printf("Udało sie wysłać plik '%s' do kleinta <3 \n", filename);
return 1;
}
char* list_all_files_in_directory(){
/*
Zwraca listę plików jakie klient może pobrać
- ograniczam sie do jednego folderu gdzie uruchamiany
jest serwer bo nie chce dac mozliwosci wiekszego przegldana klientowi
lub zmulenia spowodowanego wyswietlaniem zawartosci calego dysku
*/
char *filename_list = malloc(sizeof(char) * 1024); //alokowanie zmiennej na całą liste plikow
int filename_list_length = 0; // zmienna do sprawdzenia czy nie trzeba czasem rozszerzyc filename_list do przechowywania calej listy plikow
DIR *d;
struct dirent *dir;
d = opendir(".");
if (d) {
while ((dir = readdir(d)) != NULL) {
// printf("%s\n", dir->d_name);
if (dir->d_type == DT_REG){ // bez tego wyswietla tez foldery a nie pliki
// printf("%s\n", dir->d_name);
filename_list_length += sizeof(filename_list)/ sizeof(filename_list[0]);
if (filename_list_length >= SOCKET_DATA_LENGTH){
//powiekszanie zmiennej
filename_list = (char*)realloc(filename_list, (filename_list_length + 1024) * sizeof(char)); //rozszerzenie ablicy charów o 1024
if (! filename_list) { //jakby realloc sie nie udał
perror("realloc");
exit(-1);
}
}
strcat(filename_list, dir->d_name);
char tmp[] = "\n";
// strcat(filename_list, (char*)'\n');
strcat(filename_list, tmp);
}
}
closedir(d);
}
return filename_list;
}
int accept_connection(int server_socket, struct sockaddr_in server_struct){
/* akcpetuje polaczenie na gniezdzie
Czeka będzie jakieś połaczenie
zwraca socket dla clienta
*/
int client_socket;
int struct_len = (sizeof(struct sockaddr_in));
printf("Proba accept\n\n");
client_socket = accept(server_socket, (struct sockaddr *)&server_struct, &struct_len);
//dopoki klienci sie lacza tworz nowe watki dla klientow przychodzacych
if (client_socket < 0 ){ // nieudane accept()
printf("[Error] Accept się nie powiódł\n");
return -1; // nie wylaczam progeamu gdy np jeden z wielu klientow sie nie polaczyl
}
return client_socket;
}
int send_message_to_client(int client_socket, char * msg){
/*Wysyła wiadomosc do klienta; czsami wiadomosc jest dluzsza
@param int client_socket - gniazdo/deskryptor klienta
@param char *msg wiadomosc do klienta(string)
*/
int send_length = send(client_socket,
msg,
strlen(msg), 0);
if (send_length == 0){
printf("[Error] nie udało sie wysłac wiadomosci do klienta\n");
return -1;
}
return 1;
}
char* find_filename_in_message(char* client_message){
/*znajduje nazwe pliku pomiedzy znakami <nazwa_pliku.rozszerzenie>
i zwraca - do komunikacji z klientem by przeslac mu wybrany przez siebie plik
zakłdam że znaki < - informujacy o początku nazwy pliku
i znak `>` informujacy że ońcyz się nazwa pliku ystępuje tylko raz;
gdyby zakonczenia wystepowały wielokrotnie należy użyć `str.find_last_of(STOPDELIMITER);`
*/
client_message = "[3]<a.py>";
int message_len = strlen(client_message);
int start = -1; //poczatkowy indeks
int stop = -1; //koncowy indeks
int i =0;
while(i < message_len){
if(client_message[i] == '<'){
start = i+1;
}
if(client_message[i] == '>'){
stop = i-1;
}
i++;
}
char* filename = malloc(sizeof(char)*(stop-start+1)); //na razie alokuje na jeden znak
int j = start;
for(j; j<= stop; j++ ){
filename[j-start] = client_message[j];
}
filename[stop] = '\0';
return filename;
}
int get_message_from_client(int client_socket){ //Cała komunikacja i sterowanie z kliente,
/* Odbiera wiadomosc od klienta; sprawdza tresc -
Klient wysyła pojedyncze wiadomosci lub `[2]<nazwa_pliku.rozszerzenie>`
dlatego nie zwiększam bufora response_msg
jeśli klient wysłał widomosc o tresci [3] zamyka połączenie z tym klientem
jesli sa tam slowa kluczowe opcji to je wywoluje
@param int client_socket - gniazdo/deskryptor klienta
*/
char *response_msg = malloc( sizeof(char) * SOCKET_DATA_LENGTH);
while(1) {
// spawdza występowanie słowa np `[1]`, `[2]`, `[3]` w odpowiedzi klienta
printf("[Klient]: %s\n", response_msg);
recv(client_socket, response_msg, SOCKET_DATA_LENGTH, 0 );
if ( strstr(response_msg, "[1]")){
printf("TUTAJ [1]");
// jeśli klietn wybrał pierszą opcję to wysyła klientowi liste plikow
char* filename_list = list_all_files_in_directory();
send_message_to_client(client_socket, filename_list);
}
if ( strstr(response_msg, "[2]")){
send_message_to_client(client_socket, "Przesylanie pliku");
// sprawdzanie nazwy pliku
char* filename = find_filename_in_message(response_msg);
transfer_file_to_client( client_socket, filename );
}
if ( strstr(response_msg, "[3]")){
end_client_connection(client_socket);
return 1;
}
};
end_client_connection(client_socket);
return 1;
}
void end_client_connection(int client_socket){
/*Zamyka polaczenie z klientem*/
printf("Zakoczenie polaczenia z klientem\n");
send_message_to_client(client_socket, "[Serwer] Połaczenie zamkniete przez klienta");
close(client_socket);
}
void* server_handler(void * thread_socket){ //Najrrudniejsza funkcja [!]
/* Handler dla watku klienta
w watkach przekazywanie warosic jest trudniejsze i dziwniejsze
@ param: thread_socket - socket klienta do watku zwrocony przez accept() z accept_connection()
*/
// pthread_mutex_lock(&lock); //niepotrzebny nie wspoldziele danych miedzy klientami
static char message_from_client[SOCKET_DATA_LENGTH]; // iadomosc od klienta
int read_size, write_size;
int client_socket = *(int*) thread_socket; // client_socket zostanie zamkniety tutaj
printf("Funkcja Server handler\n");
//wyslanie takiego mojego ACK Z serwera DO klienta
send_message_to_client(client_socket, "Serwer wita :)\n");
if (get_message_from_client(client_socket)){
printf("Klient sam sie rozaczyl\n");
pthread_exit(NULL);
return 0;
}
if (client_socket){
end_client_connection(client_socket); // zamykanie po zakonczeniue komunikacji jak sie nie uda w get_message_from_client
}
pthread_exit(NULL);
return 0;
}
int main(){
printf("Czekam na połaczenie \n\n\n");
struct sockaddr_in myaddr, endpoint;
int server_socket, client_socket;
int server_port = SERVER_PORT_NUMBER; // na sztywno ustawiono dla szybszego testowania/ bez inputu
server_socket = create_socket_TCP(); //tworzenie gnaizda dla serwera
myaddr = create_server_struct(server_port); //struktura przechowujaca dane serwera
bind_server_socket(server_socket, myaddr); //1. Bindowanie
start_listen(server_socket); //2. Rozpoczenie nasluchu
// Akceptacja przychodzace polaczenie klienta
pthread_t clients_thread_list[20];
int i = 0; //licznik watkow; i jakos mi bardziej pasuje jak Licznik
while(1){
//Pętla główna dla serwera do akceptowania klientow i
client_socket = accept_connection(server_socket, myaddr); //zawiesza program aż połączy się klient
// mu
if( pthread_create(&clients_thread_list[i], NULL, server_handler, &client_socket) != 0 )
printf("Failed to create thread\n");
if( i >= BACKLOG) //gdzie BACKLOG to parametr z funkcji listen()
//mowiąca o maksymalnej liczbie połaczeń w kolejnce; po cichu z reguły ustawiona na 5 lub 10
{
i = 0; //licznik watkow
while(i < BACKLOG){
// musi byc joinowanie w watku glownym, bo inaczej main nie czeka na zakonczenie dzialania watku klietna
pthread_join(clients_thread_list[i++],NULL); //joinowanie watkow w glownym watku main
}
i = 0; //licznik watkow
}
}
close(server_socket); //zamykanie serwera
return 0;
}