python簡單實(shí)現(xiàn)TFTP協(xié)議下的上傳下載

首先,TFTP協(xié)議遵循以下格式:

tftp協(xié)議

我們只需要根據(jù)這些協(xié)議, 來決定我們每次請求數(shù)據(jù)是什么

構(gòu)造請求數(shù)據(jù)時(shí)的注意點(diǎn):(struct的格式)
例如

fileName = "123456.jpg"
dlreq = struct.pack("!H10sb5sb", 1, fileName.encode("utf-8"), 0, b"octet", 0)

注意: "!H10sb5sb"中, H標(biāo)示兩個(gè)字節(jié), s表示一個(gè)字節(jié), b標(biāo)示1個(gè)字節(jié). 具體含義如下圖:

fmt的格式
  • 注1.q和Q只在機(jī)器支持64位操作時(shí)有意思

  • 注2.每個(gè)格式前可以有一個(gè)數(shù)字,表示個(gè)數(shù)

  • 注3.s格式表示一定長度的字符串,4s表示長度為4的字符串,但是p表示的是pascal字符串

  • 注4.P用來轉(zhuǎn)換一個(gè)指針,其長度和機(jī)器字長相關(guān)

  • 注5.最后一個(gè)可以用來表示指針類型的,占4個(gè)字節(jié)

符號(hào)fmt

然后上代碼:

from socket import *
import struct
import sys
from threading import Thread

g_ip = ""
g_port = 0



def downLoad():

    g_socket = socket(AF_INET, SOCK_DGRAM)
    g_socket.bind(("", 9090))
    fileName = input("請輸入要下載的文件名: ")
    fileNameLen = len(fileName)
    # 下載請求格式控制字符串
    dlreqStr = "!H" + str(fileNameLen) + "sb5sb"
    # 構(gòu)造下載請求數(shù)據(jù)
    dlreq = struct.pack(dlreqStr, 1, fileName.encode("utf-8"), 0, b"octet", 0)

    sendAdd = (g_ip, g_port)
    g_socket.sendto(dlreq, sendAdd)

    p_Num = 0
    recvfile = ""

    while True:
        recvData, recvAddr = g_socket.recvfrom(1024)
        recvdataLen = len(recvData)
        cmd_tuple = struct.unpack("!HH", recvData[:4])
        cmd = cmd_tuple[0]
        currentPackNum = cmd_tuple[1]
        if cmd == 3:
            if currentPackNum == 1:
                recvfile = open(fileName, "wb")

            if p_Num + 1 == currentPackNum:
                recvfile.write(recvData[4:])
                p_Num += 1
                print("第 %d 次接收到數(shù)據(jù)"%p_Num)

                ackBuf = struct.pack("!HH", 4, p_Num)
                g_socket.sendto(ackBuf, recvAddr)

            if recvdataLen < 516:
                recvfile.close()
                print("已經(jīng)下載成功!")
                break

        elif cmd == 5:
            print("錯(cuò)誤!! 錯(cuò)誤塊: %d"%currentPackNum)
            break

    recvfile.close()
    g_socket.close()



def upLoad():
    upSocket = socket(AF_INET, SOCK_DGRAM)
    num = 0
    
    fileName = input("輸入要上傳的文件:")
    sendAdd = (g_ip, g_port)

    fileNameLen = len(fileName)
    # 上傳請求格式控制字符串
    ulreqStr = "!H" + str(fileNameLen) + "sb5sb"
    # 構(gòu)造下載請求數(shù)據(jù)
    ulreq = struct.pack(ulreqStr, 2, fileName.encode("utf-8"), 0, b"octet", 0)

    upSocket.sendto(ulreq, sendAdd)

    recvData1 = upSocket.recvfrom(1024)
    rand_port = recvData1[1][1]
    ack_num = struct.unpack("!H",recvData1[0][2:4])
    sendAdd = (g_ip, rand_port)

    try:
        fileRead = open(fileName, 'rb')
    except:
        errorData = struct.pack("!HHHb", 5, 5, 5, num)
        upSocket.sendto(errorData, sendAdd)
        exit() # 退出線程

    while True:
        readData = fileRead.read(512)
        sendData = struct.pack("!HH", 3, num) + readData
        upSocket.sendto(sendData, sendAdd)

        if len(sendData) < 516:
            print("傳輸成功!!")
            break

        recv_ack = upSocket.recvfrom(1024)[0]
        recv_cmd, ack_num = struct.unpack("!HH", recv_ack)

        print("接收到了 %d", ack_num)

        num +=1

        if int(recv_cmd) != 4 or int(ack_num != num - 1):
            break
    upSocket.close()
    fileRead.close()





def main():

    global g_ip
    global g_port


    g_ip = input("請輸入ip地址: ")
    g_port = int(input("請輸入端口號(hào): "))

    t1 = Thread(target = upLoad)

    t1.start()
    t1.join()
    
if __name__ == "__main__":
    main()
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容