python

关注公众号 jb51net

关闭
首页 > 脚本专栏 > python > Python EXE程序更新

Python+PyQt5开发一个Windows EXE程序在线更新工具

作者:weixin_46244623

这篇文章主要为大家详细介绍了Python+PyQt5开发一个Windows EXE程序在线更新工具,可以自动检测新版本并完成在线升级,感兴趣的小伙伴可以了解下

一、前言

在使用 PyQt5 + PyInstaller 开发 Windows 桌面工具时,一个非常现实的问题是:

程序如何自动检测新版本,并完成在线升级?

本文基于一个真实可用、已落地的更新方案,实现了:

非常适合 工具类 / 内部系统 / 单机 EXE 程序

二、运行效果说明

整体更新流程如下:

三、环境与依赖说明

Python 版本

Python 3.8+

pip 依赖安装

pip install PyQt5
pip install requests
pip install packaging

(如需打包 EXE)

pip install pyinstaller

四、服务器端准备

版本号文件(version.txt)

1.2.0

更新包(update.zip)

update.zip
├── update.exe
├── logo.png(可选)

五、完整代码

SoftVesion.py
def current_version():
    """
    Returns the version of the package
    """
    return "1.1.0"
def latest_version():
    """
    Returns the latest version of the package
    """
    return "1.1.0"
def upgrade_date():
    """
    Returns the time of the upgrade
    """
    return "2024年1月19日"
def current_title():
    """
    Returns the title of the package
    """
    return "Excel常用小工具"
def remote_version_url():
    """
    Returns the url of the latest version
    """
    return "http://82.157.62.197:97/version.txt"
def update_url():
    """
    Returns the url of the version info
    """
    return "http://192.168.31.219:8080/update.zip"

UpdateApp.py

import os
import zipfile
import requests
from packaging import version
from PyQt5.QtCore import QThread, pyqtSignal, Qt
from PyQt5.QtWidgets import QApplication,QDialog, QVBoxLayout,QMainWindow, QPushButton, QMessageBox, QProgressDialog,QLabel,QAction,QDesktopWidget
import shutil
from .SoftVesion import current_version,latest_version,remote_version_url,update_url,current_title
from PyQt5.QtGui import QStandardItemModel, QStandardItem,QIcon
import sys
import time
import subprocess
class DownloadThread(QThread):
    progressChanged = pyqtSignal(int)
    downloadFinished = pyqtSignal()
    unableToFetchUpdateLink = pyqtSignal()

    def __init__(self, url, save_path):
        super().__init__()
        self.url = url
        self.save_path = save_path

    def run(self):
        try:
            response = requests.get(self.url, stream=True)
            if response.status_code == 200:
                total_size = int(response.headers.get('content-length', 0))

                downloaded_size = 0
                with open(self.save_path, 'wb') as file:
                    for data in response.iter_content(chunk_size=8192):
                        file.write(data)
                        downloaded_size += len(data)
                        progress = int((downloaded_size / total_size) * 100)
                        self.progressChanged.emit(progress)

                self.downloadFinished.emit()
            else:
                print("Error fetching update link:")
                self.unableToFetchUpdateLink.emit()
        except Exception as e:
            print("Error fetching update link:", e)
            self.unableToFetchUpdateLink.emit()

class UpdateApp(QDialog):
    def __init__(self,parent=None):
        super().__init__(parent)
        self.current_version = current_version()  # 当前应用程序版本号
        self.latest_version = latest_version()
        self.progress_dialog = None
        self.download_thread = None
        self.remote_version_url= remote_version_url()
        self.update_url = update_url()
        self.init_ui()
        #self.show_update_notification()
    def move_to_center(self, widget):
        # 将 widget 移动到屏幕中央
        desktop_center = QDesktopWidget().availableGeometry().center()
        widget_frame = widget.frameGeometry()
        widget_frame.moveCenter(desktop_center)
        widget.move(widget_frame.topLeft())
    def init_ui(self):
        self.setFixedSize(300, 200)
        self.setWindowTitle("检查更新")
        icon = QIcon("icon.png")
        self.setWindowIcon(icon)

        self.version_label = QLabel("当前版本号: " + self.current_version, self)
        self.version_label.setAlignment(Qt.AlignCenter) 
        self.version_label.setGeometry(100, 10, 110, 20)

        self.update_button = QPushButton("获取最新版本", self)
        self.update_button.setGeometry(100, 60, 110, 40)
        self.update_button.clicked.connect(self.show_update_notification)

        if self.download_thread:
            self.download_thread.unableToFetchUpdateLink.connect(self.show_unable_to_fetch_update_link)

    def show_unable_to_fetch_update_link(self):
        QMessageBox.critical(self, "无法获取更新链接", "无法获取更新链接,请检查网络连接。")

    def update_app(self):
        try:
            remote_version = self.get_remote_version()
            if version.parse(remote_version) > version.parse(self.current_version):
                if self.has_network_connection():
                    self.perform_upgrade()
                else:
                    QMessageBox.information(self, "无法获取版本号", "无法获取远程版本号,请检查网络连接。")
            else:
                QMessageBox.information(self, "无需升级", "当前应用程序已经是最新版本。")
        except:
            QMessageBox.information(self, "无法获取版本号", "无法获取远程版本号,请检查网络连接。")

    def get_remote_version(self):
        try:
            response = requests.get(self.remote_version_url)
            if response.status_code == 200:
                return response.text.strip()
            return self.latest_version
        except:
            return self.latest_version

    def perform_upgrade(self):
        update_url = self.update_url
        temp_zip_path = "temp_update.zip"

        self.download_thread = DownloadThread(update_url, temp_zip_path)
        self.download_thread.progressChanged.connect(self.update_progress_bar)
        self.download_thread.downloadFinished.connect(self.handle_download_finished)

        self.progress_dialog = QProgressDialog("正在下载更新...", None, 0, 100, self)
        self.progress_dialog.setWindowModality(Qt.WindowModal)
        self.progress_dialog.setAutoClose(False)
        self.progress_dialog.setAutoReset(False)
        self.progress_dialog.setWindowTitle("下载进度")
        self.move_to_center(self.progress_dialog)
        self.progress_dialog.canceled.connect(self.download_thread.quit)

        self.download_thread.start()

    def update_progress_bar(self, progress):
        self.progress_dialog.setValue(progress)

    def handle_download_finished(self):
        self.progress_dialog.hide()
        if os.path.exists("temp_update.zip"):
            self.process_download("temp_update.zip")
            os.remove("temp_update.zip")
        sys.exit()

    def has_network_connection(self):
        try:
            requests.get(remote_version_url(), timeout=5)
            return True
        except:
            return False

    def show_update_notification(self):
        try:
            remote_version = self.get_remote_version()
            if version.parse(remote_version) > version.parse(self.current_version):
                reply = QMessageBox.question(
                    self, "版本更新提示",
                    f"有新版本可用:{remote_version}\n您当前的版本:{self.current_version}\n是否要更新?",
                    QMessageBox.Yes | QMessageBox.No
                )
                if reply == QMessageBox.Yes:
                    self.update_app()
            else:
                QMessageBox.information(self, "版本更新提示", "您当前的版本是最新版本。")
        except:
            pass

if __name__ == "__main__":
    app = QApplication([])
    window = UpdateApp()
    window.show()
    app.exec_()

六、关键设计说明

使用 QThread 防止界面卡死

使用 packaging.version 进行版本比较

ZIP 更新包支持扩展文件

EXE 覆盖前自动备份

更新完成后安全退出

七、适用场景

PyQt5 工具软件

内部办公系统

单机 EXE 程序

不依赖第三方更新框架

八、总结

本文提供的是一个 可直接落地、真实可用 的 PyQt5 自动更新方案,无需复杂服务器逻辑,维护成本极低。

到此这篇关于Python+PyQt5开发一个Windows EXE程序在线更新工具的文章就介绍到这了,更多相关Python EXE程序更新内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文