溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

Python如何實現(xiàn)FTP通信客戶端與服務(wù)器端功能

發(fā)布時間:2021-03-24 10:48:33 來源:億速云 閱讀:180 作者:小新 欄目:開發(fā)技術(shù)

這篇文章將為大家詳細(xì)講解有關(guān)Python如何實現(xiàn)FTP通信客戶端與服務(wù)器端功能,小編覺得挺實用的,因此分享給大家做個參考,希望大家閱讀完這篇文章后可以有所收獲。

一 代碼

1、服務(wù)端代碼

import socket
import threading
import os
import struct
#用戶賬號、密碼、主目錄
#也可以把這些信息存放到數(shù)據(jù)庫中
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)
    #切換至上一級目錄
    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)文件夾帶有空格的情況
      #只允許使用相對路徑進(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')
    #無效命令
    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()
  #驗證客戶端輸入的用戶名和密碼是否正確
  userId, userPwd = conn.recv(1024).decode().split(',')
  if userId in users and users[userId]['pwd'] == userPwd:
    conn.send(b'ok')
    #為每個客戶端連接創(chuàng)建并啟動一個線程,參數(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('請輸入用戶名:')
  #使用getpass模塊的getpass()方法獲取密碼,不回顯
  userPwd = getpass.getpass('請輸入密碼:')
  message = userId+','+userPwd
  sock.send(message.encode())
  login = sock.recv(100)
  #驗證是否登錄成功
  if login == b'error':
    print('用戶名或密碼錯誤')
    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)
    #切換至上一級目錄
    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')
    #無效命令
    else:
      print('無效命令')
  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()

二 運行結(jié)果

客戶端運行結(jié)果

Python如何實現(xiàn)FTP通信客戶端與服務(wù)器端功能

關(guān)于“Python如何實現(xiàn)FTP通信客戶端與服務(wù)器端功能”這篇文章就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,使各位可以學(xué)到更多知識,如果覺得文章不錯,請把它分享出去讓更多的人看到。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI