Python如何實現(xiàn)按文件夾加密解密其中文件
Python實現(xiàn)按文件夾加密解密其中文件工具
一個使用Python實現(xiàn)的按文件夾加密解密其中文件的程序,包含GUI界面和較為完善的功能。我們將使用PyQt6來創(chuàng)建GUI,使用cryptography庫進(jìn)行加密操作。該程序用到一些模塊/庫:
PyQt6 庫是第三方庫,需要安裝,用于創(chuàng)建圖形用戶界面(GUI)的庫。PyQt6的簡介與安裝,可參見Pyside6 安裝和簡單界面開發(fā)過程詳細(xì)介紹
cryptography 是第三方庫,需要安裝,用于加密和解密的庫。
sys、os 和 base64是Python自帶的標(biāo)準(zhǔn)庫模塊,不需要安裝。
運行效果
源碼如下:
import sys import os from PyQt6.QtWidgets import (QApplication, QWidget, QPushButton, QVBoxLayout, QHBoxLayout, QFileDialog, QLineEdit, QLabel, QMessageBox, QProgressBar, QTextEdit, QSplitter) from PyQt6.QtCore import Qt, QThread, pyqtSignal from PyQt6.QtGui import QFont, QColor from cryptography.fernet import Fernet from cryptography.hazmat.primitives import hashes from cryptography.hazmat.primitives.kdf.pbkdf2 import PBKDF2HMAC import base64 class CryptoWorker(QThread): progress = pyqtSignal(int) finished = pyqtSignal() error = pyqtSignal(str) status = pyqtSignal(str) # 新增狀態(tài)信號 def __init__(self, folder, password, mode): super().__init__() self.folder = folder self.password = password self.mode = mode # 'encrypt' or 'decrypt' def run(self): try: key = self.get_key(self.password) fernet = Fernet(key) total_files = sum([len(files) for r, d, files in os.walk(self.folder)]) processed_files = 0 self.status.emit(f"開始{'加密' if self.mode == 'encrypt' else '解密'}操作...\n") for root, dirs, files in os.walk(self.folder): for file in files: file_path = os.path.join(root, file) relative_path = os.path.relpath(file_path, self.folder) if self.mode == 'encrypt': if file.endswith('.enc'): self.status.emit(f"跳過已加密文件: {relative_path}") else: self.encrypt_file(fernet, file_path) self.status.emit(f"已加密: {relative_path}") else: # decrypt if not file.endswith('.enc'): self.status.emit(f"跳過未加密文件: {relative_path}") else: self.decrypt_file(fernet, file_path) self.status.emit(f"已解密: {relative_path}") processed_files += 1 self.progress.emit(int(processed_files / total_files * 100)) self.status.emit("\n操作完成!") self.finished.emit() except Exception as e: self.error.emit(str(e)) def get_key(self, password): password = password.encode() salt = b'salt_' kdf = PBKDF2HMAC( algorithm=hashes.SHA256(), length=32, salt=salt, iterations=100000, ) key = base64.urlsafe_b64encode(kdf.derive(password)) return key def encrypt_file(self, fernet, file_path): with open(file_path, 'rb') as f: data = f.read() encrypted = fernet.encrypt(data) encrypted_path = file_path + '.enc' with open(encrypted_path, 'wb') as f: f.write(encrypted) os.remove(file_path) def decrypt_file(self, fernet, file_path): with open(file_path, 'rb') as f: data = f.read() try: decrypted = fernet.decrypt(data) decrypted_path = file_path[:-4] # 移除 '.enc' with open(decrypted_path, 'wb') as f: f.write(decrypted) os.remove(file_path) except: raise Exception(f'解密 {file_path} 失敗,可能密碼錯誤') class Encryptor(QWidget): def __init__(self): super().__init__() self.initUI() def initUI(self): self.setWindowTitle('文件夾加密器') self.setGeometry(300, 200, 700, 600) main_layout = QVBoxLayout() # 上半部分:控制面板 control_panel = QWidget() control_layout = QVBoxLayout() # 文件夾選擇 folder_layout = QHBoxLayout() self.folder_input = QLineEdit() folder_button = QPushButton('選擇文件夾') folder_button.clicked.connect(self.select_folder) folder_layout.addWidget(self.folder_input) folder_layout.addWidget(folder_button) control_layout.addLayout(folder_layout) # 密碼輸入 password_layout = QHBoxLayout() password_label = QLabel('密碼:') self.password_input = QLineEdit() self.password_input.setEchoMode(QLineEdit.EchoMode.Password) password_layout.addWidget(password_label) password_layout.addWidget(self.password_input) control_layout.addLayout(password_layout) # 進(jìn)度條 self.progress_bar = QProgressBar() control_layout.addWidget(self.progress_bar) # 加密解密按鈕 button_layout = QHBoxLayout() encrypt_button = QPushButton('加密') decrypt_button = QPushButton('解密') encrypt_button.clicked.connect(lambda: self.process_folder('encrypt')) decrypt_button.clicked.connect(lambda: self.process_folder('decrypt')) button_layout.addWidget(encrypt_button) button_layout.addWidget(decrypt_button) control_layout.addLayout(button_layout) control_panel.setLayout(control_layout) main_layout.addWidget(control_panel) # 分割線 splitter = QSplitter(Qt.Orientation.Vertical) # 注意事項面板 notice_panel = QWidget() notice_layout = QVBoxLayout() notice_label = QLabel("注意事項:") notice_label.setFont(QFont("Arial", 12, QFont.Weight.Bold)) notice_label.setStyleSheet("color: red;") notice_text = QTextEdit() notice_text.setPlainText( "1. 請確保記住密碼,否則將無法解密文件。\n" "2. 請慎重操作!該程序會直接在原文件夾中操作,謹(jǐn)防誤操作。\n" "3. 加密后刪除原文件,解密后刪除加密文件。加密后的文件會有 .enc 擴展名。\n" "4. 本程序支持加密/解密文件夾中的所有文件。\n" "5. 已加密的文件不會被重復(fù)加密,未加密的文件不會被解密。" ) notice_text.setReadOnly(True) notice_text.setFont(QFont("Arial", 11)) notice_text.setStyleSheet("color: #D32F2F;") # 深紅色 notice_layout.addWidget(notice_label) notice_layout.addWidget(notice_text) notice_panel.setLayout(notice_layout) # 日志面板 log_panel = QWidget() log_layout = QVBoxLayout() log_label = QLabel("處理日志:") #log_label.setFont(QFont("Arial", 10, QFont.Weight.Bold)) self.log_text = QTextEdit() self.log_text.setReadOnly(True) #self.log_text.setFont(QFont("Arial", 10)) log_layout.addWidget(log_label) log_layout.addWidget(self.log_text) log_panel.setLayout(log_layout) # 添加面板到分割器 splitter.addWidget(notice_panel) splitter.addWidget(log_panel) main_layout.addWidget(splitter) self.setLayout(main_layout) def select_folder(self): folder = QFileDialog.getExistingDirectory(self, "選擇文件夾") self.folder_input.setText(folder) def process_folder(self, mode): folder = self.folder_input.text() password = self.password_input.text() if not folder or not password: QMessageBox.warning(self, '警告', '請選擇文件夾并輸入密碼') return # 清空日志 self.log_text.clear() self.worker = CryptoWorker(folder, password, mode) self.worker.progress.connect(self.update_progress) self.worker.finished.connect(self.process_finished) self.worker.error.connect(self.process_error) self.worker.status.connect(self.update_log) # 連接狀態(tài)信號 self.worker.start() def update_progress(self, value): self.progress_bar.setValue(value) def update_log(self, message): self.log_text.append(message) def process_finished(self): QMessageBox.information(self, '成功', '操作完成') self.progress_bar.setValue(0) def process_error(self, error_msg): QMessageBox.critical(self, '錯誤', f'發(fā)生錯誤:{error_msg}') self.progress_bar.setValue(0) self.log_text.append(f"\n錯誤: {error_msg}") if __name__ == '__main__': app = QApplication(sys.argv) ex = Encryptor() ex.show() sys.exit(app.exec())
到此這篇關(guān)于Python如何實現(xiàn)按文件夾加密解密其中文件的文章就介紹到這了,更多相關(guān)Python加密解密文件內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
python opencv鼠標(biāo)畫矩形框之cv2.rectangle()函數(shù)
鼠標(biāo)操作屬于用戶接口設(shè)計,以前一直使用Qt來做,但是如果只需要簡單的鼠標(biāo),鍵盤操作,直接調(diào)用opencv庫的函數(shù)也未嘗不可,下面這篇文章主要給大家介紹了關(guān)于python opencv鼠標(biāo)畫矩形框cv2.rectangle()函數(shù)的相關(guān)資料,需要的朋友可以參考下2021-10-10PySide(PyQt)使用QPropertyAnimation制作動態(tài)界面的示例代碼
文章介紹了如何使用PySide或PyQt的QPropertyAnimation類來創(chuàng)建動態(tài)界面效果,感興趣的朋友一起看看吧2025-03-03python執(zhí)行數(shù)據(jù)庫的查詢操作實例講解
在本篇文章里小編給大家整理了一篇關(guān)于python執(zhí)行數(shù)據(jù)庫的查詢操作實例講解內(nèi)容,有需要的朋友們可以參考學(xué)習(xí)下。2021-10-10Python 關(guān)于模塊和加載模塊的實現(xiàn)
這篇文章主要介紹了Python 關(guān)于模塊和加載模塊的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-03-03Python使用Windows API創(chuàng)建窗口示例【基于win32gui模塊】
這篇文章主要介紹了Python使用Windows API創(chuàng)建窗口操作,結(jié)合實例形式分析了Python基于win32gui模塊調(diào)用Windows API創(chuàng)建窗口具體操作步驟與相關(guān)實現(xiàn)技巧,需要的朋友可以參考下2018-05-05