Python實(shí)現(xiàn)的FTP通信客戶端與服務(wù)器端功能示例
本文實(shí)例講述了Python實(shí)現(xiàn)的FTP通信客戶端與服務(wù)器端功能。分享給大家供大家參考,具體如下:
一 代碼
1、服務(wù)端代碼
import socket
import threading
import os
import struct
#用戶賬號(hào)、密碼、主目錄
#也可以把這些信息存放到數(shù)據(jù)庫(kù)中
users = {'zhangsan':{'pwd':'zhangsan1234', 'home':r'c:\python 3.5'},
'lisi':{'pwd':'lisi567', 'home':'c:\\'}}
def server(conn,addr, home):
print('新客戶端:'+str(addr))
#進(jìn)入當(dāng)前用戶主目錄
os.chdir(home)
while True:
data = conn.recv(100).decode().lower()
#顯示客戶端輸入的每一條命令
print(data)
#客戶端退出
if data in ('quit', 'q'):
break
#查看當(dāng)前文件夾的文件列表
elif data in ('list', 'ls', 'dir'):
files = str(os.listdir(os.getcwd()))
files = files.encode()
conn.send(struct.pack('I', len(files)))
conn.send(files)
#切換至上一級(jí)目錄
elif ''.join(data.split()) == 'cd..':
cwd = os.getcwd()
newCwd = cwd[:cwd.rindex('\\')]
#考慮根目錄的情況
if newCwd[-1] == ':':
newCwd += '\\'
#限定用戶主目錄
if newCwd.lower().startswith(home):
os.chdir(newCwd)
conn.send(b'ok')
else:
conn.send(b'error')
#查看當(dāng)前目錄
elif data in ('cwd', 'cd'):
conn.send(str(os.getcwd()).encode())
elif data.startswith('cd '):
#指定最大分隔次數(shù),考慮目標(biāo)文件夾帶有空格的情況
#只允許使用相對(duì)路徑進(jìn)行跳轉(zhuǎn)
data = data.split(maxsplit=1)
if len(data) == 2 and os.path.isdir(data[1]) \
and data[1]!=os.path.abspath(data[1]):
os.chdir(data[1])
conn.send(b'ok')
else:
conn.send(b'error')
#下載文件
elif data.startswith('get '):
data = data.split(maxsplit=1)
#檢查文件是否存在
if len(data) == 2 and os.path.isfile(data[1]):
conn.send(b'ok')
fp = open(data[1], 'rb')
while True:
content = fp.read(4096)
#發(fā)送文件結(jié)束
if not content:
conn.send(b'overxxxx')
break
#發(fā)送文件內(nèi)容
conn.send(content)
if conn.recv(10) == b'ok':
continue
fp.close()
else:
conn.send(b'no')
#無(wú)效命令
else:
pass
conn.close()
print(str(addr)+'關(guān)閉連接')
#創(chuàng)建Socket,監(jiān)聽本地端口,等待客戶端連接
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.bind(('', 10600))
sock.listen(5)
while True:
conn, addr = sock.accept()
#驗(yàn)證客戶端輸入的用戶名和密碼是否正確
userId, userPwd = conn.recv(1024).decode().split(',')
if userId in users and users[userId]['pwd'] == userPwd:
conn.send(b'ok')
#為每個(gè)客戶端連接創(chuàng)建并啟動(dòng)一個(gè)線程,參數(shù)為連接、客戶端地址、客戶主目錄
home = users[userId]['home']
t = threading.Thread(target=server, args=(conn,addr,home))
t.daemon = True
t.start()
else:
conn.send(b'error')
2、客戶端代碼
import socket
import sys
import re
import struct
import getpass
def main(serverIP):
sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
sock.connect((serverIP, 10600))
userId = input('請(qǐng)輸入用戶名:')
#使用getpass模塊的getpass()方法獲取密碼,不回顯
userPwd = getpass.getpass('請(qǐng)輸入密碼:')
message = userId+','+userPwd
sock.send(message.encode())
login = sock.recv(100)
#驗(yàn)證是否登錄成功
if login == b'error':
print('用戶名或密碼錯(cuò)誤')
return
#整數(shù)編碼大小
intSize = struct.calcsize('I')
while True:
#接收客戶端命令,其中##>是提示符
command = input('##> ').lower().strip()
#沒有輸入任何有效字符,提前進(jìn)入下一次循環(huán),等待用戶繼續(xù)輸入
if not command:
continue
#向服務(wù)端發(fā)送命令
command = ' '.join(command.split())
sock.send(command.encode())
#退出
if command in ('quit', 'q'):
break
#查看文件列表
elif command in ('list', 'ls', 'dir'):
loc_size = struct.unpack('I', sock.recv(intSize))[0]
files = eval(sock.recv(loc_size).decode())
for item in files:
print(item)
#切換至上一級(jí)目錄
elif ''.join(command.split()) == 'cd..':
print(sock.recv(100).decode())
#查看當(dāng)前工作目錄
elif command in ('cwd', 'cd'):
print(sock.recv(1024).decode())
#切換至子文件夾
elif command.startswith('cd '):
print(sock.recv(100).decode())
#從服務(wù)器下載文件
elif command.startswith('get '):
isFileExist = sock.recv(20)
#文件不存在
if isFileExist != b'ok':
print('error')
#文件存在,開始下載
else:
print('downloading.', end='')
fp = open(command.split()[1], 'wb')
while True:
print('.', end='')
data = sock.recv(4096)
if data == b'overxxxx':
break
fp.write(data)
sock.send(b'ok')
fp.close()
print('ok')
#無(wú)效命令
else:
print('無(wú)效命令')
sock.close()
if __name__ == '__main__':
if len(sys.argv) != 2:
print('Usage:{0} serverIPAddress'.format(sys.argv[0]))
exit()
serverIP = sys.argv[1]
if re.match(r'^\d{1,3}.\d{1,3}.\d{1,3}.\d{1,3}$', serverIP):
main(serverIP)
else:
print('服務(wù)器地址不合法')
exit()
二 運(yùn)行結(jié)果
客戶端運(yùn)行結(jié)果

更多關(guān)于Python相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《Python Socket編程技巧總結(jié)》、《Python URL操作技巧總結(jié)》、《Python數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Python函數(shù)使用技巧總結(jié)》、《Python字符串操作技巧匯總》及《Python入門與進(jìn)階經(jīng)典教程》
希望本文所述對(duì)大家Python程序設(shè)計(jì)有所幫助。
相關(guān)文章
利用python為PostgreSQL的表自動(dòng)添加分區(qū)
這篇文章主要介紹了利用python為PostgreSQL的表自動(dòng)添加分區(qū),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01
100 個(gè) Python 小例子(練習(xí)題一)
這篇文章主要介紹 Python 小例子,有數(shù)字組合、個(gè)稅計(jì)算、完全平方數(shù)、三數(shù)排序、斐波那契數(shù)列、copy、九九乘法表、暫停一秒輸出等多個(gè)實(shí)例,需要的朋友可以參考一下2021-10-10
Python使用paramiko連接遠(yuǎn)程服務(wù)器執(zhí)行Shell命令的實(shí)現(xiàn)
這篇文章主要介紹了Python使用paramiko連接遠(yuǎn)程服務(wù)器執(zhí)行Shell命令的實(shí)現(xiàn),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03
python批量提取圖片信息并保存的實(shí)現(xiàn)
這篇文章主要介紹了python批量提取圖片信息并保存的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-02-02

