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

要求:

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

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;
}

  • 16
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值