python中的socket實現(xiàn)ftp客戶端和服務器收發(fā)文件及md5加密文件
客戶端向服務器發(fā)送一個請求,請求內(nèi)容是一個文件名,服務器在查找自己這邊有沒有這個文件,如果有的話就發(fā)送給客戶端
1、客戶端
生成socket對象 建立連接 輸入想要接收的文件 將輸入的文件名發(fā)送給服務器 接收服務器發(fā)回的關于即將要發(fā)送來的文件的大小 發(fā)送一條信息給服務器告訴它準備好接收了 接收文件數(shù)據(jù) 打印全部接收的提示信息client具體實現(xiàn)的代碼如下:
# Author: Mr.Xue# 2019.10.29# socket_ftp_client.pyimport socketimport hashlibclient = socket.socket() # 生成socket連接對象client.connect((’localhost’, 6961)) # 建立連接while True: cmd = input('>>').strip() # 輸入想要接收的文件 if len(cmd) == 0: continue # 輸入為空,重新再輸 if cmd.startswith(’get’): # 判斷指令是否以get開頭 client.send(cmd.encode('utf-8')) # 發(fā)送 server_response = client.recv(1024) #接收即將發(fā)送來的文件的大小 print('server response:', server_response) # 打印文件的大小 client.send(b’ready to recv file...’) # 發(fā)送消息告訴服務已經(jīng)準備好接收了 file_total_size = int(server_response.decode()) # 記錄文件的總大小 received_size = 0 # 記錄已經(jīng)接收了的文件的大小 filename = cmd.split()[1] # 取出輸入的文件名 f = open(filename + ’.new’, ’wb’) # 新建一個本地文件來存儲接收的數(shù)據(jù) m = hashlib.md5() # md5加密 while received_size < file_total_size: data = client.recv(1024) # 接收數(shù)據(jù),一次最大接收1024bytes received_size += len(data) # 記錄已接收的數(shù)據(jù)大小 m.update(data) # 用md5加密 f.write(data) # 寫入文件 else: new_file_md5 = m.hexdigest() # 16進制顯示加密文件 print('file recv done', received_size, file_total_size) f.close() server_md5 = client.recv(1024) # 接收服務器端以md5加密的這個接收文件的加密文件,拿來和接收之后的加密文件做比較 print('md5', new_file_md5, server_md5)client.close()
client端比較難的一點和上篇一樣,還是如何判斷服務器端要發(fā)送的數(shù)據(jù),客戶端是否完全都接收過來了,實現(xiàn)邏輯在上面代碼中,也可參考前一篇socket(二);在客戶端這一次多了一個新的知識點,用md5來加密接收到的文件,先生成md5實例對象,再調(diào)用update()來加密文件,最后和服務器端的加密文件進行比較,看看是不是一樣的。
2、服務器
生成socket連接對象 綁定要監(jiān)聽端口 監(jiān)聽 等待客戶端的連接 接收客戶端發(fā)來的文件名 查找這個文件是否存在 如果存在,打開文件計算大小,發(fā)送給客戶端 等待客戶端確認 發(fā)送文件具體內(nèi)容給客戶端server具體實現(xiàn)代碼如下:
# Author: Mr.Xue# 2019.10.29# socket_ftp_server.pyimport socket, os, hashlibserver = socket.socket() #生成socket對象server.bind((’localhost’, 6961)) # 綁定ip和端口server.listen() #監(jiān)聽while True: conn, addr = server.accept() # 等待客戶端連接 while True: print('等待接收文件名...') data = conn.recv(1024) # 接收客戶端發(fā)來的信息 if not data: #判斷客戶端是否斷連 print(’lost a link...’) break cmd, filename = data.decode().split() # 分割接收到的消息,提取出文件名 if os.path.isfile(filename): # 判斷文件是否存在 f = open(filename, ’rb’) # 打開文件 m = hashlib.md5() # md5加密對象 file_size = os.stat(filename).st_size # 計算文件大小 conn.send(str(file_size).encode('utf-8')) # send file size conn.recv(1024) # wait the ack 預防粘包 for line in f: m.update(line) # 加密 conn.send(line) # 發(fā)送 print('file md5:', m.hexdigest()) # 以16進制打印加密后的文件 f.close() # 關閉文件 conn.send(m.hexdigest().encode('utf-8')) # 以16進制發(fā)送加密后的文件 print(’send done’)socket.close()
服務器的具體實現(xiàn)邏輯和前一篇大體上沒什么區(qū)別,主要處理細節(jié)誒變成了查找文件處理文件和加密文件
3、測試效果
啟動服務器
xue@xue-MacBookAir:~/python_learn$ python3 socket_ftp_server.py
啟動客戶端
client
xue@xue-MacBookAir:~/python_learn$ python3 socket_ftp_client.py >>
server
xue@xue-MacBookAir:~/python_learn$ python3 socket_ftp_server.py 等待接收文件名...
客戶端發(fā)送想要接收的文件名
client
xue@xue-MacBookAir:~/python_learn$ python3 socket_ftp_client.py >>get a.txtserver response: b’90’file recv done 90 90md5 2232897a127542c1d0715e66e7ca57cc b’2232897a127542c1d0715e66e7ca57cc’>>
server
xue@xue-MacBookAir:~/python_learn$ python3 socket_ftp_server.py等待接收文件名...file md5: 2232897a127542c1d0715e66e7ca57ccsend done等待接收文件名...
上面的服務器端在發(fā)送完數(shù)據(jù)之后,接著又發(fā)送了md5的加密文件,可能會發(fā)生粘包的情況,所以上面還存在改進的地方,將預防粘包的代碼加進去就比較完美了。
總結
到此這篇關于python中的socket實現(xiàn)ftp客戶端和服務器收發(fā)文件及md5加密文件的文章就介紹到這了,更多相關python socket 實現(xiàn)ftp 收發(fā)文件md5加密文件內(nèi)容請搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持好吧啦網(wǎng)!
相關文章:
1. vue實現(xiàn)web在線聊天功能2. JavaScript實現(xiàn)頁面動態(tài)驗證碼的實現(xiàn)示例3. JavaEE SpringMyBatis是什么? 它和Hibernate的區(qū)別及如何配置MyBatis4. Springboot 全局日期格式化處理的實現(xiàn)5. SpringBoot+TestNG單元測試的實現(xiàn)6. 完美解決vue 中多個echarts圖表自適應的問題7. 解決Android Studio 格式化 Format代碼快捷鍵問題8. 在Chrome DevTools中調(diào)試JavaScript的實現(xiàn)9. Python使用urlretrieve實現(xiàn)直接遠程下載圖片的示例代碼10. Java使用Tesseract-Ocr識別數(shù)字
