-
Notifications
You must be signed in to change notification settings - Fork 1
/
serveur.c
302 lines (209 loc) · 10.2 KB
/
serveur.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
/*----------------------------------------------
Serveur à lancer avant le client
------------------------------------------------*/
#define DIR_DL "./"
#include <stdlib.h>
#include <stdio.h>
#include <linux/types.h> /* pour les sockets */
#include <sys/socket.h>
#include <sys/stat.h>
#include <netdb.h> /* pour hostent, servent */
#include <string.h> /* pour bcopy, ... */
#include "libnet.c"
#include <fcntl.h> // for open
#include <unistd.h> // for close
typedef struct sockaddr sockaddr;
typedef struct sockaddr_in sockaddr_in;
typedef struct hostent hostent;
typedef struct servent servent;
/**
fonction d'envoie de la sortie d'une commande system
*/
// a voir pour l'histoire d'un chemin fictif pour le pwd ou seulement dans les sous dossier du serveur.
void send_cmd(char * cmd,int socket){
FILE* cmdF;
char cmd_c[100]="";
char cmd_t[100]="";
char contents[300]="";
strcat(cmd_t,cmd);
//certain \n se retrouvait dans le code => suppression de ce symbole.
delete_retC(cmd_t);
cmdF = popen(cmd_t, "r");
char tmp[300]="";
if(cmdF!=NULL){
while(fgets(contents,300, cmdF) != NULL){
strcat(tmp,contents);
}
strcat(tmp, "\n");
strcpy(contents,tmp);
}else{
perror("commande interrompu");
}
pclose(cmdF);
send_string(socket,contents);
}
int main(int argc, char **argv) {
int socket_descriptor_cmd, /* descripteur de socket */
nouv_socket_descriptor_cmd, /* [nouveau] descripteur de socket */
longueur_adresse_courante, /* longueur d'adresse courante d'un client */
action,
ack; // action courante demander par le client.
sockaddr_in adresse_locale, /* structure d'adresse locale*/
adresse_client_courant; /* adresse client courant */
hostent* ptr_hote; /* les infos recuperees sur la machine hote */
char machine[TAILLE_MAX_NOM+1]; /* nom de la machine locale */
char filePath[TAILLE_MAX_NOM];
gethostname(machine,TAILLE_MAX_NOM); /* recuperation du nom de la machine */
/* recuperation de la structure d'adresse en utilisant le nom */
if ((ptr_hote = gethostbyname(machine)) == NULL) {
perror("erreur : impossible de trouver le serveur a partir de son nom.");
exit(1);
}
// create directory of file server
/* initialisation de la structure adresse_locale avec les infos recuperees */
/* copie de ptr_hote vers adresse_locale */
bcopy((char*)ptr_hote->h_addr, (char*)&adresse_locale.sin_addr, ptr_hote->h_length);
adresse_locale.sin_family = ptr_hote->h_addrtype; /* ou AF_INET */
adresse_locale.sin_addr.s_addr = INADDR_ANY; /* ou AF_INET */
/* 2 facons de definir le service que l'on va utiliser a distance */
/* (commenter l'une ou l'autre des solutions) */
/*-----------------------------------------------------------*/
/* SOLUTION 1 : utiliser un service existant, par ex. "irc" */
/*
if ((ptr_service = getservbyname("ftp","tcp")) == NULL) {
perror("erreur : impossible de recuperer le numero de port du service desire.");
exit(1);
}
adresse_locale.sin_port = htons(ptr_service->s_port);
*/
/*-----------------------------------------------------------*/
/* SOLUTION 2 : utiliser un nouveau numero de port */
adresse_locale.sin_port = htons(2001); // port de commande
/*-----------------------------------------------------------*/
printf("numero de port pour la connexion au serveur : %d \n",
ntohs(adresse_locale.sin_port) /*ntohs(ptr_service->s_port)*/);
/* creation de la socket principale de connexion "commande"*/
if ((socket_descriptor_cmd = socket(AF_INET, SOCK_STREAM, 0)) < 0) {
perror("erreur : impossible de creer la socket de connexion avec le client.");
exit(1);
}
/* association du socket socket_descriptor à la structure d'adresse adresse_locale */
if ((bind(socket_descriptor_cmd, (sockaddr*)(&adresse_locale), sizeof(adresse_locale))) < 0) {
perror("erreur : impossible de lier la socket a l'adresse de connexion.");
exit(1);
}
/* initialisation de la file d'ecoute */
listen(socket_descriptor_cmd,5);
/* attente des connexions et traitement des donnees recues */
for(;;) {
longueur_adresse_courante = sizeof(adresse_client_courant);
/* adresse_client_courant sera renseigné par accept via les infos du connect */
if ((nouv_socket_descriptor_cmd = accept(socket_descriptor_cmd, (sockaddr*)(&adresse_client_courant), &longueur_adresse_courante)) < 0) {
perror("erreur : impossible d'accepter la connexion avec le client.");
exit(1);
}
int pid = fork();
if (pid == 0) {
//envoie a la connexion du client du chemin ou il se trouve.
FILE * pwdC = popen("pwd","r");
char pwd[200];
char contents[200]="";
while (fgets(pwd, 200, pwdC) != NULL){
strcat(contents,pwd);
}
delete_retC(contents);
strcat(contents,"/serveur-folder");
send_string(nouv_socket_descriptor_cmd,contents);
//ACK
int actionR= recv(nouv_socket_descriptor_cmd,&action,sizeof(int),0);
char * filePath;
char * toDl;
while(action != EXIT){
if(action>0){ // si une action est envoyer par le client on la traite
switch(action){
case UPLOAD:
printf("commande UPLOAD \n");
int upOK=0;
int rupok = recv(nouv_socket_descriptor_cmd,&upOK,sizeof(int),0);
printf("%d\n",upOK);
printf("%d\n", rupok );
if(rupok >0 && upOK){
//fichier sur le serveur ou déposer le fichier.
toDl = recv_string(nouv_socket_descriptor_cmd);
if(toDl != NULL ){
if(reception_fichier(&nouv_socket_descriptor_cmd,toDl) == 1){
//envoie d'un aquittement au client.
ack=1;
send(nouv_socket_descriptor_cmd,&ack,sizeof(int),0);
}
}else{
printf("erreur de reception du chemin ou sauvegarder le fichier");
}
}
break;
case DOWLOAD:
printf("commande DOWNLOAD \n");
//reception du nom de fichier a envoyer au client
//todo vérification de la réception des éléments
int downok=0;
char pass[200]="";
filePath= recv_string(nouv_socket_descriptor_cmd);
if(filePath != NULL){
if(file_exists(filePath)){
downok = 1;
if(send(nouv_socket_descriptor_cmd,&downok,sizeof(int),0)>0){
transfert_fichier(nouv_socket_descriptor_cmd,filePath);
}
}else {
send(nouv_socket_descriptor_cmd,&downok,sizeof(int),0);
}
}
break;
case LS_CMD:
printf("commande LS_CMD \n");
int fok=0;
int sfok=0;
char * filePath=recv_string(nouv_socket_descriptor_cmd);
delete_retC(filePath);
if(folder_exists(filePath)){
fok=1;
sfok=send(nouv_socket_descriptor_cmd,&fok,sizeof(int),0);
if(sfok > 0){
char * cmd=recv_string(nouv_socket_descriptor_cmd);
send_cmd(cmd,nouv_socket_descriptor_cmd);
}else{
perror("erreur d'envoie de la vérification du fichier/dossier");
}
}else{ // le dossier n'est pas disponible on renvoie 0;
sfok=send(nouv_socket_descriptor_cmd,&fok,sizeof(int),0);
if(sfok < 0){
perror("erreur d'envoie de la vérification du fichier/dossier");
}
}
break;
case CD_CMD:
printf("commande CD_CMD \n");
//récupération d'une demande de vérification
char * path=recv_string(nouv_socket_descriptor_cmd);
delete_retC(path);
int f_ok= folder_exists(path);
send(nouv_socket_descriptor_cmd,&f_ok,sizeof(int),0);
break;
}
int ract=recv(nouv_socket_descriptor_cmd,&action,sizeof(int),0);
if(ract == 0 || ract < 0){
action = EXIT;
printf("problème ou déconnexion du client \n" );
close(nouv_socket_descriptor_cmd);
}
}
}
} else if (pid== -1){
perror("impossible de créer un fils");
exit(1);
}else{
close(nouv_socket_descriptor_cmd);
}
}
return 0;
}