linuxC语言tftp文件服务站程序编写

linux文件服务站类似于tftp

功能:ls 查看服务端文件列表
pwd查看服务端路径
cd 切换服务端路径
get 从服务端获取文件
put上传文件到服务端
quit客户端退出
lls查看客户端文件列表
lcd 客户端切换路径
lpwd查看客户端路径
1.客户端 client.c

#include <stdlib.h>
#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <string.h>
#include "config.h"
#include <sys/stat.h>
#include <fcntl.h>
int get_cmd(char *cmd){
        if(!strcmp(cmd,"ls")) return LS;
        if(!strcmp(cmd,"pwd")) return PWD;
        if(!strcmp(cmd,"quit")) return QUIT;
        if(!strcmp(cmd,"lls"))  return LLS;
        if(!strcmp(cmd,"lpwd")) return LPWD;
        if(strstr(cmd,"cd")!=NULL) return CD;
        if(strstr(cmd,"get")!=NULL) return GET;
        if(strstr(cmd,"put")!=NULL) return PUT;
        if(strstr(cmd,"lcd")!=NULL) return LCD;
        return -1;
}
char *getdir(char *cmd){
        char *p=NULL;
        p=strtok(cmd," ");
        p=strtok(NULL," ");
        return p;
}

int cmd_handler(struct Msg msg,int c_fd){
        int ret=get_cmd(msg.data);
        char *dir=NULL;

        int fd;
        char buf[1024]={0};
        switch(ret){
                case PWD:
                case LS:
                case CD:
                        msg.type=1;
                        write(c_fd,&msg,sizeof(msg));
                        break;
                case GET:
                        msg.type=2;
                           write(c_fd,&msg,sizeof(msg));
                        break;
                case PUT:
                        strcpy(buf,msg.data);
                        dir=getdir(buf);
                        if(access(dir,F_OK)==-1){
                                printf("dir : %sno this file\n",dir);


                        }else{
                                fd=open(dir,O_RDWR);
                                read(fd,msg.secondBuf,sizeof(msg.secondBuf));
                                close(fd);

                                write(c_fd,&msg,sizeof(msg));
                        }
                        break;
                case LLS:
                        system("ls");
                        break;
                case LPWD:
                        system("pwd");
                        break;
                case LCD:
                        dir=getdir(msg.data);
                        chdir(dir);
                        break;
                case QUIT:
                        write(c_fd,&msg,sizeof(msg));
                        exit(-1);
                        break;

        }
        return ret;
}
void handler_server_message(int c_fd,struct Msg msg){
        struct Msg msgget;
        char *dir=NULL;
        int newfd;
        memset(&msgget,0,sizeof(msgget));
        int n_read=read(c_fd,&msgget,sizeof(msgget));
        if(n_read==0){
                printf("the server is quit\n");
                exit(-1);
        }
        if(msgget.type==DOFILE){
                dir=getdir(msg.data);
                newfd=open(dir,O_RDWR|O_CREAT,0666);
                write(newfd,msgget.data,strlen(msgget.data));
                close(newfd);
                putchar('>');
                fflush(stdout);

        }else{
                printf("----------------------------\n");
                printf("\n%s\n",msgget.data);
                printf("----------------------------\n");
                putchar('>');
                fflush(stdout);
        }


}
int main(int argc,char **argv){
        if(argc!=3){
                perror("canshu:");
                exit(-1);

        }
        int c_fd;
        struct Msg msg;
        struct sockaddr_in c_addr;
        memset(&c_addr,0,sizeof(struct sockaddr_in));
        c_fd=socket(AF_INET,SOCK_STREAM,0);
        c_addr.sin_family=AF_INET;
        c_addr.sin_port=htons(atoi(argv[2]));
        inet_aton(argv[1],&c_addr.sin_addr);
        connect(c_fd,(struct sockaddr*)&c_addr,sizeof(struct sockaddr_in));
        printf("connect...\n");
        int mark=0;
        while(1){

                memset(&msg,0,sizeof(msg));
                if(mark==0){
                        printf(">");
                }
                gets(msg.data);

                if(strlen(msg.data)==0){
                        if(mark==1){
                                printf(">");
                        }
                        continue;
                }
                mark=1;
                int ret=cmd_handler(msg,c_fd);
                if(ret>IFGO){
                        putchar('>');
                        fflush(stdout);
                        continue;
                }
                else if(ret==-1){
                        printf("no this cmd\n");
                        printf(">");
                        fflush(stdout);
                        continue;

                }else{
                     handler_server_message(c_fd,msg);
                }
        }




        return 0;
}
                                                                                                                                                                                                                                





2.服务端 server.c

2. 服务端 server.c
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>          /* See NOTES */
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <string.h>
#include "config.h"
#include <sys/stat.h>
#include <fcntl.h>
int cmd_get(char *cmd){
        if(!strcmp(cmd,"ls")) return LS;
        if(!strcmp(cmd,"pwd")) return PWD;
        if(!strcmp(cmd,"quit")) return QUIT;

        if(strstr(cmd,"cd")!=NULL) return CD;
        if(strstr(cmd,"get")!=NULL) return GET;
        if(strstr(cmd,"put")!=NULL) return PUT;
        return 100;
}
char *getdir(char *cmd){
        char *p=NULL;
        p=strtok(cmd," ");
        p=strtok(NULL," ");
        return p;
}
void cmd_handler(struct Msg msg,int c_fd){
        int ret=cmd_get(msg.data);
        FILE *r;
        char *dir=NULL;
        char buf[1024]={0};
        int fd;
        printf("cmd:%s\n",msg.data);
        switch(ret){

                case LS:
                case PWD:
                        r=popen(msg.data,"r");

                        fread(msg.data,sizeof(msg.data),1,r);
                        fclose(r);
                        write(c_fd,&msg,sizeof(msg));
                        break;
                case QUIT:
                                printf("client quit\n");
                        exit(-1);
                        break;
                case CD:
                        dir=getdir(msg.data);
                        chdir(dir);
                        break;
                case GET:
                        dir=getdir(msg.data);

                        if(access(dir,F_OK)==-1){
                                strcpy(msg.data,"NO this file") ;
                                write(c_fd,&msg,sizeof(msg));
                        }else{
                                msg.type=DOFILE;
                                fd=open(dir,O_RDWR);
                                read(fd,msg.data,sizeof(msg.data));
                                close(fd);
                                write(c_fd,&msg,sizeof(msg));
                        }
                        break;
                case PUT:

                        dir=getdir(msg.data);

                        fd=open(dir,O_RDWR|O_CREAT,0666);
                        write(fd,msg.secondBuf,sizeof(msg.secondBuf));
                        close(fd);
                        break;

        }

}
int main(int argc,char **argv){

        int s_fd;
        int c_fd;
        int n_read;
        struct sockaddr_in s_addr;
        struct sockaddr_in c_addr;
        struct Msg msg;
        memset(&s_addr,0,sizeof(struct sockaddr_in));
        memset(&s_addr,0,sizeof(struct sockaddr_in));
        s_fd=socket(AF_INET,SOCK_STREAM,0);
        s_addr.sin_family=AF_INET;
        s_addr.sin_port=htons(atoi(argv[2]));
        inet_aton(argv[1],&s_addr.sin_addr);
        bind(s_fd,(struct sockaddr*)&s_addr,sizeof(struct sockaddr_in));
        listen(s_fd,10);


        int len=sizeof(struct sockaddr_in);
        while(1){
                c_fd=accept(s_fd,(struct sockaddr*)&c_addr,&len);
                if(c_fd==-1){
                        perror("connect:");
                        exit(-1);
                }
                printf("get connect IP:%s\n",inet_ntoa(s_addr.sin_addr));
                if(fork()==0){
                        while(1){

                                memset(&msg,0,sizeof(msg));
                                n_read=read(c_fd,&msg,sizeof(msg));
                                if(n_read==0){
                                        printf(" client quit\n");
                                        break;
                                }else{
                                        cmd_handler(msg,c_fd);
                                }
                        }

                }

        }
                                                                                                                                                          
  return 0;
}
  1. 头文件config.h
#define LS     0
#define GET    1
#define PWD    2

#define IFGO   3

#define LCD    4
#define LLS    5
#define CD     6
#define PUT    7 

#define QUIT   8
#define DOFILE 9
#define LPWD  10
struct Msg{
        int type;
        char data[1024];
        char secondBuf[1024];
};

### 回答1: TFTP(Trivial File Transfer Protocol)是一种简单的文件传输协议,使用UDP协议进行通信。在Linux环境下,我们可以使用C语言来实现TFTP功能。 首先,我们需要了解TFTP的基本工作原理。TFTP客户端(Client)和服务器(Server)之间通过UDP套接字进行通信,客户端通过发送和接收TFTP数据包来实现文件的上传和下载。 接下来,我们可以使用C语言编写TFTP的客户端和服务器端代码。在客户端代码中,我们需要实现以下功能: 1. 建立UDP套接字:使用socket函数创建UDP套接字来进行通信。 2. 创建请求包:根据TFTP协议,创建RRQ(Read Request)或WRQ(Write Request)数据包,包括请求的文件名和传输模式等信息。 3. 发送请求包:使用sendto函数将请求包发送给TFTP服务器。 4. 接收数据包:使用recvfrom函数接收服务器发送过来的TFTP数据包。 5. 解析数据包:根据TFTP协议,解析数据包中的操作码和数据块编号等信息。 6. 处理数据包:根据操作码的不同,执行相应的操作,如读取文件数据或发送文件数据给服务器。 7. 关闭套接字:使用close函数关闭UDP套接字。 在服务器端代码中,我们需要实现以下功能: 1. 建立UDP套接字:使用socket函数创建UDP套接字来进行通信。 2. 绑定地址:使用bind函数将套接字和服务器的IP地址以及端口号绑定在一起。 3. 接收请求包:使用recvfrom函数接收客户端发送过来的TFTP请求包。 4. 解析请求包:根据TFTP协议,解析请求包中的操作码、文件名和传输模式等信息。 5. 处理请求包:根据操作码的不同,执行相应的操作,如发送文件数据给客户端或将客户端发送过来的文件数据保存到服务器。 6. 发送数据包:使用sendto函数将TFTP数据包发送给客户端。 7. 关闭套接字:使用close函数关闭UDP套接字。 通过编写上述代码,我们便可以在Linux环境下使用C语言实现TFTP功能。这样,我们就可以通过TFTP协议进行简单的文件传输了。 ### 回答2: TFTP(Trivial File Transfer Protocol,简单文件传输协议)是一种简单的文件传输协议,主要用于在计算机网络中进行文件传输。 在Linux环境下,可以使用C语言编写TFTP客户端和服务器程序C语言是一种高级编程语言,广泛应用于系统编程和网络编程领域。 在使用C语言编写TFTP客户端时,可以使用socket编程库进行网络通信。通过创建一个TFTP请求数据包,发送给服务器,服务端接收请求数据包后会响应相应的数据包。客户端收到响应包后,可以对数据进行处理,如保存到本地文件。 而在编写TFTP服务器程序时,也需要使用socket编程库。服务器监听特定的端口,接收客户端发送的请求数据包。根据客户端请求的文件名和请求类型(读取或写入),服务器可以读取本地文件并发送给客户端,或者接收客户端发送的文件并保存到本地。 Linux提供了一些库函数和系统调用来简化TFTP程序编写,例如socket、bind、sendto、recvfrom等。可以通过调用这些函数来实现与TFTP协议相对应的数据包的创建、发送和接收,以及文件的读写。 总之,通过在Linux下使用C语言编写TFTP客户端和服务器程序,可以实现在计算机网络环境中进行简单文件传输的功能。这对于文件传输和网络通信有着重要的应用价值。 ### 回答3: TFTP(Trivial File Transfer Protocol,简单文件传输协议)是一种简化的文件传输协议,常用于局域网环境下进行文件传输。在Linux环境下,可以使用C语言编写TFTP客户端和服务器TFTP客户端使用C语言编写时,可以使用socket编程接口实现与TFTP服务器的通信。客户端通过网络连接到TFTP服务器,并发送TFTP请求(如读请求或写请求)以进行文件的传输。客户端还需实现TFTP协议中定义的RRQ(读请求)和WRQ(写请求)消息的封装和解析。 TFTP服务器使用C语言编写时,同样使用socket编程接口来监听TFTP客户端的请求。服务器收到客户端的请求后,根据请求的类型(读请求或写请求),进行相应的文件操作。服务器需要实现TFTP协议中定义的数据传输和错误处理等功能,保证文件的正确传输。 在Linux环境下,可以使用与TFTP相关的库函数来简化开发过程,如libtftp库。该库提供了一系列函数,如tftp_open、tftp_read、tftp_write等,用于实现TFTP协议的各个步骤。使用这些库函数的好处是可以减少开发人员的编码工作量,并提高代码的可读性。 总之,使用C语言编写LinuxTFTP客户端和服务器涉及到socket编程,需要实现TFTP协议的各个步骤,包括封装和解析TFTP请求消息、数据传输和错误处理等。通过使用与TFTP相关的库函数可以简化开发过程。编写TFTP程序需要具备一定的网络编程和文件操作的知识。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值