首页 > 其他分享 >用UDP协议实现跨主机文件传输,实现下载与上传文件(FTFP)

用UDP协议实现跨主机文件传输,实现下载与上传文件(FTFP)

时间:2024-08-19 20:54:48浏览次数:18  
标签:UDP return FTFP cfd 文件传输 int printf buf sin

要求:

实现下载服务器目录上任意文件与上传本地文件到服务器特定目录下

tftp协议概述

简单文件传输协议,适用于在网络上进行文件传输的一套标准协议,使用UDP传输

特点:

是应用层协议

基于UDP协议实现

数据传输模式:

octet:二进制模式(常用)

服务器端:

tftp下载模型

TFTP通信过程总结

  1. 服务器在69号端口等待客户端的请求
  2. 服务器若批准此请求,则使用 临时端口 与客户端进行通信。
  3. 每个数据包的编号都有变化(从1开始)
  4. 每个数据包都要得到ACK的确认,如果出现超时,则需要重新发送最后的数据包或ACK包
  5. 数据长度以512Byte传输的,小于512Byte的数据意味着数据传输结束。

3)tftp协议分析

差错码:

0 未定义,差错错误信息

1 File not found.

2 Access violation.

3 Disk full or allocation exceeded.

4 illegal TFTP operation.

5 Unknown transfer ID.

6 File already exists.

7 No such user.

8 Unsupported option(s) requested.

客户端:linux

头文件:

#ifndef TFTP_H
#define TFTP_H
#include <myhead.h>

#define PORT 69 //服务器绑定的端口号
//下载
int download(int cfd, struct sockaddr_in sin);
//上传
int upload(int cfd, struct sockaddr_in sin);

#endif

程序文件:

#include "TFTP.h"

int download(int cfd, struct sockaddr_in sin)
{
    //组包准备发送下载请求
    char buf[516] = "";
    char name[20] = "";
    printf("请输入要下载的文件名>>> ");
    scanf("%s", name);
    while (getchar() != 10)
        ;

    unsigned short *p1 = (short *)buf; //操作码
    *p1 = htons(1);

    char *p2 = buf + 2; //文件名
    strcpy(p2, name);

    char *p3 = p2 + strlen(p2); //第一个0
    *p3 = 0;

    char *p4 = p3 + 1; //模式
    strcpy(p4, "octet");

    size_t size = 2 + strlen(p2) + 1 + strlen(p4) + 1; //操作码+文件名+0+模式+0

    //发送下载请求
    if (sendto(cfd, buf, sizeof(buf), 0, (struct sockaddr *)&sin, sizeof(sin)) < 0)
    {
        perror("sendto error");
        return -1;
    }

    //创建下载文件并清空
    int fd = -1; //必须初始化成一个无效的文件描述符
    socklen_t addrlen = sizeof(sin);
    ssize_t res = 0;
    unsigned short num = 0; //记录本地的块编号

    //发送下载请求
    while (1)
    {
        //接收数据
        bzero(buf, sizeof(buf));
        res = recvfrom(cfd, buf, sizeof(buf), 0, (struct sockaddr *)&sin, &addrlen);
        if (res < 0)
        {
            perror("recvfrom error");
            return -1;
        }

        //printf("%d号数据包\n", num);
        if (3 == buf[1]) //数据包
        {
            //判断服务器返回的数据包的块编号与本地记录的块编号是否一致
            if (*(unsigned short *)(buf + 2) == htons((num + 1)))
            {
                num++; //更新本地记录的块编号
                if (-1 == fd)
                {
                    fd = open(name, O_WRONLY | O_CREAT | O_TRUNC, 0664);
                    if (fd < 0)
                    {
                        perror("open error");
                        return -1;
                    }
                }

                //将数据写到文件中
                if (write(fd, buf + 4, res - 4) < 0)
                {
                    perror("write error");
                    return -1;
                }

                //发送ACK
                buf[1] = 4;
                //*p1=htons(4);
                if (sendto(cfd, buf, 4, 0, (struct sockaddr *)&sin, sizeof(sin)) < 0)
                {
                    perror("sendto error");
                    return -1;
                }
                //若接收到的数据小于512跳出循环,结束下载
                if (res - 4 < 512)
                {
                    printf("%s 文件下载完毕\n", name);
                    break;
                }
            }
        }
        else if (5 == buf[1]) //错误包
        {
            printf("错误: %d %s\n", ntohs(*(short *)(buf + 2)), buf + 4);
            close(fd);
            return -1;
        }
    }
    close(fd);
    return 0;
}

int upload(int cfd, struct sockaddr_in sin)
{
    //组包准备发送上传请求
    char buf[516] = ""; //512+4
    char name[20] = "";
    printf("请输入要上传的文件名>>> ");
    scanf("%s", name);
    while (getchar() != 10)
        ;

    int fd = open(name, O_RDONLY);
    if (fd < 0)
    {
        if (errno == ENOENT)
        {
            printf(">>>文件不存在,请重新输入<<<\n");
            return -2;
        }
        else
        {
            perror("open error");
            return -1;
        }
    }
    //组包准备发送上传请求
    unsigned short *p1 = (short *)buf; //操作码
    *p1 = htons(2);

    char *p2 = buf + 2; //文件名
    strcpy(p2, name);

    char *p3 = p2 + strlen(p2); //第一个0
    *p3 = 0;

    char *p4 = p3 + 1; //模式
    strcpy(p4, "octet");

    size_t size = 2 + strlen(p2) + 1 + strlen(p4) + 1; //操作码+文件名+0+模式+0

    //发送上传请求
    if (sendto(cfd, buf, sizeof(buf), 0, (struct sockaddr *)&sin, sizeof(sin)) < 0)
    {
        perror("sendto error");
        return -1;
    }

    //循环接收发送数据包
    ssize_t res;
    unsigned short num = 0;
    socklen_t addrlen = sizeof(sin);
    while (1)
    {
        //将数据从文件中读取到buf中
        bzero(buf, sizeof(buf));
        res = recvfrom(cfd, buf, sizeof(buf), 0, (struct sockaddr *)&sin, &addrlen);
        if (res < 0)
        {
            perror("recvfrom error");
            return -1;
        }
        //printf("%d号数据包\n", num);

        //操作码的范围是1-5,因为是网络字节序
        //所以有效操作码存储在高位,即buf[1]的位置
        if (4 == buf[1]) // 服务器返回应答包
        {
            //判断当前数据包的编号是否等于应答包的编号
            if (num == ntohs(*(unsigned short *)(buf + 2)))
            {
                //修改操作码为数据包
                buf[1] = 3;
                //填充块编号
                num++;
                *(unsigned short *)(buf + 2) = htons(num);

                //读取数据
                res = read(fd, buf + 4, sizeof(buf) - 4);
                if (res < 0)
                {
                    perror("read error");
                    return -1;
                }
                else if (0 == res)
                {
                    printf("%s 文件上传完毕!\n", name);
                    break;
                }

                //发送数据包
                //发送的数据包大小为,读取到的字节数res+操作码2byte+块编号2bytes
                if (sendto(cfd, buf, sizeof(buf), 0, (struct sockaddr *)&sin, sizeof(sin)) < 0)
                {
                    perror("sendto error");
                    return -1;
                }
            }
            else
            {
                printf("文件上传失败,请检查网络环境\n");
                break;
            }
        }
        else if (5 == buf[1]) //错误包
        {
            printf("错误: %d %s\n", ntohs(*(short *)(buf + 2)), buf + 4);
            close(fd);
            return -1;
        }
    }
    close(fd);
    return 0;
}

主函数:
 

#include "TFTP.h"

int main(int argc, const char *argv[])
{
    if (argc < 2)
    {
        printf("使用方式:%s 服务器IP\n", argv[0]);
        return 0;
    }
    if (strlen(argv[1]) >= 16 || strlen(argv[1]) <= 12)
    {
        printf(" IP 地址错误\n");
        return -1;
    }
    char IP[16];
    strcpy(IP, argv[1]); // 复制 IP 地址
    //创建报式套接字
    int cfd = socket(AF_INET, SOCK_DGRAM, 0);
    if (cfd < 0)
    {
        perror("socket");
        return -1;
    }
    printf("cfd = %d\n", cfd);
    //将端口号快速重用
    int reuse = 1;
    if (setsockopt(cfd, SOL_SOCKET, SO_REUSEADDR, &reuse, sizeof(reuse)) == -1)
    {
        perror("setsockopt error");
        return -1;
    }
    printf("端口号快速重用成功\n");

    //绑定客户端的地址信息结构体到套接字上--->非必须绑定
    //若不绑定,则操作系统会给客户端绑定运行主机的IP和随机的端口号
    //填充要连接的服务器地址信息结构体,真实的地址信息结构体根据地址族制定
    //要发给谁,就填谁的地址信息
    //AF_INET : man 7 ip
    struct sockaddr_in sin;
    socklen_t addrlen = sizeof(sin);
    sin.sin_family = AF_INET;            //必须填AF_INET
    sin.sin_port = htons(PORT);          //端口号:服务器绑定的端口号
    sin.sin_addr.s_addr = inet_addr(IP); //服务器绑定的IP

    char choose = 0;
    while (1)
    {
        printf("%s", "\033[1H\033[2J"); //在Terminal中打印这个就是清屏
        printf("------------------------\n");
        printf("---------1. 下载--------\n");
        printf("---------2. 上传--------\n");
        printf("---------3. 退出--------\n");
        printf("------------------------\n");
        printf("请输入>>> ");

        choose = getchar();
        while (getchar() != 10)
            ; //循环获取字符,直到遇到\n结束循环

        switch (choose)
        {
        case '1':
            download(cfd, sin);
            break;
        case '2':
            upload(cfd, sin);
            break;
        case '3':
            //关闭文件描述符
            close(cfd);
            return 0;
            break;
        default:
            printf("输入错误\n");
        }
        printf("\n3秒后清屏\n");
        sleep(3);
    }
    //关闭文件描述符
    close(cfd);
    return 0;
}

标签:UDP,return,FTFP,cfd,文件传输,int,printf,buf,sin
From: https://blog.csdn.net/sjrhsk_hahaha/article/details/141306074

相关文章

  • 实现网络聊天室(UDP)
    项目需求:如果有用户登录,其他用户可以收到这个人的登录信息如果有人发送信息,其他用户可以收到这个人的群聊信息如果有人下线,其他用户可以收到这个人的下线信息服务器可以发送系统信息服务器端:#include<myhead.h>structsockaddr_inserveraddr,caddr;enumtype_t//枚举{......
  • tcp与udp的总结+connect阻塞+tcp三次握手、四次挥手+常见的服务器IO(发送数据+接收数
    一,TCP与UDP的基本总结TCP(传输控制协议)和UDP(用户数据报协议)是两种主要的传输层协议。TCP是面向连接的,提供可靠、顺序的传输,适用于需要高可靠性的应用,如网页浏览和文件传输。它通过重传机制和流量控制确保数据完整性。UDP是无连接的,速度快但不保证数据的可靠性和顺序,适用于对实时性......
  • 【面试】阐述TCP和UDP的区别
    面试模拟场景面试官:你能阐述一下TCP和UDP的区别吗?###参考回答示例1.连接性TCP:面向连接(Connection-Oriented):TCP是一种面向连接的协议,在传输数据之前需要建立连接。在TCP通信过程中,客户端和服务器之间通过“三次握手”来建立连接,然后再进行数据传输,确保两者之间的......
  • UDP的可靠性传输——KCP
    目录一:TCP是怎么做到可靠的?1、UDP和TCP的区别 2、ARQ协议(AutomaticRepeat-reQuest)2.1、ARQ协议-停等式(stop-and-wait)2.2、ARQ协议-回退n帧(go-back-n)2.3、ARQ协议-选择性重传(selectiverepeat)3、RTT和RTO4、流量控制5、如何进行流量控制(滑动窗口)6、流量控......
  • 【TCP/IP】UDP协议数据格式和报文格式
    学习一个网络协议,主要就是学习“数据格式”/“报文格式”源端口/目的端口端口号是属于传输层的概念UDP报头使用两个自己的长度来表示端口号之所以端口号的范围是0~65535,是因为底层网络协议做出了强制要求如果使用一个10w这样的端口,就会在系统底层被“截断”UDP......
  • 浅谈TCP协议、UDP协议
    一、介绍说明TCP(传输控制协议)面向连接:TCP在数据传输之前必须建立连接。这通过一个称为三次握手的过程来完成,确保连接的两端都准备好进行数据传输。可靠性:TCP提供可靠的数据传输,确保数据包正确无误地到达目的地。如果数据包在传输过程中丢失或损坏,TCP会重新发送这些数据包......
  • TCP/UDP网络聊天室
        本博客仅对网络聊天室项目进行分享,仅供学习讨论使用,欢迎大家讨论。UDP网络聊天室项目要求        利用UDP协议,实现一套聊天室软件。服务器端记录客户端的地址,客户端发送消息后,服务器群发给各个客户端软件,服务器也可以自己发送通知给所有客户端。  ......
  • 【网络】UDP回显服务器和客户端的构造,以及连接流程
    回显服务器(EchoServer)最简单的客户端服务器程序,不涉及到业务流程,只是对与API的用法做演示客户端发送什么样的请求,服务器就返回什么样的响应,没有任何业务逻辑,没有进行任何计算或者处理0.构造方法网络编程必须要使用网卡,就需要用到Socket对象创建一个DatagramS......
  • 第六章 网络互连与互联网(五):TCP 和 UDP 协议
    五、TCP和UDP协议在TCP/IP协议簇中有两个传输协议,即传输控制协议(TCP)和用户数据报协议(UDP)。TCP是面向连接的,而UDP是无连接的。1、TCP服务(1)TCP协议提供面向连接的、可靠的传输服务,适用于各种可靠的或不可靠的网络。(2)TCP用户送来的是字节流形式的数据,这些数据缓存......
  • Python 通过UDP传输超过64k的信息
    在UDP中,单个数据包的最大尺寸通常受到网络层的限制,这通常被称为最大传输单元(MTU)。在以太网环境中,标准的MTU大小通常为1500字节。尽管有些网络环境可能支持更大的数据包,但是UDP数据包的理论最大限制是65535字节(64KB),这是由于UDP头部的16位长度字段决定的。然而,如果你需要发送超过这......