详解Python3的TFTP文件传输
作者:Le-Bill 发布时间:2023-06-01 22:29:17
标签:Python3,TFTP
TFTP文件传输
功能:
1、获取文件列表
2、上传文件
3、下载文件
4、退出
第一部分,TftpServer部分。
①导入相关模块
from socket import *
import os
import signal
import sys
import time
②确定文件路径
# 文件库路径
FILE_PATH = "/home/tarena/"
③建立一个类,用来实现服务器功能模块
class TftpServer(object):
def __init__(self, connfd):
self.connfd = connfd
def do_list(self):
# 获取列表
file_list = os.listdir(FILE_PATH)
# 如果对应的路径内没有文件,返回Empty
if not file_list:
self.connfd.send('Empty'.encode())
return
# 路径存在文件,向客户端发送OK
else:
self.connfd.send(b'OK')
time.sleep(0.1)
files = ""
for file in file_list:
# 排除以'.'开头的隐藏文件
if file[0] != '.' and \
os.path.isfile(FILE_PATH + file):
files = files + file + '#'
# 返回文件列表
self.connfd.send(files.encode())
# 下载文件功能
def do_get(self, filename):
try:
fd = open(FILE_PATH + filename, 'rb')
except:
self.connfd.send("File doesn't exist".encode())
return
# 如果能正常打开,发送OK
self.connfd.send(b"OK")
time.sleep(0.1)
# 开始发送文件
try:
for line in fd:
self.connfd.send(line)
fd.close()
except Exception as e:
print(e)
time.sleep(0.1)
self.connfd.send(b'##')
print("File send over")
# 开始上传文件
def do_put(self, filename):
try:
fd = open(FILE_PATH + filename, 'w')
except:
self.connfd.send("Some error")
# 如果能正常打开文件,则发送OK
self.connfd.send(b'OK')
# 开始发送
while True:
# data为文件内容
data = self.connfd.recv(1024).decode()
if data == "##":
break
fd.write(data)
fd.close()
print("上传完毕")
④主流程控制
def main():
# 创建套接字/地址/端口
HOST = '0.0.0.0'
PORT = 8888
ADDR = (HOST, PORT)
sockfd = socket()
# 设置端口可重用
sockfd.setsockopt(SOL_SOCKET, SO_REUSEADDR, 1)
# 绑定地址
sockfd.bind(ADDR)
# 设置监听队列大小
sockfd.listen(5)
signal.signal(signal.SIGCHLD, signal.SIG_IGN)
print("Listen to port 8888....")
while True:
try:
connfd, addr = sockfd.accept()
except KeyboardInterrupt:
sockfd.close()
sys.exit("Server exit")
except Exception as e:
print(e)
continue
print("Client login:", addr)
# 创建父子进程
pid = os.fork()
if pid < 0:
print("Process creation failed")
continue
elif pid == 0:
# 子进程负责请求接收和发送,所以节省资源,关闭连接套字
sockfd.close()
tftp = TftpServer(connfd)
# 接收客户端请求
while True:
data = connfd.recv(1024).decode()
if not data:
continue
# 调用do_list方法获取文件列表
elif data[0] == 'L':
tftp.do_list()
# data ==> G filename
# 文件名以G开头,以空格为间隔发送过来
elif data[0] == 'G':
filename = data.split(' ')[-1]
tftp.do_get(filename)
elif data[0] == 'P':
filename = data.split(' ')[-1]
tftp.do_put(filename)
elif data[0] == 'Q':
print("客户端退出")
sys.exit(0)
else:
connfd.close()
continue
⑤运行主控制流程,等待客户端连接
if __name__ == "__main__":
main()
第二部分,TftpClient
①导入相关模块
from socket import *
import sys
import time
②实现基本的请求功能
class TftpServer(object):
def __init__(self, sockfd):
self.sockfd = sockfd
def do_list(self):
self.sockfd.send(b"L") # 发送请求类型
# 等待接收服务器端确认
data = self.sockfd.recv(1024).decode()
if data == 'OK':
data = self.sockfd.recv(4096).decode()
files = data.split('#')
for file in files:
print(file)
print("%%%%%There is file list%%%%%\n")
else:
# 失败的原因由服务器发送过来
print(data)
def do_get(self, filename):
self.sockfd.send(('G '+filename).encode())
data = self.sockfd.recv(1024).decode()
if data == 'OK':
fd = open(filename, 'w')
while True:
data = self.sockfd.recv(1024).decode()
if data == "##":
break
fd.write(data)
fd.close()
print("%s Download over\n" % filename)
else:
print(data)
def do_put(self, filename):
try:
fd = open(filename, 'rb')
except:
print("There is no such file")
return
self.sockfd.send(("P " + filename).encode())
data = self.sockfd.recv(1024).decode()
if data == 'OK':
for line in fd:
self.sockfd.send(line)
fd.close()
time.sleep(0.1)
self.sockfd.send(b'##')
print("%s upload over" % filename)
else:
print(data)
def do_quit(self):
self.sockfd.send(b'Q')
③主流程控制
# 套接字连接
def main():
if len(sys.argv) < 3:
print("argv is error")
return
HOST = sys.argv[1]
PORT = int(sys.argv[2])
ADDR = (HOST, PORT)
sockfd = socket()
sockfd.connect(ADDR)
tftp = TftpServer(sockfd) # tftp对象调用请求方法
while True:
print("=======命令选项========")
print("******* list *********")
print("*******get file ******")
print("*******put file ******")
print("******* quit *********")
print("======================")
cmd = input("请输入命令>>")
if cmd.strip() == 'list':
tftp.do_list()
elif cmd[:3] == "get":
filename = cmd.split(' ')[-1]
tftp.do_get(filename)
elif cmd[:3] == "put":
filename = cmd.split(' ')[-1]
tftp.do_put(filename)
elif cmd.strip() == "quit":
tftp.do_quit()
sockfd.close()
sys.exit("Welcome")
else:
print("Enter the right order!!!")
continue
④运行客户端
if __name__ == "__main__":
main()
第三部分,展示
一下就不做逐一显示,如有问题,烦请之处修正,共同进步!
来源:https://blog.csdn.net/weixin_42122355/article/details/80791857


猜你喜欢
- 在DreamWeaver中编写CSS,这种编写习惯本站(twocity.cn)并不提倡,不过由于"可视化"和操作简便,使
- 1.from_unixtime的语法及用法(1)语法:from_unixtime(timestamp ,date_format)即from_
- 安装环境:python版本2.7.5 ,win7系统安装Djangohttps://www.djangoproject.com/downlo
- dotnet run 介绍dotnet 相关命令是属于 .NET Core command-line (CLI) 的一部分,Microsof
- 你有没有觉得你的CSS样式表文件过于臃肿?其实如果你注意并培养一些比较好的CSS书写习惯,我想你的CSS样式表过于”肥胖”的问题会得到很好的
- python的子进程嘛,就是利用python打开一个子进程(当然像是一句废话),但是可能和我们理解的不太一样。一:如何理解?我们可能的理解:
- Pytest-sugar是一款用来改善控制台显示的插件,增加了进度条显示,使得在用例执行过程中可以看到进度条,而且进度条是根据用例是否通过标
- 要注意的是记得要先引用element操作模块 ,否则是无法绑定的格式:$(document).on(事件,标识,function(){});
- mysql中replace函数替换字符串介绍下mysql的REPLACE()函数语法:REPLACE()函数的语法如下:REPLACE(st
- 本次测试基于MovieLens数据集实现的基于物品的协同过滤,目前只是在小样本上实现,主要问题是计算太耗内存,后期代码继续优化与完善。 数据
- Python的matplotlib包可以轻松的将数据可视化,博主最近遇到了一个问题,博主想同时在两个窗口展示两张图,但是代码运行结果总是显示
- 快速入门模块提供三个类来处理一对一映射类型的一些操作'bidict', 'inverted', 'n
- 可以,具体说明和代码见下: <%@ Language=VBScript %><%Option
- vue在data中定义图片相对路径:data() { return { active: 1, ico
- Anaconda安装安装步骤:1、官网下载安装包:https://www.anaconda.com/distribution/2、运行并选择
- 在服务器上,通常为了快速登录数据库,我们会使用mysql -hhost -uusername -ppassword db的方式登录数据库,如
- 通过?pandas.DataFrame.shift命令查看帮助文档Signature: pandas.DataFrame.shift(sel
- 一、简介说明:索引是数据库对象之一,用于加快数据的检索,类似于书籍的索引。在数据库中索引可以减少数据库程序查询结果时需要读取的数据量,类似于
- 本文实例讲述了python迭代器的简单用法,分享给大家供大家参考。具体分析如下:生成器表达式是用来生成函数调用时序列参数的一种迭代器写法生成
- 前言在Django中有大量的通用类视图,例如ListView,DetailView,CreateView,UpdateView等等,将所有重