python實(shí)現(xiàn)ftp文件傳輸系統(tǒng)(案例分析)
最近做了一個(gè)簡(jiǎn)單的文件傳輸系統(tǒng),基于ftp協(xié)議,使用python語(yǔ)言開(kāi)發(fā),雖然python里面已經(jīng)有ftplib模塊,可以很容易的實(shí)現(xiàn)ftp服務(wù)器。這次我使用的是socket實(shí)現(xiàn)client與ftp server之間的通訊和文件傳輸,client另起一個(gè)flask服務(wù)器,用于用戶在瀏覽器端的交互。系統(tǒng)實(shí)現(xiàn)的功能有:用戶登錄注冊(cè),用戶查看ftp服務(wù)器端文件和下載上傳刪除操作,支持多進(jìn)程、多用戶。
一,登錄注冊(cè)
該項(xiàng)目使用的是mongo數(shù)據(jù)庫(kù),其實(shí)用戶登錄注冊(cè)功能很好實(shí)現(xiàn),沒(méi)有什么技術(shù)細(xì)節(jié),這里就略過(guò)了。數(shù)據(jù)庫(kù)管理代碼如下:
import pymongo from pymongo.collection import Collection class DBManager(object): def __init__(self): client = pymongo.MongoClient("mongodb://localhost:27017/") self.db = client["FTPDB"] self.users = self.db['user'] #保存用戶登錄信息 def saveUserInfo(self,account,password): users = self.users.find() for item in users: accountDB = item['account'] if accountDB == account: return "false" data = [] userInfo = {} userInfo['account'] = account userInfo['password'] = password data.append(userInfo) collection = Collection(self.db,"user") collection.insert(data) return "true" def confirmUserLoginInfo(self,account,password): users = self.users.find() ''' result狀態(tài): 1:表示初始狀態(tài),即不存在用戶 2:表示存在該用戶、密碼不正確 3:驗(yàn)證成功 ''' result = 1 for item in users: accountDB = item['account'] passwordDB = item['password'] if accountDB == account: if passwordDB == password: result = 3 else: result = 2 return result
前端注冊(cè)js代碼如下:
function register() { account = $("#account").val(); password = $("#password").val(); confirmPassword = $("#confirmPassword").val(); if(account == null || password == null || confirmPassword == null){ alert("請(qǐng)先輸入必要信息") return; } if(password != confirmPassword){ alert("密碼不一致"); return; } var request = { type:"register", account:account, password:password } sendData("http://localhost:8080/register",request) } //向服務(wù)器發(fā)送數(shù)據(jù) function sendData(url,json) { $.ajax({ url: url, //請(qǐng)求的url地址 dataType: "json", //返回格式為json async: true, //請(qǐng)求是否異步,默認(rèn)為異步,這也是ajax重要特性 data: json, //參數(shù)值 type: "post", //請(qǐng)求方式 success:function(data){ //alert(data) if(data.toString() == "false"){ alert("用戶名已存在"); }else{ window.location.href = "http://localhost:8080/index"; } }, error:function (error) { console.log(error); } }); }
二,文件管理(文件查看、刪除、上傳、下載)
客戶端與服務(wù)器端約定命令格式,服務(wù)器通過(guò)解析客戶端命令來(lái)執(zhí)行操作。
server.py from socket import * import os,sys import signal import time # 全局變量 HOST = '0.0.0.0' PORT = 8686 ADDR = (HOST,PORT) FILE_PATH = '../serverFiles/' # 處理僵尸進(jìn)程 signal.signal(signal.SIGCHLD,signal.SIG_IGN) # 服務(wù)端功能類 class Server(object): def __init__(self): self.connfd = "" def do_list(self,account): # 獲取文件列表 file_list = os.listdir(FILE_PATH+account) if not file_list: self.connfd.send("服務(wù)器文件庫(kù)為空".encode()) return else: self.connfd.send(b"OK") time.sleep(0.1) files = "" for file in file_list: if file[0] != '.' and os.path.isfile(FILE_PATH + account +"/"+ file): files += file + '#' self.connfd.send(files.encode()) def delete(self,accout,fileName): os.remove(FILE_PATH + accout + "/" + fileName) self.connfd.send(b"OK") time.sleep(0.1) def do_get(self,account,filename): try: fd = open(FILE_PATH + account +"/"+ filename,'rb') except IOError: self.connfd.send("文件不存在".encode()) return else: #print("發(fā)送OK") self.connfd.send(b'OK') time.sleep(0.1) # 發(fā)送文件內(nèi)容 while True: data = fd.read(1024) if not data: time.sleep(0.1) self.connfd.send(b'##') break #print("正在發(fā)送數(shù)據(jù)") self.connfd.send(data) fd.close() def do_put(self,account,filename): if os.path.exists(FILE_PATH + account +"/"+ filename): self.connfd.send('該文件已存在'.encode()) return fd = open(FILE_PATH + account +"/"+ filename,'wb') self.connfd.send(b'OK') # 接收文件內(nèi)容 while True: data = self.connfd.recv(1024) if data == b'**': break fd.write(data) fd.close() def socket_tcp(self): s = socket() s.setsockopt(SOL_SOCKET,SO_REUSEADDR,True) s.bind(ADDR) s.listen(5) print("Listen the port 8686...") return s def do_request(self,connfd): self.connfd = connfd while True: data = connfd.recv(1024).decode() datas = data.split(' ') if not data or datas[1] == 'QUIT@#': connfd.close() return elif datas[1] == "LIST@#": #print("list") self.do_list(datas[0]) elif datas[1] == 'GET@#': filename = datas[-1] self.do_get(datas[0],filename) elif datas[1] == 'PUT@#': filename = datas[-1] self.do_put(datas[0],filename) elif datas[1] == 'delete@#': filename = datas[-1] self.delete(datas[0],filename) def run(self): # 創(chuàng)建套接字 s = self.socket_tcp() while True: try: connfd,addr = s.accept() except KeyboardInterrupt: sys.exit("服務(wù)器退出") except Exception as e: print(e) continue print("Connect from",addr) # 創(chuàng)建子進(jìn)程 pid = os.fork() if pid == 0: s.close() self.do_request(connfd) #處理客戶端具體請(qǐng)求 os._exit(0) else: connfd.close() if __name__ == "__main__": server = Server() server.run() client.py """ client.py """ import socket import os,sys import time # 服務(wù)器地址 ADDR = ("127.0.0.1",8686) FILE_PATH = "./clientFiles/" # 客戶端功能類 class Client(object): def __init__(self,account): self.sockfd = "" self.account = account #獲得服務(wù)器文件列表 def server_list(self): ftpServerFiles = [] self.sockfd.send((self.account+' LIST@# ').encode()) # 等待回復(fù) data = self.sockfd.recv(128).decode() if data == "OK": files = self.sockfd.recv(4096).decode() for file in files.split('#'): #print(file) ftpServerFiles.append(file) else: # 無(wú)法完成操作 print(data) ftpServerFiles = ftpServerFiles[:-1] return ftpServerFiles #獲得用戶文件夾列表 def client_list(self): # 獲取文件列表 userFiles = [] file_list = os.listdir(FILE_PATH+self.account+"/") if not file_list: return else: time.sleep(0.1) files = "" for file in file_list: if file[0] != '.' and os.path.isfile(FILE_PATH + self.account + "/" + file): userFiles.append(file) return userFiles #退出 def do_quit(self): self.sockfd.send((self.account+' QUIT@# ').encode()) self.sockfd.close() sys.exit('謝謝使用') #用戶下載服務(wù)器文件 def do_get(self,filename): self.sockfd.send((self.account+' GET@# '+filename).encode()) data = self.sockfd.recv(128).decode() if data == 'OK': fd = open(FILE_PATH + self.account + "/" + filename,'wb') #復(fù)寫(xiě) while True: data = self.sockfd.recv(1024) if data == b'##': #print("##") break #print("正在寫(xiě)入數(shù)據(jù)") fd.write(data) fd.close() else: print(data) #用戶將文件上傳到服務(wù)器文件夾 def do_put(self,filename): try: fd = open(FILE_PATH + self.account + "/" + filename,'rb') except IOError: print("文件不存在") return # 獲取文件名 filename = filename.split('/')[-1] # else: self.sockfd.send((self.account+' PUT@# '+filename).encode()) data = self.sockfd.recv(128).decode() # 發(fā)送文件 if data == 'OK': while True: data = fd.read(1024) if not data: time.sleep(0.1) self.sockfd.send(b'**') break self.sockfd.send(data) fd.close() return "true" else: print(data) return "false" #刪除用戶文件 def removeU(self,fileName): os.remove(FILE_PATH + self.account + "/" + fileName) return "true" #刪除用戶文件 def removeF(self,fileName): self.sockfd.send((self.account+' delete@# '+fileName).encode()) # 等待回復(fù) data = self.sockfd.recv(128).decode() if data == "OK": return "true" def menu_display(self): print("\n------命令選擇-------") print("*** clist ***") print("*** slist ***") print("*** get list ***") print("*** put list ***") print("*** quit ***") print("----------------------") def run(self,cmd): # 創(chuàng)建套接字 sockfd = socket.socket() try: sockfd.connect(ADDR) except Exception as e: print(e) return result = "" self.sockfd = sockfd # choice(cmd,ftp) if cmd == "slist": result = self.server_list() return result elif cmd == "clist": result = self.client_list() return result elif cmd =='quit': self.do_quit() elif cmd[:3] == 'get': filename = cmd.strip().split(' ')[-1] self.do_get(filename) elif cmd[:3] == 'put': filename = cmd.strip().split(' ')[-1] result = self.do_put(filename) return result elif cmd[:7] == 'removeU': filename = cmd.strip().split(' ')[-1] self.removeU(filename) elif cmd[:7] == 'removeF': filename = cmd.strip().split(' ')[-1] self.removeF(filename) else: print("請(qǐng)輸入正確命令!") if __name__ == "__main__": client = Client("ffy") client.run()
運(yùn)行界面:
總結(jié)
到此這篇關(guān)于python實(shí)現(xiàn)ftp文件傳輸系統(tǒng)的文章就介紹到這了,更多相關(guān)python ftp文件傳輸內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- python和C++共享內(nèi)存?zhèn)鬏攬D像的示例
- Python如何實(shí)現(xiàn)自帶HTTP文件傳輸服務(wù)
- 詳解python logging日志傳輸
- Python socket連接中的粘包、精確傳輸問(wèn)題實(shí)例分析
- python實(shí)現(xiàn)同一局域網(wǎng)下傳輸圖片
- python實(shí)現(xiàn)ftp文件傳輸功能
- python實(shí)現(xiàn)FTP文件傳輸?shù)姆椒ǎǚ?wù)器端和客戶端)
- Python樹(shù)莓派學(xué)習(xí)筆記之UDP傳輸視頻幀操作詳解
- Python實(shí)現(xiàn)基于socket的udp傳輸與接收功能詳解
- python 中Arduino串口傳輸數(shù)據(jù)到電腦并保存至excel表格
- python使用socket實(shí)現(xiàn)的傳輸demo示例【基于TCP協(xié)議】
- python 使用socket傳輸圖片視頻等文件的實(shí)現(xiàn)方式
- python 利用opencv實(shí)現(xiàn)圖像網(wǎng)絡(luò)傳輸
相關(guān)文章
Python創(chuàng)建二維數(shù)組實(shí)例(關(guān)于list的一個(gè)小坑)
下面小編就為大家?guī)?lái)一篇Python創(chuàng)建二維數(shù)組實(shí)例(關(guān)于list的一個(gè)小坑)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-11-11python機(jī)器學(xué)習(xí)deepchecks庫(kù)訓(xùn)練檢查模型特點(diǎn)探索
這篇文章主要介紹了python機(jī)器學(xué)習(xí)deepchecks庫(kù)的訓(xùn)練檢查模型特點(diǎn)實(shí)例探索,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01聊聊通過(guò)celery_one避免Celery定時(shí)任務(wù)重復(fù)執(zhí)行的問(wèn)題
Celery Once 也是利用 Redis 加鎖來(lái)實(shí)現(xiàn), Celery Once 在 Task 類基礎(chǔ)上實(shí)現(xiàn)了 QueueOnce 類,該類提供了任務(wù)去重的功能,今天通過(guò)本文給大家介紹通過(guò)celery_one避免Celery定時(shí)任務(wù)重復(fù)執(zhí)行的問(wèn)題,感興趣的朋友一起看看吧2021-10-10python使用tkinter庫(kù)實(shí)現(xiàn)五子棋游戲
這篇文章主要為大家詳細(xì)介紹了python使用tkinter庫(kù)實(shí)現(xiàn)五子棋游戲,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-06-06如何使用Cython對(duì)python代碼進(jìn)行加密
這篇文章主要介紹了如何使用Cython對(duì)python代碼進(jìn)行加密,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-07-07pytorch模型部署 pth轉(zhuǎn)onnx的方法
這篇文章主要介紹了pytorch模型部署 pth轉(zhuǎn)onnx的相關(guān)知識(shí),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-05-05