基于Pyside6开发一个通用的在线升级工具

UI

main.ui

<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0"><class>MainWindow</class><widget class="QMainWindow" name="MainWindow"><property name="geometry"><rect><x>0</x><y>0</y><width>562</width><height>96</height></rect></property><property name="font"><font><family>微软雅黑</family><pointsize>10</pointsize></font></property><property name="windowTitle"><string>在线升级</string></property><widget class="QWidget" name="centralwidget"><widget class="QProgressBar" name="pbDownload"><property name="geometry"><rect><x>10</x><y>60</y><width>541</width><height>31</height></rect></property><property name="value"><number>0</number></property><property name="alignment"><set>Qt::AlignCenter</set></property><property name="textVisible"><bool>false</bool></property></widget><widget class="QLabel" name="label"><property name="geometry"><rect><x>10</x><y>10</y><width>541</width><height>41</height></rect></property><property name="font"><font><family>微软雅黑</family><pointsize>10</pointsize></font></property><property name="text"><string>在线升级中,请稍等...</string></property><property name="alignment"><set>Qt::AlignCenter</set></property></widget></widget></widget><resources/><connections/>
</ui>

main.py

# -*- coding: utf-8 -*-################################################################################
## Form generated from reading UI file 'main.ui'
##
## Created by: Qt User Interface Compiler version 6.6.3
##
## WARNING! All changes made in this file will be lost when recompiling UI file!
################################################################################from PySide6.QtCore import (QCoreApplication, QDate, QDateTime, QLocale,QMetaObject, QObject, QPoint, QRect,QSize, QTime, QUrl, Qt)
from PySide6.QtGui import (QBrush, QColor, QConicalGradient, QCursor,QFont, QFontDatabase, QGradient, QIcon,QImage, QKeySequence, QLinearGradient, QPainter,QPalette, QPixmap, QRadialGradient, QTransform)
from PySide6.QtWidgets import (QApplication, QLabel, QMainWindow, QProgressBar,QSizePolicy, QWidget)class Ui_MainWindow(object):def setupUi(self, MainWindow):if not MainWindow.objectName():MainWindow.setObjectName(u"MainWindow")MainWindow.resize(562, 96)font = QFont()font.setFamilies([u"\u5fae\u8f6f\u96c5\u9ed1"])font.setPointSize(10)MainWindow.setFont(font)self.centralwidget = QWidget(MainWindow)self.centralwidget.setObjectName(u"centralwidget")self.pbDownload = QProgressBar(self.centralwidget)self.pbDownload.setObjectName(u"pbDownload")self.pbDownload.setGeometry(QRect(10, 60, 541, 31))self.pbDownload.setValue(0)self.pbDownload.setAlignment(Qt.AlignCenter)self.pbDownload.setTextVisible(False)self.label = QLabel(self.centralwidget)self.label.setObjectName(u"label")self.label.setGeometry(QRect(10, 10, 541, 41))self.label.setFont(font)self.label.setAlignment(Qt.AlignCenter)MainWindow.setCentralWidget(self.centralwidget)self.retranslateUi(MainWindow)QMetaObject.connectSlotsByName(MainWindow)# setupUidef retranslateUi(self, MainWindow):MainWindow.setWindowTitle(QCoreApplication.translate("MainWindow", u"\u5728\u7ebf\u5347\u7ea7", None))self.label.setText(QCoreApplication.translate("MainWindow", u"\u5728\u7ebf\u5347\u7ea7\u4e2d,\u8bf7\u7a0d\u7b49...", None))# retranslateUi

PyDesigner

主程序

upgrade_service.py

import timeimport configparser
import requests
import sys
import threading
import psutil
import os
import zipfile
import warningsfrom PySide6.QtWidgets import QApplication, QMainWindow
from PySide6.QtCore import Qt, QTimer
from PySide6.QtGui import QIcon, QMouseEvent
from ui.main import Ui_MainWindow
from qt_material import apply_stylesheet'''
pip install pyinstaller pyside6 qt_material psutil requestspyside6-uic -o ../../ui/main.py ../../ui/main.uipyinstaller --clean --icon=upgrade.ico D:\\1_local_project\IMVision\IMVDXUpgrade\\upgrade_service.py
pyinstaller upgrade_service.spec --noconfirm
'''warnings.filterwarnings('ignore', category=DeprecationWarning)class UpgradeInfo(object):def __init__(self):self.username = ''self.version = ''self.url = ''self.extra_file = 0self.extra_path = ''self.remark = ''class MainWindow(QMainWindow, Ui_MainWindow):def __init__(self):super(MainWindow, self).__init__()self.setupUi(self)self.setWindowFlags(Qt.FramelessWindowHint)# 开启鼠标追踪self.setMouseTracking(True)# 记录鼠标按下时的位置self.drag_start_position = Noneself.setWindowIcon(QIcon('icons/upgrade.ico'))self.label.setStyleSheet(self.label.styleSheet() + 'font-size: 20px;')self.upgrade_finished = Falseself.finished_chunk_size = 0self.total_file_count = 0threading.Thread(target=self.download_file).start()self.timer = QTimer(self)self.timer.timeout.connect(self.listen_download_state)self.timer.start(100)def mousePressEvent(self, event: QMouseEvent) -> None:if event.button() == Qt.LeftButton:self.drag_start_position = event.pos()def mouseMoveEvent(self, event: QMouseEvent) -> None:if self.drag_start_position is not None:delta = event.pos() - self.drag_start_positionself.move(self.pos() + delta)def mouseReleaseEvent(self, event: QMouseEvent) -> None:if event.button() == Qt.LeftButton:self.drag_start_position = Nonedef listen_download_state(self):if self.upgrade_finished:self.timer.stop()cf.set('upgrade', 'state', '0')cf.set('upgrade', 'currentVersion', new_version)cf.write(open('service.ini', 'w'))if self.total_file_count > 0:os.startfile(main_file)time.sleep(0.2)self.close()def updateLabel(self, c, t, overwrite=False):self.label.setText('正在升级...(' + c + '/' + t + ')') if not overwrite else self.label.setText(t)def download_file(self):upgrade_files = []self.finished_count = 0self.total_file_count = 0self.weight_file = 0self.extra_file = 0response = requests.get(upgrade_url + 'upgrade/v1/info?username=' + username, timeout=2)if response.status_code == 200:res = response.json()if res['code'] == 0 and res['data'] is not None:data = res['data']self.weight_file_tag = data['weight_file']self.extra_file_tag = data['extra_file']self.total_file_count = data['total_file_count']upgrade_files = (data['extra_path'].replace('[', '').replace(']', '').replace(' ', '').split(','))if self.total_file_count == 0:self.updateLabel('0', '当前已是最新版本!', True)time.sleep(1)self.upgrade_finished = Truereturnif self.weight_file_tag == 1:self.total_file_count += len(weight_files)self.updateLabel('0', str(self.total_file_count))time.sleep(0.2)# 下载主程序try:response = requests.get(upgrade_url + 'upgrade/v1/download?filename=' + main_file + '&username=' + username,stream=True)end_process(main_file)time.sleep(0.5)end_process(main_file)time.sleep(0.5)self.pbDownload.setMaximum(int(response.headers.get('Content-Length', 0)))with open(main_file_tmp, 'wb') as file:self.finished_chunk_size = 0for chunk in response.iter_content(chunk_size=8192 * 10):if chunk:file.write(chunk)self.finished_chunk_size += len(chunk)self.pbDownload.setValue(self.finished_chunk_size)QApplication.processEvents()time.sleep(0.5)# 下载完成,删除main_file,将临时文件改成main_fileos.remove(main_file)os.rename(main_file_tmp, main_file)except:# 下载失败,删除临时文件os.remove(main_file_tmp)self.updateLabel('0', '升级失败!请检查网络连接!', True)time.sleep(1)self.upgrade_finished = Truereturnself.updateLabel('1', str(self.total_file_count))time.sleep(0.2)self.finished_count = 1if self.weight_file_tag == 1:for weight_file in weight_files:weight_file_tmp = weight_file + '.tmp'try:# 下载权重文件response = requests.get(upgrade_url + 'upgrade/v1/download?filename=' + weight_file + '&username=' + username,stream=True)self.pbDownload.setMaximum(int(response.headers.get('Content-Length', 0)))with open('weights/' + weight_file_tmp, 'wb') as file:self.finished_chunk_size = 0for chunk in response.iter_content(chunk_size=8192 * 10):if chunk:file.write(chunk)self.finished_chunk_size += len(chunk)self.pbDownload.setValue(self.finished_chunk_size)QApplication.processEvents()time.sleep(0.5)# 下载完成,删除weight_file,将临时文件改成weight_fileos.remove('weights/' + weight_file)os.rename('weights/' + weight_file_tmp, 'weights/' + weight_file)self.finished_count += 1except:# 下载失败,删除临时文件os.remove('weights/' + weight_file_tmp)self.updateLabel('0', '升级失败!请检查网络连接!', True)time.sleep(1)self.upgrade_finished = Truereturnself.updateLabel(str(self.finished_count), str(self.total_file_count))# 最后下载第三方文件(zip格式)if self.extra_file_tag == 1:try:for idx, filename in enumerate(upgrade_files):response = requests.get(upgrade_url + 'upgrade/v1/download?filename=' + filename + '&username=' + username, stream=True)extra_file_zip = '_internal\\' + filenameself.pbDownload.setMaximum(int(response.headers.get('Content-Length', 0)))with open(extra_file_zip, 'wb') as file:self.finished_chunk_size = 0for chunk in response.iter_content(chunk_size=8192):if chunk:file.write(chunk)self.finished_chunk_size += len(chunk)self.pbDownload.setValue(self.finished_chunk_size)QApplication.processEvents()# 解压缩zipunzip_file(extra_file_zip, './_internal')self.updateLabel(str(idx + self.finished_count + 1), str(self.total_file_count))except:self.updateLabel('0', '升级失败!请检查网络连接!', True)time.sleep(1)self.upgrade_finished = Truereturnself.updateLabel('0', '升级完成!启动新版本...', True)time.sleep(0.5)self.upgrade_finished = Truedef close(self):super(MainWindow, self).close()def end_process(process_name):try:if is_process_running_by_name(process_name):os.system('taskkill /f /im ' + process_name)except:passdef is_process_running_by_name(process_name):for proc in psutil.process_iter(['name']):if proc.info['name'].find(process_name) == 0:return Truereturn Falsedef unzip_file(zip_path, extract_path='.'):if not os.path.exists(extract_path):os.makedirs(extract_path)with zipfile.ZipFile(zip_path, 'r') as zip_ref:zip_ref.extractall(extract_path)os.remove(zip_path)extra = {'font_family': '微软雅黑','density_scale': '0','button_shape': 'default','pyside6': True
}if __name__ == '__main__':cf = configparser.ConfigParser()cf_file = 'service.ini'if os.path.exists(cf_file):cf.read(cf_file)upgrade_url = cf.get('server', 'url')main_file = cf.get('upgrade', 'mainfile')weight_files = cf.get('upgrade', 'weightfiles').split(',')main_file_tmp = main_file + '.tmp'current_version = cf.get('upgrade', 'currentversion')new_version = cf.get('upgrade', 'newversion')state = cf.getint('upgrade', 'state')username = cf.get('sys', 'username')else:sys.exit(0)if state == 0 or (current_version == new_version):sys.exit(0)app = QApplication(sys.argv)main_window = MainWindow()apply_stylesheet(app, theme='dark_teal2.xml', extra=extra)main_window.show()sys.exit(app.exec())

配置文件 service.ini

后台服务(Java)

    @ApiOperation("下载升级文件")@RequestMapping("download")public void download(@RequestParam String filename, @RequestParam String username) {UpgradeSetDO upgradeSetDO = upgradeSetMapper.selectOne(new QueryWrapper<UpgradeSetDO>().eq("username", username).eq("state", 1).last(" order by id desc limit 1"));ServletRequestAttributes requestAttributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();assert requestAttributes != null;HttpServletResponse response = requestAttributes.getResponse();assert response != null;String filePath = "C:\\IMVD\\UPGRADE\\" + username + "\\" + upgradeSetDO.getVersion() + "\\" + filename;File file = new File(filePath);response.setHeader("Content-Disposition", "attachment;filename=" + filename);response.setHeader("Content-Length", String.valueOf(file.length()));try (OutputStream outputStream = response.getOutputStream();BufferedInputStream bis = new BufferedInputStream(new FileInputStream(filePath))) {byte[] buff = new byte[1024];int i = bis.read(buff);while (i != -1) {outputStream.write(buff, 0, buff.length);outputStream.flush();i = bis.read(buff);}} catch (IOException e) {log.error(e.getMessage());}}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/888264.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Redis(配置文件属性解析)

一、tcp-backlog深度解析 tcp-backlog是一个TCP连接的队列&#xff0c;主要用于解决高并发场景下客户端慢连接问题。配置文件中的“511”就是队列的长度&#xff0c;对联与TCP的三次握手有关&#xff0c;不同的linux内核&#xff0c;backlog队列中存放的元素&#xff08;客户端…

24.12.02 Element

import { createApp } from vue // 引入elementPlus js库 css库 import ElementPlus from element-plus import element-plus/dist/index.css //中文语言包 import zhCn from element-plus/es/locale/lang/zh-cn //图标库 import * as ElementPlusIconsVue from element-plus/i…

mybatis-plus 对于属性为null字段不更新

MyBatis-Plus 默认情况下会根据字段的值是否为 null 来决定是否生成对应的 UPDATE 语句。这是由 更新策略 决定的&#xff0c;默认的行为是 忽略 null 值&#xff0c;即如果字段值为 null&#xff0c;该字段将不会出现在 UPDATE 语句中。 默认行为分析 MyBatis-Plus 默认的 Fi…

C++小问题

怎么分辨const修饰的是谁 是限定谁不能被改变的&#xff1f; 在C中&#xff0c;const关键字的用途和位置非常关键&#xff0c;它决定了谁不能被修改。const可以修饰变量、指针、引用等不同的对象&#xff0c;并且具体的作用取决于const的修饰位置。理解const的规则能够帮助我们…

在线家具商城基于 SpringBoot:设计模式与实现方法探究

第3章 系统分析 用户的需求以及与本系统相似的在市场上存在的其它系统可以作为系统分析中参考的资料&#xff0c;分析人员可以根据这些信息确定出本系统具备的功能&#xff0c;分析出本系统具备的性能等内容。 3.1可行性分析 尽管系统是根据用户的要求进行制作&#xff0c;但是…

TS问题之class

类 派生类包含了一个构造函数&#xff0c;它 必须调用 super()&#xff0c;它会执行基类的构造函数。 而且&#xff0c;在构造函数里访问 this的属性之前&#xff0c;我们 一定要调用 super()。 这个是TypeScript强制执行的一条重要规则。public 在TypeScript里&#xff0c;成…

TongRDS分布式内存数据缓存中间件

命令 优势 支持高达10亿级的数据缓冲&#xff0c;内存优化管理&#xff0c;避免GC性能劣化。 高并发系统设计&#xff0c;可充分利用多CPU资源实现并行处理。 数据采用key-value多索引方式存储&#xff0c;字段类型和长度可配置。 支持多台服务并行运行&#xff0c;服务之间可互…

项目整合logback日志打印线程id

项目打印日志能帮助我们解决很多的问题&#xff0c;提示我们出现的问题&#xff0c;通过日志我们可以准确的定位问题快速找到问题点解决问题。 <?xml version"1.0" encoding"UTF-8"?> <!-- 日志级别从低到高分为TRACE < DEBUG < INFO &l…

Flutter-Web打包后上线白屏

问题描述 Flutter上线后进行测试发现界面白屏&#xff0c;打开开发者模式查看网络发现加载main.js文件404 问题原因 我上线的地址是https://xxx:8091/homedots,但是我打包后的index文件中的baseUrl是"/",将地址改成”/homedots/"&#xff0c;注意homedots后面…

算法训练营day22(二叉树08:二叉搜索树的最近公共祖先,插入,删除)

第六章 二叉树part08 今日内容&#xff1a; ● 235. 二叉搜索树的最近公共祖先 ● 701.二叉搜索树中的插入操作 ● 450.删除二叉搜索树中的节点 详细布置 235. 二叉搜索树的最近公共祖先 相对于 二叉树的最近公共祖先 本题就简单一些了&#xff0c;因为 可以利用二叉搜索树的…

Rust循环引用与多线程并发

循环引用与自引用 循环引用的概念 循环引用指的是两个或多个对象之间相互持有对方的引用。在 Rust 中&#xff0c;由于所有权和生命周期的严格约束&#xff0c;直接创建循环引用通常会导致编译失败。例如&#xff1a; // 错误的循环引用示例 struct Node {next: Option<B…

ambari metrics单机模式改成集群模式

最近碰到了ambari平台ambari metrics相关的lib较大&#xff0c;导致系统盘使用率较高。今天对这个组件进行转移到其他磁盘使用率低的服务器上&#xff0c;在安装好metrice collector组件后&#xff0c;发现启动时一直报如下错误。 通过报错可以定位到&#xff0c;该组件的模式是…

前端 递归优化

在前端开发中&#xff0c;递归是一种常见的编程技巧&#xff0c;但它也可能带来性能问题&#xff0c;特别是当递归深度很深或递归调用非常频繁时。以下是一些优化递归的方法&#xff1a; 1. 尾递归优化 尾递归是指递归调用是函数中的最后一个操作&#xff0c;没有额外的计算。…

Python 面向对象编程详解

Python 面向对象编程详解 面向对象编程&#xff08;OOP&#xff09;是一种编程范式&#xff0c;它使用“对象”来设计软件。在 Python 中&#xff0c;面向对象编程非常强大&#xff0c;允许开发者通过类&#xff08;class&#xff09;和对象&#xff08;object&#xff09;来模…

langchain实现基于sql的问答

1. 数据准备 import requestsurl "https://storage.googleapis.com/benchmarks-artifacts/chinook/Chinook.db"response requests.get(url)if response.status_code 200:# Open a local file in binary write modewith open("Chinook.db", "wb&qu…

pip更换国内源,加速Python包下载(附2024年12月最新国内镜像源列表)

pip是什么 pip 是 Python 包管理工具&#xff0c;它允许用户从 Python 包索引&#xff08;PyPI&#xff09;安装和管理软件包。pip 是 Python 的官方包安装程序&#xff0c;它提供了一个命令行界面&#xff0c;用户可以通过它来安装、卸载、查看和管理 Python 包。以下是 pip …

安全关系型数据库查询新选择:Rust 语言的 rust-query 库深度解析

在当今这个数据驱动的时代&#xff0c;数据库作为信息存储和检索的核心组件&#xff0c;其重要性不言而喻。然而&#xff0c;对于开发者而言&#xff0c;如何在保证数据安全的前提下&#xff0c;高效地进行数据库操作却是一项挑战。传统的 SQL 查询虽然强大&#xff0c;但存在诸…

linux-10 关于shell(九)认证、授权、审计

之前提到过的一些基本应用&#xff0c;对Linux系统而言&#xff0c;安装完成以后&#xff0c;它给我们提供一个登录界面&#xff0c;对吧&#xff1f;这个登录界面说白了就是验证用户的&#xff0c;身份的&#xff0c;我昨天提到过&#xff0c;一般而言&#xff0c;每一个使用者…

VSCode中“Run Code”运行程序时,终端出现中文乱码解决方法

问题描述 在VSCode中“Run Code”运行程序时&#xff0c;终端输出结果出现中文乱码现象&#xff1a; 解决方法 1. 检查系统cmd的默认编码 查看Windows终端当前编码方式的命令&#xff1a; chcp输出结果是一段数字代码&#xff0c;如936&#xff0c;这说明当前的cmd编码方式…

【Python】ASCII-generator 将图像、文本或视频转换为 ASCII 艺术 生成字符图(测试代码)

目录 预览效果安装环境报错分析基本例程总结 欢迎关注 『Python』 系列&#xff0c;持续更新中 欢迎关注 『Python』 系列&#xff0c;持续更新中 预览效果 原图 黑白图 彩色图 安装环境 拉取代码 https://github.com/vietnh1009/ASCII-generatorpython3.8 pip install…