day8--socketserver作業

来源:http://www.cnblogs.com/gengcx/archive/2017/09/03/7420714.html
-Advertisement-
Play Games

fileno()文件描述符 handle_request()處理單個請求 server_forever(poll_interval=0.5)處理多個請求,poll_interval每0.5秒檢測是否關閉, 作業:開發一個支持多用戶線上的FTP程式 要求: 1.用戶加密認證; 2.允許同時多用戶登錄; ...


fileno()文件描述符

handle_request()處理單個請求

server_forever(poll_interval=0.5)處理多個請求,poll_interval每0.5秒檢測是否關閉,

作業:開發一個支持多用戶線上的FTP程式

要求:

    1.用戶加密認證;

    2.允許同時多用戶登錄;

    3.每個用戶有自己的家目錄,且只能訪問自己的家目錄;

    4.對用戶進行磁碟配額,每個用戶的可用空間不同;

    5.允許用戶在ftp.server上隨意切換目錄;

    6.允許用戶查看當前目錄下文件;

    7.允許上傳和下載文件,保證文件一致性;

    8.文件傳輸過程中顯示進度條;

    9.附加功能,支持文件的斷點續傳。

發送文件的大致思路:

 

接收文件的大致思路:

遍歷空的文件,如果一個文件是空的,Python是不會執行後面縮進的代碼的,空的文件。

http://www.cnblogs.com/liujiacai/p/7417953.html

http://www.cnblogs.com/lianzhilei/p/5869205.html

http://www.bubuko.com/infodetail-1758633.html

    由於不是做運維的,對一些系統的操作不是很熟悉,因此首先完成了文件的上傳與下載:下麵來看看這兩個功能:

   

    客戶端:

 

import socket,os,json,sys


class Myclient(object):
    '''定義客戶端'''
    def __init__(self):
        '''定義socket()實例'''
        self.client = socket.socket()                                         #生成soket()實例

    def connect(self,ip,port):
        """定義連接"""
        self.client.connect((ip,port))

    def interactive(self):
        '''定義和用戶交互'''
        while True:
            cmd = input("請輸入指令>>:").strip()                              #用戶輸入指令
            if len(cmd) == 0:
                print("指令不能為空!!!")
                continue                                                      #指令不能為空
            cmd_str = cmd.split()[0]                                          #指令由兩種形式
            if hasattr(self,cmd_str):                                         #判斷指令是否存在
                func = getattr(self,cmd_str)                                  #指令存在,則獲取對應的方法
                func(cmd)
            elif cmd_str == 'q':
                break
            else:
                self.help()
                continue


    def help(self):
        msg = """
        ls
        pwd
        cd
        put   filename
        get   filename
        """
        print("指令名稱",msg)

    def put(self,cmd):
        '''定義上傳數據介面'''
        cmd_list = cmd.split()
        if len(cmd_list) > 1:
            '''指令是由指令名和文件名構成'''
            filename = cmd_list[1]                                              #獲取文件名
            if os.path.isfile(filename):                                        #檢測上傳的文件名是否存在
                '''文件名存在,則要告訴伺服器端上傳的文件名,文件大小,並檢測伺服器是否有相應的方法'''
                filesize = os.stat(filename).st_size
                msg_dic = {
                    "action":"put",
                    "filename":filename,
                    "filesize":filesize
                }
                '''轉換為位元組碼的形式進行傳輸'''
                self.client.send(json.dumps(msg_dic).encode("utf-8"))           #轉換為json格式進行上傳,便於伺服器接收後處理
                server_response = self.client.recv(1024)                        #等待伺服器傳回信息,防止粘包
                self.client.send("收到了".encode("utf-8"))
                if int(server_response.decode("utf-8")):
                    '''根據客戶端反饋,伺服器端是否存在相同的文件名'''
                    file_exist = int(self.client.recv(1024).decode("utf-8"))
                    if file_exist:
                        '''伺服器端存在相應的文件'''
                        while True:
                            cover_flag = input("文件已存在,是否覆蓋原文件(y/n):")
                            if cover_flag == "y":
                                self.client.send("y".encode("utf-8"))
                                break
                            elif cover_flag == "n":
                                new_filename = input("請輸入新的文件名>>:")
                                self.client.send(("n "+new_filename).encode("utf-8"))
                                break
                            else:
                                print("您輸入的指令有誤,請重新輸入!!!")
                                continue
                    else:
                        self.client.send("新創建".encode("utf-8"))

                    '''上面準備工作完畢,開始傳輸文件'''
                    with open(filename,'rb') as f:
                        for line in f:
                            self.client.send(line)                                  #發送數據
                    '''客戶端等待伺服器端傳來消息,告知上傳成功'''
                    response = self.client.recv(1024)
                    print(response.decode("utf-8"))                                 #數據是否上傳成功,告知用戶
                else:
                    print("伺服器端不存在相應的指令!!!")

            else:
                print("要上傳的文件名不存在!")


        else:
            print("沒有輸入要上傳的文件名!!!")

    def get(self,cmd):
        '''從伺服器端下載數據'''
        cmd_list = cmd.split()                                                    #獲取用戶輸入指令
        if len(cmd_list) > 1:                                                     #是由指令和文件名構成
            filename = cmd_list[1]
            msg_dic = {
                "filename":filename,
                "action":"get"
            }                                                                     #把指令封裝成字典發送給伺服器
            self.client.send(json.dumps(msg_dic).encode("utf-8"))                 #把指令以json形式傳給伺服器
            server_response = int(self.client.recv(1024).decode("utf-8"))         #接收服務端發揮的指令,判斷是否支持此方法
            self.client.send("收到指令!".encode("utf-8"))
            if server_response:
                file_exists = int(self.client.recv(1024).decode("utf-8"))         #下載文件存在與否標識符
                self.client.send("收到標識符".encode("utf-8"))                     #告知伺服器收到標識符,防止粘包
                if file_exists:
                    '''文件存在,則準備接收數據,會接收伺服器發送過來的文件大小'''
                    file_mess = json.loads(self.client.recv(3072).decode("utf-8"))
                    self.client.send("文件信息接收成功,開始進行文件接收!".encode("utf-8"))
                    filesize = file_mess["filesize"]
                    while True:
                        '''確定文件名'''
                        if os.path.isfile(filename):
                            '''說明客戶端存在文件,就要讓用戶選擇是否覆蓋'''
                            choice_cover = input("文件名重覆,是否覆蓋現有文件(y/n)>>:")
                            if choice_cover == "y":
                                filename = filename
                                break                                       #覆蓋文件之後退出迴圈
                            elif choice_cover == "n":
                                filename = input("請輸入新的文件名>>:")
                                '''由於用戶輸入了新的用戶名,我們不知道此文件是否重名,因此重新檢測一下,確保不重名'''
                                continue
                        else:
                            filename = filename
                            break                                           #得到文件名之後退出迴圈
                    recvive_size = 0
                    with open(filename,'wb') as file:
                        while recvive_size < filesize:
                            data = self.client.recv(1024)
                            file.write(data)
                            recvive_size += len(data)
                        print("數據下載完成!!!")
                        self.client.send("接收完畢".encode("utf-8"))
                else:
                    print("要下載的文件不存在!!!")

            else:
                print("伺服器不支持此功能!!!")

        else:
            print("對不起,輸入有誤!")
            self.help()

if __name__ == "__main__":
    try:
        client = Myclient()
        client.connect("localhost",9998)
        client.interactive()
    except KeyboardInterrupt as e:
        print("客戶端斷開!!!",e)

 

    上面客戶端只實現了上傳和下載功能,沒有實現其他系統操作有關的功能,後續將逐步完善;

    伺服器端:

'''定義伺服器端'''
import socketserver,json,os

class MyTcpServer(socketserver.BaseRequestHandler):
    '''定義伺服器端,socketserver,繼承的是BaseRequestHandler類'''
    def handle(self):
        '''伺服器端所有的功能都在handle()裡面進行處理,因此接收數據也是在handle()中,只是處理的時候,調用下麵模塊而已'''
        while True:
            data = self.request.recv(3072)
            if len(data) == 0:
                print("客戶端斷開")
                break
            msg_dic = json.loads(data.decode("utf-8"))            #伺服器端接收指令
            cmd = msg_dic["action"]
            if hasattr(self,cmd):                                                    #判斷伺服器是否有相應的指令
                self.request.send("1".encode("utf-8"))
                self.request.recv(1024)
                func = getattr(self,cmd)
                func(msg_dic)
            else:
                self.request.send("0".encode("utf-8"))
                self.request.recv(1024)

    def put(self,msg_dic):
        '''定義上傳的伺服器代碼'''
        filename = msg_dic['filename']
        filesize = msg_dic["filesize"]
        '''判斷伺服器端是否存在對應的文件名'''
        recvive_size = 0
        if os.path.isfile(filename):
            '''如果伺服器存在相應的文件名,則詢問客戶端是否覆蓋'''
            self.request.send("1".encode("utf-8"))                              #文件存在,迴圈客戶端是否覆蓋
            response = self.request.recv(1024).decode("utf-8")
            response = response.split()
            '''如果用戶讓覆蓋則覆蓋,否則重新創建'''

            if response[0] == "y":
                '''覆蓋'''
                with open(filename,"wb") as f1:
                   while recvive_size < filesize:
                       data = self.request.recv(1024)
                       f1.write(data)
                       recvive_size += len(data)
                self.request.send("文件接收成功!".encode("utf-8"))

            else:
                '''不覆蓋,新創建'''
                new_filename = response[1]
                with open(new_filename,'wb') as f:
                    while recvive_size < filesize:
                        data1 = self.request.recv(1024)
                        f.write(data1)
                        recvive_size += len(data1)
                self.request.send("文件接收成功!".encode("utf-8"))

        else:
            '''如果伺服器不存在相應文件名,則創建'''
            self.request.send("0".encode("utf-8"))
            self.request.recv(1024)
            with open(filename,'wb') as f2:
                while recvive_size < filesize:
                    data1 = self.request.recv(1024)
                    f2.write(data1)
                    recvive_size += len(data1)
            self.request.send("文件接收成功!".encode("utf-8"))

    def get(self,msg_dict):
        '''基礎交流結束,伺服器支持文件下載功能,開始文件下載操作'''
        filename = msg_dict["filename"]
        if os.path.isfile(filename):
            self.request.send("1".encode("utf-8"))                             #文件存在告知客戶端
            self.request.recv(1024)
            filesize = os.stat(filename).st_size
            '''首先告知客戶端文件大小,讓這邊做好接收準備'''
            file_mess = {
                "filesize":filesize,
            }
            self.request.send(json.dumps(file_mess).encode("utf-8"))           #以json格式發送給客戶端
            self.request.recv(1024)                                            #防止粘包,客戶端回應伺服器接收完畢
            with open(filename,"rb") as f_obj:
                for line in f_obj:
                    self.request.send(line)
            self.request.recv(1024)

        else:
            self.request.send("0".encode("utf-8"))                             #文件不存在,告知客戶端
            self.request.recv(1024)

if __name__ == "__main__":
    try:
        IP,PORT = "localhost",9998
        server = socketserver.ThreadingTCPServer((IP,PORT),MyTcpServer)              #定義socketserver實例
        server.serve_forever()
    except KeyboardInterrupt as e:
        print("關閉伺服器")

    上面代碼是伺服器端,客戶端與伺服器端其實是相呼應的,兩者實現數據的交換;其實就是收發數據,中間摻雜我們想要實現的功能。

    下麵來看幾個容易犯錯的地方:

    1、BrokenPipeError: [Errno 32] Broken pipe

 

  客戶端代碼:

 

import socket,os,json,sys


class Myclient(object):
    '''定義客戶端'''
    def __init__(self):
        '''定義socket()實例'''
        self.client = socket.socket()                                         #生成soket()實例

    def connect(self,ip,port):
        """定義連接"""
        self.client.connect((ip,port))

    def interactive(self):
        '''定義和用戶交互'''
        while True:
            cmd = input("請輸入指令>>:").strip()                              #用戶輸入指令
            if len(cmd) == 0:
                print("指令不能為空!!!")
                continue                                                      #指令不能為空
            cmd_str = cmd.split()[0]                                          #指令由兩種形式
            if hasattr(self,cmd_str):                                         #判斷指令是否存在
                func = getattr(self,cmd_str)                                  #指令存在,則獲取對應的方法
                func(cmd)
            elif cmd_str == 'q':
                break
            else:
                self.help()
                continue


    def help(self):
        msg = """
        ls
        pwd
        cd
        put   filename
        get   filename
        """
        print("指令名稱",msg)

    def put(self,cmd):
        '''定義上傳數據介面'''
        cmd_list = cmd.split()
        if len(cmd_list) > 1:
            '''指令是由指令名和文件名構成'''
            filename = cmd_list[1]                                              #獲取文件名
            if os.path.isfile(filename):                                        #檢測上傳的文件名是否存在
                '''文件名存在,則要告訴伺服器端上傳的文件名,文件大小,並檢測伺服器是否有相應的方法'''
                filesize = os.stat(filename).st_size
                msg_dic = {
                    "action":"put",
                    "filename":filename,
                    "filesize":filesize
                }
                '''轉換為位元組碼的形式進行傳輸'''
                self.client.send(json.dumps(msg_dic).encode("utf-8"))           #轉換為json格式進行上傳,便於伺服器接收後處理
                server_response = self.client.recv(1024)                        #等待伺服器傳回信息,防止粘包
                self.client.send("收到了".encode("utf-8"))
                if int(server_response.decode("utf-8")):
                    '''根據客戶端反饋,伺服器端是否存在相同的文件名'''
                    file_exist = int(self.client.recv(1024).decode("utf-8"))
                    if file_exist:
                        '''伺服器端存在相應的文件'''
                        while True:
                            cover_flag = input("文件已存在,是否覆蓋原文件(y/n):")
                            if cover_flag == "y":
                                self.client.send("y".encode("utf-8"))
                                break
                            elif cover_flag == "n":
                                new_filename = input("請輸入新的文件名>>:")
                                self.client.send(("n "+new_filename).encode("utf-8"))
                                break
                            else:
                                print("您輸入的指令有誤,請重新輸入!!!")
                                continue
                    else:
                        self.client.send("新創建".encode("utf-8"))

                    '''上面準備工作完畢,開始傳輸文件'''
                    with open(filename,'rb') as f:
                        for line in f:
                            self.client.send(line)                                  #發送數據
                    '''客戶端等待伺服器端傳來消息,告知上傳成功'''
                    response = self.client.recv(1024)
                    print(response.decode("utf-8"))                                 #數據是否上傳成功,告知用戶
                else:
                    print("伺服器端不存在相應的指令!!!")

            else:
                print("要上傳的文件名不存在!")


        else:
            print("沒有輸入要上傳的文件名!!!")

    def get(self,cmd):
        '''從伺服器端下載數據'''
        cmd_list = cmd.split()                                                    #獲取用戶輸入指令
        if len(cmd_list) > 1:                                                     #是由指令和文件名構成
            filename = cmd_list[1]
            msg_dic = {
                "filename":filename,
                "action":"get"
            }                                                                     #把指令封裝成字典發送給伺服器
            self.client.send(json.dumps(msg_dic).encode("utf-8"))                 #把指令以json形式傳給伺服器
            server_response = int(self.client.recv(1024).decode("utf-8"))         #接收服務端發揮的指令,判斷是否支持此方法
            self.client.send("收到指令!".encode("utf-8"))
            if server_response:
                file_exists = int(self.client.recv(1024).decode("utf-8"))         #下載文件存在與否標識符
                self.client.send("收到標識符".encode("utf-8"))                     #告知伺服器收到標識符,防止粘包
                if file_exists:
                    '''文件存在,則準備接收數據,會接收伺服器發送過來的文件大小'''
                    file_mess = json.loads(self.client.recv(3072).decode("utf-8"))
                    self.client.send("文件信息接收成功,開始進行文件接收!".encode("utf-8"))
                    filesize = file_mess["filesize"]
                    while True:
                        '''確定文件名'''
                        if os.path.isfile(filename):
                            '''說明客戶端存在文件,就要讓用戶選擇是否覆蓋'''
                            choice_cover = input("文件名重覆,是否覆蓋現有文件(y/n)>>:")
                            if choice_cover == "y":
                                filename = filename
                                break                                       #覆蓋文件之後退出迴圈
                            elif choice_cover == "n":
                                filename = input("請輸入新的文件名>>:")
                                '''由於用戶輸入了新的用戶名,我們不知道此文件是否重名,因此重新檢測一下,確保不重名'''
                                continue
                        else:
                            filename = filename
                            break                                           #得到文件名之後退出迴圈
                    recvive_size = 0
                    with open(filename,'wb') as file:
                        while recvive_size < filesize:
                            data = self.client.recv(1024)
                            file.write(data)
                            recvive_size += len(data)
                        print("數據下載完成!!!")
                        self.client.send("接收完畢".encode("utf-8"))
                else:
                    print("要下載的文件不存在!!!")

            else:
                print("伺服器不支持此功能!!!")

        else:
            print("對不起,輸入有誤!")
            self.help()

if __name__ == "__main__":
    try:
        client = Myclient()
        client.connect("localhost",9998)
        client.interactive()
    except KeyboardInterrupt as e:
        print("客戶端斷開!!!",e)

 

    伺服器端:

'''定義伺服器端'''
import socketserver,json,os

class MyTcpServer(socketserver.BaseRequestHandler):
    '''定義伺服器端,socketserver,繼承的是BaseRequestHandler類'''
    def handle(self):
        '''伺服器端所有的功能都在handle()裡面進行處理,因此接收數據也是在handle()中,只是處理的時候,調用下麵模塊而已'''
        # while True:
        data = self.request.recv(3072)
        if len(data) == 0:
            print("客戶端斷開")
                # break
        msg_dic = json.loads(data.decode("utf-8"))            #伺服器端接收指令
        cmd = msg_dic["action"]
        if hasattr(self,cmd):                                                    #判斷伺服器是否有相應的指令
            self.request.send("1	   

您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 自己畫一個轉圈圈的控制項 ...
  • 首先說一下foreach有的也叫增強for迴圈,foreach其實是for迴圈的一個特殊簡化版。 再說一下foreach的書寫格式: for(元素類型 元素名稱 : 遍曆數組(集合)(或者能進行迭代的)){ 語句 } foreach雖然是for迴圈的簡化版本,但是並不是說foreach就比for更好 ...
  • 特別聲明本隨筆copy於egon(林海峰)。 一 IO模型介紹 為了更好地瞭解IO模型,我們需要事先回顧下:同步、非同步、阻塞、非阻塞 同步(synchronous) IO和非同步(asynchronous) IO,阻塞(blocking) IO和非阻塞(non-blocking)IO分別是什麼,到底有 ...
  • 建立資料庫訪問類的封裝 <?php class DBDA { public $host = "localhost"; //伺服器地址 public $uid = "root"; //資料庫的用戶名 public $pwd = ""; //資料庫的密碼 public $dbname = "";//數據 ...
  • Spring與SpringMVC整合! 問:實際上SpringMVC就運行在Spring環境之下,還有必要整合麽?SpringMVC和Spring都有IOC容器,是不是都需要保留呢? 答案是:通常情況下,類似於數據源,事務,整合其他框架都是放在spring的配置文件中(而不是放在SpringMVC的 ...
  • 冒泡排序是一種非常常見的排序演算法。如同水中的一排泡泡,先冒出最大的一個泡泡。再冒出剩餘泡泡中的最大泡泡,依次類推,它的排序規則如下: 1. 從第一個元素開始,比較相鄰的兩個元素,如果後面的小於前面的,交換兩個的位置,一直比較到最後一個 2. 迴圈1中的操作,但已經確定的最大的元素不再參與比較 3. ...
  • 多線程 前言 我看了不止一個人說多線程是雞肋,但是就依照我個人覺得多線程在一些小型的爬蟲中還是可以顯著的提高速度的,相比多進程來說應該還是挺簡單的 使用多線程 繼承threading.Thread 繼承threading.Thread模塊是一個很好的一個選擇,就像java中也是可以繼承類和實現介面一 ...
  • Java類載入器算是一個老生常談的問題,大多Java工程師也都對其中的知識點倒背如流,最近在看源碼的時候發現有一些細節的地方理解還是比較模糊,正好寫一篇文章梳理一下。 關於Java類載入器的知識,網上一搜一大片,我自己也看過很多文檔,博客。資料雖然很多,但還是希望通過本文儘量寫出一些自己的理解,自己 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...