一.FTP任务目录:

       1. 多用户同时登陆:     socketserver

       2. 用户登陆,加密认证: md5加密

       3. 上传/下载文件,保证文件一致性:md5摘要

       4. 传输过程中现实进度条

  5. 不同用户家目录不同,且只能访问自己的家目录, 上传下载时,必须在自己目录: os.path.join('D:\sylar\nbftp_server\users',username,'a.text')

       6. 对用户进行磁盘配额、不同用户配额可不同:   上传、下载之前做文件夹大小的判断。

       7. 用户登陆server后,可在家目录权限下切换子目录: 

    客户端向服务端发命令,subprocess.popen(命令)

       8. 查看当前目录下文件,新建文件夹              : 执行命令

       9. 删除文件和空文件夹                          : 执行命令

       10. 充分使用面向对象知识+反射

       11. 支持断点续传                               :断点续传

二.FTP任务分析(进度条、计算文件大小、断点续传、搭建框架示例)

1、进度条(\r移动到行首、print不换行end=””)

    import time
    def jdt(now,all): # 进度条函数
        per = int(now / all * 100)
        print('\r%s %s%%' % ('*'*per , per) ,end='')
        time.sleep(0.05)

    for i in range(101):
        jdt(i,100) # 执行进度条函数

2、计算文件大小

  之前我们学过一种计算文件大小的方式:os.path.getsize(file_path),现在再来学习一种方式:

    import os
    size1 = os.path.getsize('server.py')
    size2 = os.stat('server.py').st_size
    print(size1,size2)  # 1844 1844

3、断点续传

  我们先来写一个断点续传(脚本主要实现了客户端向服务端上传文件,上传过程中中断的话,再次上传此文件时接着上次中断的地方继续上传)的简单示例,然后从中提取一些编程思想:

import os
    import json
    import socketserver
    import shutil

    CODE = {
        '1001':'上传文件,从头开始上传'
    }

    def upload(cmd_dict,conn,username):
        """
        服务端完成上传文件(含断点续传)
        :param cmd_dict:
        :param conn:
        :return:
        """
        # 2. 获取文件信息
        file_md5 = cmd_dict['md5']
        file_name = cmd_dict['file_name']

        file_md5_path = os.path.join(username, file_md5)
        file_name_path = os.path.join(username, file_name)
        upload_file_size = cmd_dict['size']

        # 3. 判断文件是否存在
        exist = os.path.exists(file_md5_path)
        if not exist:  # 不续传
            # 3.1.1 可以开始上传了,我已经准备好。
            response = {'code': 1001}
            conn.sendall(json.dumps(response).encode('utf-8'))

            # 3.1.2 接收上传的文件内容
            f = open(file_md5_path, 'wb')
            recv_size = 0
            while recv_size < upload_file_size:
                data = conn.recv(1024)
                f.write(data)
                f.flush()
                recv_size += len(data)
                return
            f.close()

            # 3.1.3 改名字
            shutil.move(file_md5_path, file_name_path)

        else:  # 续传
            # 3.2 续传+大小
            exist_size = os.stat(file_md5_path).st_size
            response = {'code': 1002, 'size': exist_size}
            conn.sendall(json.dumps(response).encode('utf-8'))

            f = open(file_md5_path, 'ab')
            recv_size = exist_size
            while recv_size < upload_file_size:
                data = conn.recv(1024)
                f.write(data)
                f.flush()
                recv_size += len(data)
            f.close()

            # 3.1.3 改名字
            shutil.move(file_md5_path, file_name_path)

    class NbServer(socketserver.BaseRequestHandler):
        def handle(self):
            """
            self.request 是客户端的socket对象
            :return:
            """
            # 1. 接收命令
            upload_cmd_bytes = self.request.recv(8096)
            cmd_dict = json.loads(upload_cmd_bytes.decode('utf-8'))

            if cmd_dict['cmd'] == 'upload':
                upload(cmd_dict,self.request,'lili')  # 服务端代码有个文件夹(lili)才能运行
            elif cmd_dict['cmd'] == 'download':
                pass

    if __name__ == '__main__':
        server =     socketserver.ThreadingTCPServer(('127.0.0.1',8001),NbServer)
        server.serve_forever()

服务端代码
server

相关文章: