Я изучаю программирование сокетов на языке C, и нам нужно написать TCP-клиент / сервер, где клиент отправляет файл на сервер построчно, чтобы он мог выполнить одну из двух операций (min, avg) надчтобы отправить выходные данные клиенту.
Входной файл имеет следующий формат:
AVG 1 2 3 4 5 6
MIN 1 2 3 4 5 -8
MIN 1 2 3 0 5 6
Проблема в том, что сервер читает только одну строку и вычисляет ее, а также вывод впочему-то за клиентом следуют нули !!
код сервера:
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <sys/types.h>
#include <time.h>
#define MAXLINE 4096
double average(int num[],int iterations); //to find the average value
int minimun(int num[],int iterations); //to find the minimum value
double get_op(char b[],int op); //to distinguish between the operations
int startswith(const char *pre, const char *str); //to find if the operatioon is valid only min or avg
int main(int argc, char *argv[]){
if(argc != 2){ //recieves portnumber
printf("Missing port number please try again.\n");
return 1;
}
int listenfd = 0, connfd = 0 , line = 0;
char SENDBUFFER[2048] = "\0";
char recvBuff[2048] = "\0";
struct sockaddr_in serv_addr;
listenfd = socket(AF_INET, SOCK_STREAM, 0);
memset(&serv_addr, '0', sizeof(serv_addr));
memset(recvBuff, '0', sizeof(recvBuff));
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = inet_addr("127.0.0.1");
serv_addr.sin_port = htons(atoi(argv[1]));
bind(listenfd, (struct sockaddr*)&serv_addr, sizeof(serv_addr));
listen(listenfd, 4); //listens to 4 clients max
while(1)
{
socklen_t szaddr = sizeof(serv_addr);
connfd=accept(listenfd,(struct sockaddr*)&serv_addr,&szaddr);
int pid;
if( (pid = fork()) == 0 ) { // Child
close(listenfd);//close listening sock
printf("Connected...");
while(1){
read(connfd,recvBuff,sizeof(recvBuff)-1);
int op;
line++;
double ans;
if(startswith("AVG ",recvBuff)){
op = 1;
ans = get_op(recvBuff,1);
sprintf(SENDBUFFER," %f",ans);
strcat(recvBuff," = ");
strcat(recvBuff,SENDBUFFER);
write(connfd, recvBuff, strlen(recvBuff));
}
if(startswith("MIN ",recvBuff)){
op = 2;
ans = get_op(recvBuff,2);
sprintf(SENDBUFFER," %f",ans);
strcat(recvBuff," = ");
strcat(recvBuff,SENDBUFFER);
write(connfd, recvBuff, strlen(recvBuff));
}
else{
memset(SENDBUFFER, '0', sizeof(SENDBUFFER));
strcat(SENDBUFFER,"Unsupported Operation on Line #");
char *str = "";
sprintf(str, "%d", line);
strcat(SENDBUFFER,str);
write(connfd, SENDBUFFER, strlen(SENDBUFFER));
}
}
close(connfd);
sleep(1);// child terminates
}else {// Parent
close(connfd); // close connected socket
}
}
return 0;
}
int minimun(int num[],int iterations){
int c;
int min = num[0];
for (c = 0;c<iterations; c++){
if (num[c] <= min){
min = num[c];
}
}
return min;
}
int startswith(const char *pre, const char *str){
char cp;
char cs;
if (!*pre)
return 1;
while ((cp = *pre++) && (cs = *str++)){
if (cp != cs)
return 0;
}
if (!cs)
return 0;
return 1;
}
double average(int num[],int iterations){
int sum = 0,c = 0;
double avg = 0;
int i;
for(i = 0;i<iterations;i++){
sum=sum+num[i];
}
avg=(double)sum/iterations;
return avg;
}
double get_op(char b[],int op){
int arr[2048], idx=0, d, l=0;
char *p; int counter = 0;
for (p = b; *p != 0; p+=l) {
l = 1;
if (isdigit(*p)){
sscanf(p, "%d%n", &d, &l);
arr[idx++] = d;
}
}
if(op==1)//avg
return average(arr,idx);
else //min
return (double)(minimun(arr,idx));
}
и это код клиента:
/* Client Code - Client.c */
#include <unistd.h>
#include <sys/socket.h>
#include <sys/types.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <time.h>
#include <stdlib.h>
#define MAXLINE 4096
#define filename "path to file"
int main(int argc,char *argv[])
{
char SENDBUFFER [MAXLINE] = "\0";
char recvBuff [MAXLINE] = "\0";
struct sockaddr_in serv_addr;
//int len;
int sockfd ;
if(argc<5){
printf("usage:serv_IP_addr, serv_port,num_of_microsec,nam_of_fil to read the input from\n");
exit(0);}
memset(SENDBUFFER, '0', sizeof(SENDBUFFER));
if((sockfd = socket(AF_INET, SOCK_STREAM, 0))< 0)
{
printf("\n Error : Could not create socket \n");
return 1;
}
memset(&serv_addr,0,sizeof(serv_addr));
serv_addr.sin_family = AF_INET;
serv_addr.sin_port = htons(atoi(argv[2])); // port
serv_addr.sin_addr.s_addr = inet_addr(argv[1]);
/* Attempt a connection */
if(connect(sockfd, (struct sockaddr *)&serv_addr, sizeof(serv_addr))<0){
printf("\n Error : Connect Failed \n");
return 1;
}
FILE *file; //file to read from
file = fopen("file path", "r"); //path to file
if (file == NULL){
printf("File not found!\n");
return 1;
}
printf("Found file %s\n", filename);
printf("Sending the file now\n");
//write function
while(fgets(SENDBUFFER, MAXLINE,file)!=NULL){
write(sockfd, SENDBUFFER, sizeof(SENDBUFFER)-1);
}
//read function
memset(recvBuff,'0',sizeof(recvBuff));
int n ;
while((n=read(sockfd,recvBuff,sizeof(recvBuff)-1))> 0){
if(fputs(recvBuff,stdout)==EOF){
printf("\n Error : nothing recieved\n");
}
}
printf("\nDone Sending the File!\n");
printf("Now Closing Connection.\n");
fclose(file);
close(sockfd);
return 0;
}
PS: я добавил fork() потому что я пытаюсь заставить сервер обрабатывать максимум 4 клиента, в то время как другие ждут, но еще не закончили.