Skip to content

about

本篇贴出手动使用多线程实现多用户ftp和使用socketserver实现ftp。

client

无论是手动实现的server,还是使用socketserver实现的server,client代码都不变。

client
python
#!/bin/bash
# -*- coding: utf-8 -*-


import socket
import struct
import json


class Client(object):

    def __init__(self, server_address):
        self.server_address = server_address
        self.connect_socket()

    def _mk(self, data):
        ''' 创建文件夹 '''
        self.send_msg(data)
        data = self.recv_msg()
        print(data['msg'])

    def _del(self, data):
        ''' delect dir ,不能删除文件 '''
        self.send_msg(data)
        data = self.recv_msg()
        print(data['msg'])

    def _touch(self, data):
        ''' 创建文件 '''
        self.send_msg(data)
        data = self.recv_msg()
        print(data['msg'])

    def _ls(self, data):
        ''' 查看当前目录下有什么文件或目录 '''
        # print('ls_data', data)
        self.send_msg(data)
        data = self.recv_msg()
        print(data['msg'])

    def _cd(self, data):
        ''' 切换目录 '''
        # print('cd_data', data)
        self.send_msg(data)
        data = self.recv_msg()
        print(data['msg'])

    def handle(self):
        ''' 解析指令并交给具体的方法处理 '''
        while 1:
            cmd = input('>>: ').strip()
            if not cmd: continue
            data = {"choice": cmd.split(' '), 'action_type': cmd.split(' ')[0]}
            if hasattr(self, '_%s' % data["choice"][0]):
                getattr(self, '_%s' % data["choice"][0])(data)
            else:
                print('invalid command')

    def send_msg(self, data):
        head_data = json.dumps(data).encode()
        head_size = struct.pack('i', len(head_data))
        self.request.send(head_size)
        self.request.send(head_data)

    def recv_msg(self):
        head_size = self.request.recv(4)
        st_data = struct.unpack('i', head_size)[0]
        head_data = self.request.recv(st_data).decode()
        return json.loads(head_data)

    def connect_socket(self):
        self.request = socket.socket()
        print(11, self.server_address)
        self.request.connect(self.server_address, )


if __name__ == '__main__':
    Client(('127.0.0.1', 8881)).handle()

server

手动使用多线程实现ftp
python
#!/bin/bash
# -*- coding: utf-8 -*-


import threading
import socket
import struct
import json


class MyFtp(object):

    def _mk(self, data):
        ''' 创建文件夹 '''
        # print('mk', data)
        data['msg'] = "mk msg"
        self.send_msg(data)

    def _del(self, data):
        ''' delect dir ,不能删除文件 '''
        # print('delect', data)
        data['msg'] = "delect msg"
        self.send_msg(data)

    def _touch(self, data):
        ''' 创建文件 '''
        # print('touch', data)
        data['msg'] = "touch msg"
        self.send_msg(data)

    def _ls(self, data):
        """ 查看当前目录下的文件 """
        # print('ls', data)
        data['msg'] = "ls msg"
        self.send_msg(data)

    def _cd(self, data):
        """ 切换目录 """
        # print('cd', data)
        data['msg'] = "cd msg"
        self.send_msg(data)

    def handle(self, request, client_address):
        self.request, self.client_address = request, client_address
        while True:
            try:
                data_msg = self.recv_msg()
                if hasattr(self, '_%s' % data_msg['action_type']):
                    getattr(self, '_%s' % data_msg['action_type'])(data_msg)
            except (ConnectionResetError, ConnectionAbortedError):
                break
        self.request.close()
        print('the {} disconnects...'.format(self.client_address))

    def send_msg(self, data):
        head_data = json.dumps(data).encode()
        head_size = struct.pack('i', len(head_data))
        self.request.send(head_size)
        self.request.send(head_data)

    def recv_msg(self):
        head_size = self.request.recv(4)
        st_data = struct.unpack('i', head_size)[0]
        head_data = self.request.recv(st_data).decode()
        return json.loads(head_data)


class MyThreadingTCPServer(object):
    request_queue_size = 5
    allow_reuse_address = False

    def __init__(self, server_address, request_handler_class):
        self.server_address = server_address
        self.RequestHandlerClass = request_handler_class
        self.request = socket.socket(family=socket.AF_INET, type=socket.SOCK_STREAM)
        self.request.bind(server_address)
        self.request.listen(self.request_queue_size)

    def serve_forever(self):
        while True:
            print('waiting for the link.....')
            self.conn, self.client_address = self.request.accept()
            t = threading.Thread(target=self.RequestHandlerClass.handle,
                                 args=(self.RequestHandlerClass(), self.conn, self.client_address))
            t.start()

if __name__ == '__main__':
    # 手动实现并发
    obj = MyThreadingTCPServer(('127.0.0.1', 8881), MyFtp)
    obj.serve_forever()
使用socketserver实现多用户ftp
python
#!/bin/bash
# -*- coding: utf-8 -*-

import struct
import json
import socketserver


class MyFtp(socketserver.BaseRequestHandler):

    def _mk(self, data):
        ''' 创建文件夹 '''
        # print('mk', data)
        data['msg'] = "mk msg"
        self.send_msg(data)

    def _del(self, data):
        ''' delect dir ,不能删除文件 '''
        # print('delect', data)
        data['msg'] = "delect msg"
        self.send_msg(data)

    def _touch(self, data):
        ''' 创建文件 '''
        # print('touch', data)
        data['msg'] = "touch msg"
        self.send_msg(data)

    def _ls(self, data):
        """ 查看当前目录下的文件 """
        # print('ls', data)
        data['msg'] = "ls msg"
        self.send_msg(data)

    def _cd(self, data):
        """ 切换目录 """
        # print('cd', data)
        data['msg'] = "cd msg"
        self.send_msg(data)

    def handle(self):
        while True:
            print('waiting for the link.....')
            data_msg = self.recv_msg()
            if hasattr(self, '_%s' % data_msg['action_type']):
                getattr(self, '_%s' % data_msg['action_type'])(data_msg)

    def send_msg(self, data):
        head_data = json.dumps(data).encode()
        head_size = struct.pack('i', len(head_data))
        self.request.send(head_size)
        self.request.send(head_data)

    def recv_msg(self):
        head_size = self.request.recv(4)
        st_data = struct.unpack('i', head_size)[0]
        head_data = self.request.recv(st_data).decode()
        return json.loads(head_data)


if __name__ == '__main__':
    # 使用socketserver实现并发
    obj = socketserver.ThreadingTCPServer(('127.0.0.1', 8881), MyFtp)
    obj.serve_forever()

that's all