TFTP客户端1.TFTP协议介绍2.TFTP下载过程相关代码

1.TFTP协议介绍

TFTP(Trivial File Transfer Protocol,简单文件传输协议)

是TCP/IP协议族中的一个用来在客户端与服务器之间进行简单文件传输的协议。 TFTP是一个传输文件的简单协议,通常使用UDP协议而实现,但tftp并没有要求实现的具体协议,在特殊需求的场合可以同tcp实现。此协议设计的时候是进行小文件传输的。因此它不具备通常的FTP的许多功能,它只能从文件服务器上获得或写入文件,不能列出目录,不进行认证,它传输8位数据。传输中有三种模式:netascii,这是8位的ASCII码形式,另一种是octet,这是8位源数据类型;最后一种mail已经不再支持,它将返回的数据直接返回给用户而不是保存为文件。

特点:

  • 简单
  • 占用资源小
  • 适合传递小文件
  • 适合在局域网进行传递
  • 端口号为69
  • 基于UDP实现

2.TFTP下载过程

TFTP服务器默认监听69号端口 当客户端发送“下载”请求(即读请求)时,需要向服务器的69端口发送 服务器若批准此请求,则使用一个新的、临时的 端口进行数据传输

传输过程

当服务器找到需要现在的文件后,会立刻打开文件,把文件中的数据通过TFTP协议发送给客户端 如果文件的总大小较大(比如3M),那么服务器分多次发送,每次会从文件中读取512个字节的数据发送过来 因为发送的次数有可能会很多,所以为了让客户端对接收到的数据进行排序,所以在服务器发送那512个字节数据的时候,会多发2个字节的数据,用来存放序号,并且放在512个字节数据的前面,序号是从1开始的 因为需要从服务器上下载文件时,文件可能不存在,那么此时服务器就会发送一个错误的信息过来,为了区分服务发送的是文件内容还是错误的提示信息,所以又用了2个字节 来表示这个数据包的功能(称为操作码),并且在序号的前面

功能

TFTP数据包的格式

相关代码

pack 和unpack

import struct

data = struct.pack('!H8sb5sb',1,b'girl.jpg',0,b'octet',0)
print(type(data))
ret1 = struct.unpack('!H',data[:2])
print(ret1[0])

ret1 = struct.unpack('!8s',data[2:10])
print(ret1)
print(ret1[0].decode('utf-8'))

ret1 = struct.unpack('!H8sb5sb',data)
print(ret1)

download-01

import socket
import struct
import os
import time

def main():
    #创建文件
    myFile = open('girl.jpg','wb')
    #socket对象,发送下载的请求信息,接收信息和发送确认信息
    udpSocket = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
    #服务器地址
    destAddr = ('192.168.11.74',69)
    #pack信息
    data = struct.pack('!H8sb5sb', 1, b'girl.jpg', 0, b'octet', 0)
    #发送
    udpSocket.sendto(data,destAddr)
    #循环
    while True:
        time.sleep(0.2)
        #接收
        recvData,redvAddr = udpSocket.recvfrom(1024)
        print(redvAddr)
        #unpack信息,获取操作码
        operNum = struct.unpack('!H',recvData[:2])[0]
        #判断
        if operNum==3:
            # unpack信息,获取块编号
            blockNum = struct.unpack('!H', recvData[2:4])[0]
            print(blockNum)
            #获取数据,写到文件中
            myFile.write(recvData[4:])
            #准备ack数据
            ackData = struct.pack('!HH', 4,blockNum)
            #发送ack确认到服务器
            udpSocket.sendto(ackData,redvAddr)

            if len(recvData)<516:
                break

        if operNum==5:
            print('发生异常啦......')
            break
    #关闭文件
    myFile.close()

if __name__ == '__main__':
    main()

运行结果:

服务器的文件

>>

显然下载过后的文件是没问题的

download-02

import socket
import struct
import os
import time

def main():
    #创建文件
    myFile = open('Timor.avi','wb')
    #socket对象,发送下载的请求信息,接收信息和发送确认信息
    udpSocket = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
    #服务器地址
    destAddr = ('192.168.11.74',69)
    #pack信息
    data = struct.pack('!H23sb5sb',1,'冯提莫 - 刚好遇见你.m4r'.encode('gbk'),0,b'octet',0)
    #发送
    udpSocket.sendto(data,destAddr)
    #定义一个变量,记录写的次数
    num = 0
    #循环
    while True:
        #接收
        recvData,redvAddr = udpSocket.recvfrom(1024)

        #unpack信息,获取操作码
        operNum = struct.unpack('!H',recvData[:2])[0]
        #判断
        if operNum==3:
                # unpack信息,获取块编号
                blockNum = struct.unpack('!H', recvData[2:4])[0]
                print(blockNum)

                #判断
                num = num+1
                if num==65536:
                    num = 0

                if num == blockNum:
                    #获取数据,写到文件中
                    myFile.write(recvData[4:])
                    #准备ack数据
                    ackData = struct.pack('!HH', 4,blockNum)
                    #发送ack确认到服务器
                    udpSocket.sendto(ackData,redvAddr)
                num = blockNum

            if len(recvData)<516:
                break

        if operNum==5:
            print('发生异常啦......')
            break
    #关闭文件
    myFile.close()

if __name__ == '__main__':
    main()

运行结果:

原服务器中的文件

>>

下载后的文件

上传

import socket
import struct
import os


def main():
    #以读字节的方式打开文件
    myFile=open('xx.avi','rb')
    #创建socket UDP对象
    udpSocket = socket.socket(socket.AF_INET,socket.SOCK_DGRAM)
    destAddr = ('192.168.11.66',69)
    #组成上传文件请求包
    date = struct.pack('!H5sb5sb',2,b'x.avi',0,b'octet',0)
    #发送上传文件请求包
    udpSocket.sendto(date,destAddr)

    #数据包编号
    blockNum = 0
    while True:
        recvDate, recvAddr = udpSocket.recvfrom(1024)
        recvNum = struct.unpack('!H', recvDate[2:4])[0]
        print('recvNum:%s' % recvNum)
        date1 = myFile.read(512)
        # 判断是否有发送失败的情况
        if blockNum == recvNum:
            #发送数据包
            blockNum += 1
            if blockNum == 65536:
                blockNum = 0
            blockDate1 = struct.pack('!HH%ss'%len(date1),3,blockNum,date1)
            print('blockNum:%s' % blockNum)
            udpSocket.sendto(blockDate1, recvAddr)
            if len(date1)<512:
                break
    myFile.close()

if __name__ == '__main__':
    main()

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏aoho求索

基于可靠消息方案的分布式事务(四):接入Lottor服务

在上一篇文章中,通过Lottor Sample介绍了快速体验分布式事务Lottor。本文将会介绍如何将微服务中的生产方和消费方服务接入Lottor。

2441
来自专栏后端之路

activeMq消息转投rabbitMq研究

在研究activemq转投消息到rabbitmq的过程中还是发现了很多有趣的细节。 消息发送端分为PERSISTENT与NON_PERSISTENT,该类型表示...

2648
来自专栏云飞学编程

Python爬虫框架scrapy抓取旅行家网所有游记!从此出游不发愁!

安装scrapy,pip可以解决你的问题: pip install scrapy。

711
来自专栏大学生计算机视觉学习DeepLearning

c++ 网络编程(二)TCP/IP linux 下多进程socket通信 多个客户端与单个服务端交互代码实现回声服务器

原文链接:https://www.cnblogs.com/DOMLX/p/9612820.html

2688
来自专栏北京马哥教育

varnish学习总结

什么是web cache? Web缓存是指一个Web资源(如html页面,图片,js,数据等)存在与Web服务器和客户端(浏览器)直接的副本。缓存会根据进来...

3095
来自专栏崔庆才的专栏

分布式爬虫原理之Scrapy分布式实现

4186
来自专栏码字搬砖

工作中用到的sh脚本(持续更新)

day=(date−d‘−0day′‘+echo“(date−d‘−0day′‘+echo“(date -d ‘-0 day’ ‘+%Y-%m-%d’) e...

1575
来自专栏腾讯IVWEB团队的专栏

前端 fetch 通信

随着前端异步的发展, XHR 这种耦合方式的书写不利于前端异步的 Promise 回调。而且,写起来也是很复杂.。fetch API 本来是在 SW(Servi...

1.4K0
来自专栏JetpropelledSnake

Python Web学习笔记之socket套接字

套接字是为特定网络协议(例如TCP/IP,ICMP/IP,UDP/IP等)套件对上的网络应用程序提供者提供当前可移植标准的对象。它们允许程序接受并进行连接,如发...

3568
来自专栏智能计算时代

Envoy架构概览(10):热启动,动态配置,初始化,排水,脚本

热启动 易于操作是特使的主要目标之一。除了强大的统计数据和本地管理界面之外,Envoy还具有“热”或“实时”重启的能力。这意味着Envoy可以完全重新加载自己(...

3782

扫码关注云+社区