在这一小节中实现了文件的下载,具体的思路是根据用户的uid和用户提供的文件名filename联合两张表,取得md5唯一标识符,然后操作这个标识符对应的文件发送给客户端。

  实现下载的小小网盘程序

  client.cpp增加下面这个函数以实现文件的下载。

 int file_pull(struct Addr addr,struct User user,char *filenames)
{
struct sockaddr_in servAddr;
struct hostent *host;
struct Control control;
struct File file;
int sockfd;
FILE * fp=NULL; host=gethostbyname(addr.host);
servAddr.sin_family=AF_INET;
servAddr.sin_addr=*((struct in_addr *)host->h_addr);
servAddr.sin_port=htons(addr.port); if(host==NULL)
{
perror("获取IP地址失败");
exit(-);
}
if((sockfd=socket(AF_INET,SOCK_STREAM,))==-)
{
perror("socket创建失败");
exit(-);
}
if(connect(sockfd,(struct sockaddr *)&servAddr,sizeof(struct sockaddr_in))==-)
{
perror("connect 失败");
exit(-);
} //传输控制信号
control.control=FILE_PULL;
control.uid=user.uid;
if(send(sockfd,(char *)&control,sizeof(struct Control),)<)
{
perror("控制信号发送失败");
exit(-);
}
strcpy(file.filename,filenames);
file.uid=user.uid;
if(send(sockfd,(char *)&file,sizeof(struct File),)<)
{
perror("文件指纹发送失败");
exit(-);
}
if((fp=fopen("data","wb"))==NULL)
{
perror("文件打开失败");
exit(-);
}
int size=;
int data_len=;
char buffer[BUFFER_SIZE];
memset(buffer,,sizeof(buffer));
recv(sockfd,buffer,,);
if(buffer[]=='n')
{
printf("服务器中没有该文件,请确认后再输入,如不知道是否有文件,可以使用file list查看\n");
return ;
}
memset(buffer,,sizeof(buffer));
while(data_len=recv(sockfd,buffer,BUFFER_SIZE,))
{
if(data_len<)
{
perror("接收数据有误");
}
size++;
if(size==)
{
printf("正在接收来自服务器的文件");
}
else
{
printf(".");
}
int write_len=fwrite(buffer,sizeof(char),data_len,fp);
if(write_len>data_len)
{
perror("写入数据有误");
}
bzero(buffer,BUFFER_SIZE);
}
printf("\n文件接收完毕\n");
fclose(fp);
rename("data",filenames);
close(sockfd);
return ;
}

  server.cpp 同样的实现一个相同的功能

 int main(int argc,char *argv[])
{
struct sockaddr_in server_addr;
struct sockaddr_in client_addr;
struct User user;
struct Control control;
char ch[];
int clientfd;
pid_t pid;
socklen_t length;
bzero(&server_addr,sizeof(server_addr));
server_addr.sin_family=AF_INET;
server_addr.sin_addr.s_addr=htons(INADDR_ANY);
server_addr.sin_port=htons(SERVER_PORT); //创建套接字
int sockfd=socket(AF_INET,SOCK_STREAM,);
if(sockfd<)
{
perror("创建套接字失败");
exit(-);
} if(bind(sockfd,(struct sockaddr *)&server_addr,sizeof(server_addr))==-)
{
perror("bind 失败");
exit(-);
} if(listen(sockfd,LISTEN_QUEUE))
{
perror("listen 失败");
exit(-);
} length=sizeof(struct sockaddr); while()
{
clientfd=accept(sockfd,(struct sockaddr *)&client_addr,&length);
if(clientfd==-)
{
perror("accept 失败");
continue;
}
printf(">>>>>%s:%d 连接成功,当前所在的ID(fd)号: %d \n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port),clientfd);
print_time(ch);
printf("加入的时间是:%s\n",ch); //来一个连接就创建一个进程进行处理
pid=fork();
if(pid<)
{
perror("fork error");
}
else if(pid==)
{
recv(clientfd,(char *)&control,sizeof(struct Control),);
printf("用户 %d 使用命令 %d\n",control.uid,control.control);
switch(control.control)
{
case USER_CHECK_LOGIN:
{
//身份验证处理
recv(clientfd,(char *)&user,sizeof(struct User),);
printf("客户端发送过来的用户名是:%s,密码:%s\n",user.username,user.password);
if((user.uid=mysql_check_login(user))>)
{
printf("验证成功\n");
}
else
{
printf("验证失败\n");
}
send(clientfd,(char *)&user,sizeof(struct User),);
break;
}
case FILE_PUSH:
{
char buffer[BUFFER_SIZE];
int data_len;
FILE * fp=NULL;
struct File file;
//获取文件指纹
recv(clientfd,(char *)&file,sizeof(struct File),);
printf("获取到的用户名ID: %d 文件名:%s MD5:%s\n",file.uid,file.filename,file.md5);
//对文件进行验证,如果文件已经存在就不用进行接收了
int t=mysql_check_md5(file);
char ch[]={};
printf("t=%d\n",t);
if(t!=)
{
printf("该文件存在,使用秒传功能\n");
strcpy(ch,"yes");
send(clientfd,ch,,);
mysql_file_in(file.uid,t);
//continue;
}
strcpy(ch,"no");
send(clientfd,ch,,);
printf("md5验证后得到的fid:%d\n",t);
bzero(buffer,BUFFER_SIZE);
if((fp=fopen("data","wb"))==NULL)
{
perror("文件打开失败");
exit(-);
}
//循环接收数据
int size=;//表示有多少个块
while(data_len=recv(clientfd,buffer,BUFFER_SIZE,))
{
if(data_len<)
{
perror("接收数据错误");
exit(-);
}
size++;
if(size==)
printf("正在接收来自%s:%d的文件\n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
else
printf(".");
//向文件中写入
int write_len=fwrite(buffer,sizeof(char),data_len,fp);
if(write_len>data_len)
{
perror("写入数据错误");
exit(-);
}
bzero(buffer,BUFFER_SIZE);
}
if(size>)
{
printf("\n%s:%d的文件传送完毕\n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
//如果文件传输成功那么就可以写入数据库了
mysql_file_in(file);
}
else
printf("\n%s:%d的文件传送失败\n",inet_ntoa(client_addr.sin_addr),ntohs(client_addr.sin_port));
fclose(fp);
rename("data",file.md5);//这里可以修改文件的名字
exit();
break;
}
case FILE_PULL:
{
struct File file;
char ch[];
memset(ch,,sizeof(ch));
//获取接下来要发送的文件
recv(clientfd,(char *)&file,sizeof(struct File),);
//根据uid和filename获取服务器中的唯一文件,然后发送
int t=mysql_get_md5_from_file(&file);
printf("获取到的MD5:%s\n",file.md5);
if(t==-||file.md5[]==)//服务器没有对应的文件
{
printf("没有对应的文件\n");;
strcpy(ch,"no");
send(clientfd,ch,,);
continue;
}
strcpy(ch,"yes");
send(clientfd,ch,,); FILE * fp = NULL;
if((fp=fopen(file.md5,"rb"))==NULL)
{
perror("文件打开失败");
}
char buffer[BUFFER_SIZE];
bzero(buffer,BUFFER_SIZE);
printf("正在传输文件");
int len=;
while((len=fread(buffer,sizeof(char),BUFFER_SIZE,fp))>)
{
if(send(clientfd,buffer,BUFFER_SIZE,)<)
{
perror("发送数据失败");
}
bzero(buffer,BUFFER_SIZE);
printf(".");
}
printf("传输完成\n");
fclose(fp);
break;
}
case FILE_LIST:
{
break;
}
case FILE_DELECT:
{
break;
}
default:
{
break;
}
}
shutdown(clientfd,);//这里要注意加一个shutdown否则客户端接收不到结束符而一直等待接收数据。普通的close是不会发送结束符的
close(clientfd);//短连接结束
exit();//退出子进程
}
} return ;
} /////////////////// int mysql_get_md5_from_file(struct File * file)
{
//select md5 from files,relations where files.fid=relations.fid and file.md5=file.md5;
MYSQL conn;
MYSQL_RES * res_ptr;
MYSQL_ROW result_row;
int res;int row;int column;
int rt;
char sql[]={};
rt=;
strcpy(sql,"select md5 from files,relations where files.fid=relations.fid and files.filename=\"");
strcat(sql,file->filename);
strcat(sql,"\"");
printf("查询的sql:%s\n",sql);
mysql_init(&conn);
if(mysql_real_connect(&conn,"localhost","root","","filetranslate",,NULL,CLIENT_FOUND_ROWS))
{
res=mysql_query(&conn,sql);
if(res)
{
perror("select sql error!");
}
else
{
res_ptr=mysql_store_result(&conn);
if(res_ptr)
{
column=mysql_num_fields(res_ptr);
row=mysql_num_rows(res_ptr)+;
if(row<=)
{
;
}
else
{
result_row=mysql_fetch_row(res_ptr);
if(result_row[]==NULL)
{
rt=-;
strcpy(file->md5,"");
}
else
strcpy(file->md5,result_row[]);
}
}
else
{
rt=-;
printf("没有数据\n");
}
}
}
else
{
perror("Connect Failed1");
exit(-);
}
mysql_close(&conn);
return rt;
}

  mysql_get_md5_from_file这个函数是利用用户的uid和文件名进行查询的。因为保存在服务器中的文件是以MD5的值作为文件名的(目的是不同的用户可以有相同的文件名,但是所对应的文件确实不同的),查询后返回一个唯一文件标识MD5值,根据这个值取得文件然后发送给客户端。

  还有一个要注意的是在server.cpp的199行处,使用了shutdown来结束服务器的发送,一开始我没有使用该函数,造成的结果是服务器发送数据后,调用close,调用exit退出子进程,结束socket连接。虽然是关闭了socket,但是在客户端却在recv处一直处于阻塞,好像是还有数据没有接收完。经过调试还有查资料才知道,这里要调用shutdown,否则客户端接收不到结束符而一直等待接收数据。普通的close是不会发送结束符的。

  #include <sys/socket.h>

  int shutdown(int sockfd, int how);  //返回值: 如果成功则返回0,否则出错返回-1

  套接字通信是双向的。可以采用函数shutdown来禁止套接字上的输入/输出。如果how是SHUT_RD(关闭读端 0 ),那么就无法从套接字读取数据;如果how是SHUT_WR(关闭写端 1 ),那么无法使用套接字发送数据;使用SHUT_RDWR则将同时无法读取和发送数据(2).

  既然能够close关闭套接字,那么为什么还要用shutdown呢?理由如下:首先,close只有在最后一个活动引用被关闭时才释放网络端点【这就是为什么我以前的章节可以用close来结束上传,那是因为客户端的发送数据是在一个函数里面的一个连接,函数结束,连接也就断了。而这次出现不能下载,就是因为我的服务器是使用多进程的,而进程clientfd实在fork的前面定义的,所以当发送完毕后调用close是因为clientfd还被引用到,而不是最后一个活动应用】。这意味着如果复制一个套接字(如采用dup),套接字直到关闭了最后一个引用它的文件描述符之后才会被释放。而shutdown允许使一个套接字处于不活动状态,无论引用它的文件描述符数目多少。其次,有时只有关闭套接字双向传输中的一个方向会很方便。例如,如果想让所通信的进程能够确定数据发送何时结束,可以关闭该套接字的写端,然而通过该套接字读端仍然可以继续接收数据。

  下面给出运行时的截图

  运行的顺序具体看一下命令就知道了,最后出现了一个问题,就是使用不同的用户居然可以下载src这个文件,想想应该是数据库sql没有写好。我们修改如下:

  在上面server.cpp代码的第222行处加上下面代码即可,实现对用户的权限控制。

     strcat(sql," and relations.uid=");
sprintf(ch,"%d",file->uid);
strcat(sql,ch);
strcat(sql,";");

  好了,至此我们已经实现了文件的上传和下载了,并且还能进行用户权限的控制。话说FTP是不是就差不多这个样子啊。

  本文地址: http://www.cnblogs.com/wunaozai/p/3892729.html

Socket网络编程--小小网盘程序(4)的更多相关文章

  1. Socket网络编程--小小网盘程序(5)

    各位好呀!这一小节应该就是这个小小网盘程序的最后一小节了,这一节将实现最后的三个功能,即列出用户在服务器中的文件列表,还有删除用户在服务器中的文件,最后的可以共享文件给好友. 列出用户在服务器中的文件 ...

  2. Socket网络编程--小小网盘程序(2)

    这一节将不会介绍太多的技术的问题,这节主要是搭建一个小小的框架,为了方便接下来的继续编写扩展程序.本次会在上一小节的基础上加上一个身份验证的功能. 因为网盘程序不像聊天程序,网盘是属于主动向服务器拉取 ...

  3. Socket网络编程--小小网盘程序(3)

    接上一小节,这次增加另外的两张表,用于记录用户是保存那些文件.增加传上来的文件的文件指纹,使用MD5表示. 两张表如下定义: create table files( fid int, filename ...

  4. Socket网络编程--小小网盘程序(1)

    这个系列是准备讲基于Linux Socket进行文件传输.简单的文件传输就是客户端可以上传文件,可以从服务器端下载文件.就这么两个功能如果再加上身份验证,就成了FTP服务器了,如果对用户的操作再加上一 ...

  5. linux下C语言socket网络编程简例

    原创文章,转载请注明转载字样和出处,谢谢! 这里给出在linux下的简单socket网络编程的实例,使用tcp协议进行通信,服务端进行监听,在收到client的连接后,发送数据给client:clie ...

  6. 5.3linux下C语言socket网络编程简例

    原创文章,转载请注明转载字样和出处,谢谢! 这里给出在Linux下的简单socket网络编程的实例,使用tcp协议进行通信,服务端进行监听,在收到客户端的连接后,发送数据给客户端:客户端在接受到数据后 ...

  7. Socket网络编程--聊天程序(9)

    这一节应该是聊天程序的最后一节了,现在回顾我们的聊天程序,看起来还有很多功能没有实现,但是不管怎么说,都还是不错的.这一节我们将讲多服务器问题(高大上的说法就是负载问题了.)至于聊天程序的文件发送(也 ...

  8. Socket网络编程系列教程序

    C语言的用途相当多,可以用在数据结构.数据库.网络.嵌入式等方面,历经40多年不衰,真是厉害!最近一直想从某一应用方面写一个系列教程,好好地把某一方面讲深讲透.         正好博主对网络方面的编 ...

  9. Socket网络编程--FTP客户端

    Socket网络编程--FTP客户端(1)(Windows) 已经好久没有写过博客进行分享了.具体原因,在以后说. 这几天在了解FTP协议,准备任务是写一个FTP客户端程序.直接上干货了. 0.了解F ...

随机推荐

  1. Ubuntu 下常用快捷键

    参考链接:Ubuntu终端以及应用下快捷键大全https://linux.cn/article-3025-1.html 桌面常用快捷键 Alt + F1:聚焦到桌面左侧任务导航栏,可按上下键进行导航 ...

  2. VSCode从非根目录编译golang程序(转)

    1.问题提出 “习惯在项目目录里建src放源码文件,根目录里放配置文件或者别的什么,在交付时直接忽视掉src目录就行了,但vscode好像不能这样愉快的玩耍...”??? 要实现把源码放到src目录下 ...

  3. hql查询后释放内存

    Session session=getSession(); //进行session查询,取得前16个数据 Query q=session.createQuery(hql).setFirstResult ...

  4. 获取AFP服务信息

    获取AFP服务信息   如果苹果系统开放TCP 548端口,说明其开启了AFP服务.这个时候,可以使用Nmap的afp-serverinfo脚本获取对应的服务信息.获取的信息包括服务名.机器类型.AF ...

  5. C# 使用 iTextSharp 将 PDF 转换成 TXT 文本

    var pdfReader = new PdfReader("xxx.pdf"); StreamWriter output = new StreamWriter(new FileS ...

  6. BZOJ 4552 [Tjoi2016&Heoi2016]排序 线段树的分裂和合并

    https://www.lydsy.com/JudgeOnline/problem.php?id=4552 https://blog.csdn.net/zawedx/article/details/5 ...

  7. UVA.12230.Crossing Rivers(期望)

    题目链接 /* 到达一条河时,船在河中的位置是随机的,所以船到达岸边需要的时间在 0~2l/v 均匀分布,所以船到岸的期望为 (0+2l/v)/2 过河需要 l/v 的时间,所以过一条河总的期望为 ( ...

  8. error MSB3073 解决办法

    发现拷贝命令编译错误,查看输出列表发现时无法找到相应的路径. 1.顺着这个思路第一个想到的是中文路径的问题,先修改了盘符的中文名称,发现还是无法解决具体的问题. 2.后来反复查阅网上的资料,发现帮助并 ...

  9. bootStrap中的ul导航2

    <!doctype html><html > <head> <meta charset="utf-8"> <link rel= ...

  10. 以添加评论组件为例看angular2请求数据的处理

    在NiceFish项目中,数据请求处理并没有用Promise的那一套方法,用的是Observable(观察者模式),我将其理解成生产者和消费者模式 如下简单例子:出自(https://segmentf ...