第七周作业——简单FTP
开发简单的FTP:
1. 用户登陆
2. 上传/下载文件
3. 不同用户家目录不同
4. 查看当前目录下文件
5. 充分使用面向对象知识
1、目录结构
zuoye-ftp
├── chenliang #用户chenliang的FTP根目录
├── download #所有用户下载的时候默认文件路径
├── lw #用户lw的FTP根目录
├── lzd #用户lzd的FTP根目录
├── wangjun #用户wangjun的FTP根目录
├── wdl #用户wdl的FTP根目录
├── xueli #用户xueli的FTP根目录
├── create-users.py #创建用户专用py文件
├── ros.zip #模拟上下载的测试文件ros.zip
├── socket_client.py #客户端文件
├── socket_server.py #服务端文件
├── user_data.bak #shelve文件
├── user_data.dat #shelve文件
└── user_data.dir #shelve文件
2、创建用户运行create-users.py,默认帐号密码是一致的。已有用户chenliang,lw,lzd,wangjun,wdl,xueli
3、创建的用户,在执行服务端的时候,自动会生成一个自己用户的FTP根目录,同时便于测试会复制ROS.zip到自己根目录
4、先运行socket_server.py,默认socket的服务端端口号为6969
5、再运行socket_client.py
create-users.py
- #!/usr/bin/env python
- #-*-coding:utf-8-*-
- #Author:lzd
- #创建用户名和密码,默认用户名和密码一致
- import shelve
- userlist=[]
- user_data=shelve.open("user_data")
- # for i in user_data.items():
- # print(i)
- print("当前已有的所有用户的用户名如下:")
- for i in user_data.keys():
- userlist.append(user_data[i])
- print(userlist)
- while True:
- username=input("请输入用户名\n")
- if username in userlist:
- print("用户名重复啦,请重新输入用户名!")
- continue
- else:
- password=input("请输入用户密码\n")
- break
- user_data[username]=password
- user_data.close()
- print("用户创建成功~!")
socket_server.py
- #!/usr/bin/env python
- #-*-coding:utf-8-*-
- #Author:lzd
- #服务器端
- import os
- gongzuopath=os.getcwd()
- # print(gongzuopath)
- import time
- #print(time.strftime("%Y%m%d%H%M%S",time.localtime()))
- import socket
- server = socket.socket()
- server.bind(('localhost',6969)) #绑定要监听端口
- server.listen() #监听
- import shutil
- import shelve
- userlist=[]
- user_data=shelve.open("user_data")
- for i in user_data.keys():
- userlist.append(user_data[i])
- if not os.path.exists(gongzuopath+r"/"+i):
- os.mkdir(gongzuopath+r"/"+i)
- if not os.path.exists(gongzuopath+r"/"+i+"/ros.zip"):
- shutil.copyfile(gongzuopath+r"/ros.zip",gongzuopath+r"/"+i+"/ros.zip")
- print("我要开始等电话了")
- while True:
- Flag = True
- conn, addr = server.accept() # 等电话打进来
- # conn就是客户端连过来而在服务器端为其生成的一个连接实例
- print(conn, addr)
- print("电话来了")
- while Flag:
- data = conn.recv(1024)
- print("recv:",data)
- if not data:
- print("client has lost...")
- break
- clientuser=data.decode("utf-8")
- if clientuser in userlist:
- conn.send("请输入用户密码".encode("utf-8"))
- data = conn.recv(1024)
- print("recv:",data)
- if not data:
- print("client has lost...")
- break
- if data.decode("utf-8")==user_data[clientuser]:
- conn.send("用户名密码验证成功".encode("utf-8"))
- while Flag:
- data = conn.recv(1024)
- print("recv:", data)
- if data=="choose3".encode("utf-8"):
- if ",".join(os.listdir(gongzuopath+r"/"+clientuser)).encode("utf-8"):
- conn.send(",".join(os.listdir(gongzuopath+r"/"+clientuser)).encode("utf-8"))
- else:
- conn.send("当前目录下为空".encode("utf-8"))
- elif data=="choose1".encode("utf-8"):
- conn.send("服务端准备接收文件".encode())
- data = conn.recv(1024).decode("utf-8") #收到文件大小,单位byte
- filesize=data
- print("recv:文件大小为", data)
- tmpdata="服务端准备接收大小为"+data+"的文件"
- conn.send(tmpdata.encode())
- serverfile=gongzuopath+r"/"+clientuser+"/"+time.strftime("%Y%m%d%H%M%S",time.localtime())
- f = open(serverfile, "wb")
- while str(os.stat(serverfile).st_size) != filesize:
- data=conn.recv(10240000)
- f.write(data)
- f.flush()
- else:
- conn.send("文件发送完毕".encode("utf-8"))
- elif data=="choose2".encode("utf-8"):
- if ",".join(os.listdir(gongzuopath + r"/" + clientuser)).encode("utf-8"):
- conn.send(",".join(os.listdir(gongzuopath + r"/" + clientuser)).encode("utf-8"))
- else:
- conn.send("当前目录下为空".encode("utf-8"))
- continue
- data = conn.recv(1024).decode("utf-8")
- print(data)
- if not data:
- continue
- print("recv:客户端要下载的文件名", data)
- fpath=gongzuopath+r"/"+clientuser+r"/"+data
- fpath_size = str(os.stat(fpath).st_size)
- f = open(fpath, "rb")
- filedata = f.read()
- conn.send(fpath_size.encode("utf-8"))
- conn.sendall(filedata)
- data = conn.recv(1024)
- print(data.decode("utf-8")) # 文件接收完毕
- elif data=="choose4".encode("utf-8"):
- Flag=False
- else:
- conn.send("用户名密码验证失败".encode("utf-8"))
- else:
- conn.send("输入的用户名不存在".encode())
- server.close()
socket_client.py
- #!/usr/bin/env python
- #-*-coding:utf-8-*-
- #Author:lzd
- #客户端
- import socket
- import os
- gongzuopath=os.getcwd()
- # print(os.stat(gongzuopath+"/lzd/"+"ros.zip").st_size)
- client = socket.socket() #声明socket类型,同时生成socket连接对象
- client.connect(('localhost',6969))
- while True:
- username=input("请输入用户名\n")
- if len(username) == 0: continue
- client.send(username.encode("utf-8"))
- data=client.recv(1024)
- print(data.decode())
- if data.decode()=="请输入用户密码":
- password=input("")
- client.send(password.encode("utf-8"))
- data=client.recv(1024)
- print(data.decode())
- if data.decode()=="用户名密码验证失败":
- continue
- elif data.decode()=="用户名密码验证成功":
- while True:
- print("%s当前的工作目录为/%s"%(username,username))
- print("""
- 1、上传文件;
- 2、下载文件;
- 3、查看当前目录下的所有文件;
- 4、退出程序
- 请输入你想操作的数字?
- """)
- choose=input("")
- if choose=="":
- print("当前用户的工作目录下所有文件如下:")
- print(",".join(os.listdir(gongzuopath + r"/" + username)))
- fpath2=input("请输入你想上传到服务器的文件名,为便于测试,当前目录下有个1M的文件ros.zip\n")
- print("默认会在当前用户的目录下生成一个当前时间的文件。")
- if not fpath2:
- continue
- fpath=gongzuopath+r"/"+username+r"/"+fpath2
- fpath_size=str(os.stat(fpath).st_size)
- f=open(fpath,"rb")
- filedata=f.read()
- client.send("choose1".encode("utf-8"))
- data = client.recv(1024)
- print(data.decode("utf-8")) #服务器准备接收文件
- client.send(fpath_size.encode("utf-8"))
- data = client.recv(1024)
- print(data.decode("utf-8")) #服务端准备接收大小为%s的文件
- client.sendall(filedata)
- data = client.recv(1024)
- print(data.decode("utf-8")) # 文件发送完毕
- if choose=="":
- client.send("choose2".encode("utf-8"))
- data = client.recv(1024)
- print("您当前的工作目录下,所有的文件如下:")
- print(data.decode("utf-8")) # 显示当前工作目录下所有文件
- if data.decode("utf-8")=="当前目录下为空":
- continue
- print("您想下载哪个文件到本地?默认下载到当前脚本的工作目录的download目录")
- fpath2 = input("请输入你想下载的文件名\n")
- client.send(fpath2.encode("utf-8"))
- fpath = gongzuopath + r"/download/" + fpath2
- data = client.recv(1024).decode("utf-8")
- file_size=data
- print("您想下载的文件大小为:",data)
- f = open(fpath, "wb")
- while str(os.stat(fpath).st_size) != file_size:
- data = client.recv(10240000)
- f.write(data)
- f.flush()
- else:
- client.send("文件接收完毕".encode("utf-8"))
- print("文件接收完毕")
- if choose=="":
- client.send("choose3".encode("utf-8"))
- data = client.recv(1024)
- print("您当前的工作目录下,所有的文件如下:")
- print(data.decode("utf-8"))
- if choose=="":
- client.send("choose4".encode("utf-8"))
- client.close()
- exit()
第七周作业——简单FTP的更多相关文章
- 2017-2018-1 我爱学Java 第六七周 作业
团队六七周作业 完善版需求规格说明书 制定团队编码规范 数据库设计 后端架构设计 TODOList 参考资料 完善版需求规格说明书 <需求规格说明书>初稿不足之处: 1.开发工具写错 2. ...
- 2018-2019-1 20189221 《Linux内核原理与分析》第七周作业
2018-2019-1 20189221 <Linux内核原理与分析>第七周作业 实验六 分析Linux内核创建一个新进程的过程 代码分析 task_struct: struct task ...
- 2017-2018-1 JAVA实验站 第六、七周作业
2017-2018-1 JAVA实验站 第六.七周作业 详情请见团队博客
- 2017-2018-1 JaWorld 第六、七周作业
2017-2018-1 JaWorld 第六.七周作业 修改需求规格说明书 上次的<需求规格说明书>初稿有哪些不足? 王译潇同学回答: 1. 引言和目的性考虑的不是很周全. 2. ...
- 2017-2018-1 20179205《Linux内核原理与设计》第七周作业
<Linux内核原理与设计>第七周作业 视频学习及操作分析 创建一个新进程在内核中的执行过程 fork.vfork和clone三个系统调用都可以创建一个新进程,而且都是通过调用do_for ...
- 2019-2020-1 20199325《Linux内核原理与分析》第七周作业
第七周作业 1.进程描述符task_struct数据结构(一) 为了管理进程,内核必须对每个进程进行清晰的描述,进程描述符提供了内核所需了解的进程信息. struct task_struct数据结构很 ...
- 2019-2020-1 20199329《Linux内核原理与分析》第七周作业
<Linux内核原理与分析>第七周作业 一.本周内容概述: 对Linux系统如何创建一个新进程进行追踪 分析Linux内核创建一个新进程的过程 二.本周学习内容: 1.学习进程的描述 操作 ...
- 2020-2021-1 20209307《Linux内核原理与分析》第七周作业
这个作业属于哪个课程 <2020-2021-1Linux内核原理与分析)> 这个作业要求在哪里 <2020-2021-1Linux内核原理与分析第七周作业> 这个作业的目标 & ...
- 1903021116—吉琛—Java第七周作业—客户类测试
项目 内容 课程班级博客链接 19信计班 这个作业要求链接 第七周作业链接 博客名称 学号-姓名-Java第七周作业-客户类测试 要求 每道题要有题目,代码(使用插入代码,不会插入代码的自己查资料解决 ...
随机推荐
- LeetCode OJ:Search a 2D Matrix II(搜寻二维矩阵)
Write an efficient algorithm that searches for a value in an m x n matrix. This matrix has the follo ...
- 网页宽高clientWidth clientHeight获得数值不对的问题
当网页内容撑不满一屏时,通过以下代码获得整个网页高度会有问题 document.body.clientHeight;document.body.clientWidth; 得到的宽高不对,可能是因为ht ...
- Maven入门-2.Maven一些核心概念介绍
1.Maven仓库2.Maven坐标3.Maven插件和目标4.Maven生命周期4.1 clean:清理项目4.2 default:构建项目(重要)4.3 site:建立项目站点 1.Maven仓库 ...
- oracle和sql server 比较
Oracle SQLServer 比较 字符数据类型 CHAR CHAR 都是固定长度字符资料但oracle里面最大度为2kb,SQLServer里面最大长度为8kb 变长字符数据类型 ...
- C语言编程的两个工具:valgrind和core
检查内存泄漏: valgrind --leak-check=full ./ecox_rws_helper 来检查内存泄漏 程序崩溃看错误: ulimit -c unlimited 然后执行程序,会在当 ...
- Dom4j下载及使用Dom4j读写XML简介(转)
Dom4j下载及使用Dom4j读写XML简介 要使用dom4j读写XML文档,需要先下载dom4j包,dom4j官方网站在 http://www.dom4j.org/目前最新dom4j包下载地址:ht ...
- Android 仿微信朋友圈查看
项目要做一个类似于这样的功能,就做了. 项目下载地址:http://download.csdn.net/detail/u014608640/9917626 一,看下效果: 二.activity类 pu ...
- 《Drools7.0.0.Final规则引擎教程》第4章 4.4 LHS简介&Pattern
LHS简介 在规则文件组成章节,我们已经了解了LHS的基本使用说明.LHS是规则条件部分的统称,由0个或多个条件元素组成.前面我们已经提到,如果没有条件元素那么默认就是true. 没有条件元素,官方示 ...
- 服务器重装和配置:Ubuntu16.04 + Anaconda3 + GTX1080驱动 + CUDA8 + cuDNN + 常用工具安装
前一篇[基于Ubuntu16.04的GeForce GTX 1080驱动安装,遇到的问题及对应的解决方法]是在机器原有系统上安装GPU驱动,后来决定备份数据后重装系统,让服务器环境更干净清爽. 1.安 ...
- 深入理解java虚拟机-第13章-线程安全与锁优化
第十三章 线程安全与锁优化 线程安全 java语言中的线程安全 1 不可变.Immutable 的对象一定是线程安全的 2 绝对线程安全 一个类要达到不管运行时环境如何,调用者都不需要额外的同步措施, ...