使用 Qt 打造高效的 .run 软件包管理器
在软件开发领域,.run 软件包因其便携性和自解压特性而备受青睐,特别是由 makeself 工具生成的 .run 软件包。这些软件包通常包含一个完整的程序或库,以及一个用于解压和安装的脚本。然而,手动管理这些软件包(尤其是进行安装、卸载和版本切换)可能会变得繁琐且容易出错。因此,开发一个高效的 .run 软件包管理器显得尤为重要。Qt 框架以其跨平台特性和丰富的功能集,成为开发桌面应用程序的理想选择。本文将详细介绍如何通过定义 UpdateInstaller 类,利用Qt 实现.run 软件包的安装、卸载及版本切换功能。
一、基础知识回顾
1. Qt 框架简介
Qt 是一个功能强大、跨平台的应用程序开发框架,广泛应用于GUI程序开发,同时也支持非GUI程序开发,如命令行工具和服务器。Qt的信号与槽机制是其独特的通信方式,允许对象之间进行松耦合通信。QProcess 类则是Qt提供的一个用于启动外部程序和与之交互的类,它可以捕获程序的输出和错误信息,并提供控制程序执行的方法。QProcess 非常适合用于执行系统命令、脚本或运行其他独立程序。
2. .run软件包概述
makeself 是一个简单的工具,用于创建自解压的 .run 软件包。这些软件包通常包含一个 bash 脚本,用于解压和安装软件包内容。.run 软件包的结构通常很简单,但功能强大,因为它们可以包含任何类型的文件,并且安装过程完全由用户定义的脚本控制。
常见的 .run 软件包操作包括安装(通过运行 .run 脚本并遵循其指示)和卸载(通常需要手动删除安装的文件或执行一个卸载脚本)。
参考:使用Makeself打造自解压的.run安装程序
二、UpdateInstaller 类设计
类概述
UpdateInstaller 类的设计目标是封装 QProcess 的功能,提供一个用户友好的接口来管理 .run 软件包,包括安装、卸载及版本切换功能。
UpdateInstaller 类是负责软件更新流程的核心类。它协调各个组件来执行下载、校验、备份、安装和错误处理等任务。这个类封装了更新过程的复杂性,使得外部调用者只需调用一个方法即可启动整个更新流程。
成员变量与属性
- QProcess process_ :用于执行外部命令和脚本的进程对象。
- InstallState state_:表示当前安装状态的枚举值。
- QString currentPackageName_:当前正在安装或卸载的软件包名称。
- QString currentPackageVersion_(可选):当前正在安装或管理的软件包版本(如果适用)。
- bool isSilentMode_(可选):是否以静默模式运行,不输出详细日志。
成员方法
-
install(const QString &packagePath, const QString &packageName, const QString &packageVersion = “”)
- 参数:
packagePath(.run软件包路径),packageName(软件包名称),packageVersion(可选,软件包版本)。 - 功能:执行 .run 软件包的安装过程。构建并执行包含安装命令的字符串,同时更新安装状态。
- 参数:
-
uninstall(const QString &packageName)
- 参数:
packageName(要卸载的软件包名称)。 - 功能:执行软件包的卸载过程。构建并执行包含卸载命令的字符串。
- 参数:
-
switchVersion(const QString &packageName, const QString &newVersion, const QString &passwd = “”)
- 参数:
packageName(软件包名称),newVersion(要切换到的版本),passwd(可选,用于sudo命令的密码)。 - 功能:执行软件包的版本切换过程。注意,出于安全考虑,不建议在代码中硬编码密码,应通过安全方式获取用户输入。
- 参数:
-
checkSoftwareFormat(const QString &packagePath)
- 参数:
packagePath( .run 软件包路径)。 - 功能:验证.run软件包的有效性。可以读取文件头或特定标记来确认文件格式。
- 参数:
-
myReadyReadStandardOutput() 和 myReadyReadStandardError()
- 功能:这两个槽函数分别处理 QProcess 的标准输出和错误输出。它们读取输出内容,并通过信号传递给外部调用者。
-
processFinished(int exitCode, QProcess::ExitStatus exitStatus)
- 功能:处理 QProcess 结束时的状态。根据退出码和退出状态更新安装状态,并通过信号通知外部调用者。
信号
- messageOutput(const QString &msg):输出日志消息。
- progressChanged(int percent)(可选):更新安装进度(如果可能的话)。
- installStateChanged(int state):安装状态改变时发出。
错误处理与日志记录
- 错误处理:在方法实现中,应捕获并处理可能的异常或错误状态,例如文件不存在、命令执行失败等。通过发出错误消息或更改安装状态来通知外部调用者。
- 日志记录:可以扩展日志记录功能,以便更好地追踪和调试安装、卸载和版本切换过程中的问题。可以使用 Qt 的日志框架(如
qDebug()、qWarning()等)或自定义日志系统。
三、实现 UpdateInstaller 类
1. 环境准备
在开始实现 UpdateInstaller 类之前,请确保 Qt 开发环境已经搭建完毕,并且有一个示例 .run 软件包用于测试。Qt 的官方文档提供了详细的安装指南,而.run软件包可以通过任何提供此类软件包的网站或资源获取。
2. 代码实现
以下是 UpdateInstaller 类的实现示例,包括处理 QProcess 的输出和错误、更新安装状态以及实现一些基本的验证和错误处理机制。
#include <QCoreApplication>
#include <QProcess>
#include <QObject>
#include <QString>
#include <QByteArray>
#include <QDebug>
#include <QFile>
#include <QFileInfo>class UpdateInstaller : public QObject {Q_OBJECTpublic:enum InstallState {InstallState_Verifying = 0,InstallState_Uncompressing,InstallState_Configuring,InstallState_Completed,InstallState_Failed};UpdateInstaller(QObject *parent = nullptr): QObject(parent), state_(InstallState_Verifying), process_(new QProcess(this)) {connect(process_, &QProcess::readyReadStandardOutput, this, &UpdateInstaller::myReadyReadStandardOutput);connect(process_, &QProcess::readyReadStandardError, this, &UpdateInstaller::myReadyReadStandardError);connect(process_, &QProcess::finished, this, &UpdateInstaller::processFinished);}~UpdateInstaller() {}void install(const QString &path, const QString &name) {if (!QFile::exists(path)) {emit messageOutput("File does not exist.");return;}QString command = QString("sudo sh %1 -target /opt/app/%2").arg(path).arg(name);process_->start(command);emit installStateChanged(InstallState_Verifying);}static bool uninstall(const QString &name) {QString command = QString("sudo rm -rf /opt/app/%1").arg(name);int result = system(command.toStdString().c_str());return result == 0;}static void switchVersion(const QString &ver, const QString &passwd = "") {// 出于安全考虑,不建议在代码中硬编码密码// 这里仅作为示例,实际使用中应通过安全方式获取密码QString command = QString("echo %1 | sudo -S sh /opt/app/some_package/setup.sh install %2").arg(passwd).arg(ver);QProcess::startDetached(command); // 使用 startDetached 以避免阻塞主线程}static bool checkSoftwareFormat(const QString &path) {QFile file(path);if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) {return false;}QByteArray header = file.read(100); // 读取前100个字节作为示例return header.contains("makeself"); // 假设 makeself 是有效的包标识}signals:void messageOutput(const QString &msg);void progressChanged(int percent);void installStateChanged(int state);public slots:void myReadyReadStandardOutput() {QByteArray output = process_->readAllStandardOutput();emit messageOutput(QString::fromUtf8(output));}void myReadyReadStandardError() {QByteArray error = process_->readAllStandardError();emit messageOutput(QString::fromUtf8(error));}void processFinished(int exitCode, QProcess::ExitStatus exitStatus) {if (exitStatus == QProcess::NormalExit && exitCode == 0) {emit installStateChanged(InstallState_Completed);emit messageOutput("Installation completed successfully.");} else {emit installStateChanged(InstallState_Failed);emit messageOutput("Installation failed with exit code " + QString::number(exitCode));}}private:QProcess *process_;InstallState state_;
};int main(int argc, char *argv[]) {QCoreApplication a(argc, argv);UpdateInstaller installer;QObject::connect(&installer, &UpdateInstaller::messageOutput, [](const QString &msg){qDebug() << "Message Output:" << msg;});QObject::connect(&installer, &UpdateInstaller::progressChanged, [](int percent){qDebug() << "Progress Changed:" << percent << "%";});QObject::connect(&installer, &UpdateInstaller::installStateChanged, [](int state){switch (state) {case UpdateInstaller::InstallState_Verifying:qDebug() << "Verifying package...";break;case UpdateInstaller::InstallState_Uncompressing:qDebug() << "Uncompressing package...";break;case UpdateInstaller::InstallState_Configuring:qDebug() << "Configuring package...";break;case UpdateInstaller::InstallState_Completed:qDebug() << "Installation completed.";break;case UpdateInstaller::InstallState_Failed:qDebug() << "Installation failed.";break;default:qDebug() << "Unknown install state:" << state;}});QString packagePath = "/path/to/package.run";QString packageName = "example_package";if (UpdateInstaller::checkSoftwareFormat(packagePath)) {installer.install(packagePath, packageName);} else {qDebug() << "Invalid software format.";}return a.exec();
}
注意事项:
- 错误处理:在实际应用中,错误处理应更加详细和健壮,比如处理不同的错误码、提供用户友好的错误信息等。
- 安全性:确保执行的脚本和路径是可信的,避免潜在的安全风险。特别是在处理密码时,应避免在代码中硬编码密码,而应通过安全方式获取用户输入。
- 日志记录:可以扩展日志记录功能,以便更好地追踪和调试安装、卸载和版本切换过程中的问题。
- 跨平台支持:当前示例针对类Unix系统(如 Linux 和 macOS),对于Windows系统,可能需要不同的实现逻辑。特别是路径和命令的处理方式可能会有所不同。
- 进度更新:在实际应用中,进度更新需要根据具体的安装步骤进行解析和更新。可能需要从被执行的脚本中获取进度信息,并通过信号和槽机制将其传递给用户界面。
UpdateInstaller类设计细节优化与完善
相关文章:
使用 Qt 打造高效的 .run 软件包管理器
在软件开发领域,.run 软件包因其便携性和自解压特性而备受青睐,特别是由 makeself 工具生成的 .run 软件包。这些软件包通常包含一个完整的程序或库,以及一个用于解压和安装的脚本。然而,手动管理这些软件包(尤其是进行…...
python学opencv|读取视频(二)制作gif
【1】引言 前述已经完成了图像和视频的读取学习,本次课学习制作gif格式动图。 【2】教程 实际上想制作gif格式动图是一个顺理成章的操作,完成了图像和视频的处理,那就自然而然会对gif的处理也产生兴趣。 不过在opencv官网、matplotlib官网…...
19. Three.js案例-创建一个带有纹理映射的旋转平面
19. Three.js案例-创建一个带有纹理映射的旋转平面 实现效果 知识点 WebGLRenderer (WebGL渲染器) WebGLRenderer 是 Three.js 中用于渲染场景的主要类。它利用 WebGL 技术在浏览器中绘制 3D 图形。 构造器 new THREE.WebGLRenderer(parameters)参数类型描述parametersobj…...
ASP.NET|日常开发中常用属性详解
JAVA |日常开发中常用属性详解 前言一、控件属性(以 TextBox 控件为例)1.1 Text 属性:1.2 MaxLength 属性:1.3 ReadOnly 属性:1.4 IsPostBack 属性(在ASP.NET Web Forms 中)…...
vscode CMakeLists中对opencv eigen的引用方法
CMakeLists.txt 项目模式(只有一个main函数入口) cmake_minimum_required(VERSION 3.5)project(vsin01 VERSION 0.1 LANGUAGES CXX)set(CMAKE_CXX_STANDARD 17) set(CMAKE_CXX_STANDARD_REQUIRED ON)set(OpenCV_DIR G:/MinGW_Opencv/opencv4.10/opencv…...
使用Goland对6.5840项目进行go build出现异常
使用Goland对6.5840项目进行go build出现异常 Lab地址: https://pdos.csail.mit.edu/6.824/labs/lab-mr.html项目地址: git://g.csail.mit.edu/6.5840-golabs-2024 6.5840运行环境: mac系统 goland git clone git://g.csail.mit.edu/6.5840-golabs-2024 6.5840 cd 6.5840/src…...
Plugin - 插件开发06_开源项目JPom中的插件实现机制
文章目录 Pre工程结构概述1. 插件接口与实现分析2. 插件工厂初始化分析3. 插件项包装类解析4. 插件工厂方法解析5. 插件加载与资源释放机制6. 实现类小结附PluginFactory Pre 插件 - 通过SPI方式实现插件管理 插件 - 一份配置,离插件机制只有一步之遥 插件 - 插件…...
关于成功插入 SQLite 但没有数据的问题
背景 技术栈:SpringBoot Mybatis-flex SQLite 项目中集成了SQLite,配置如下: spring:datasource:url: jdbc:sqlite::resource:db/project.dbdriver-class-name: org.sqlite.JDBC在进行测试时,使用Mybatis-flex往表中插入数据&…...
单片机+Qt上位机
目录 一、引言 通信方式 优势 案例 常见问题及解决方法 二、单片机与 Qt 上位机的通信方式 (一)使用 QT 上位机和 STC 单片机实现串口通信 三、单片机 Qt 上位机的优势 (一)高效便捷的 USB 通信上位机解决方案 …...
C++ 类和对象(中)
1.类的六个默认成员函数 如果一个类中什么成员都没有,简称为空类。 空类中真的什么都没有吗?其实并不是,任何类在什么都不写时,编译器会自动生成以下六个默认成员函数。 默认成员函数:用户没有显式实现,编…...
在做题中学习(79):最小K个数
解法:快速选择算法 说明:堆排序也是经典解决问题的算法,但时间复杂度为:O(NlogK),K为k个元素 而将要介绍的快速选择算法的时间复杂度为: O(N) 先看我的前两篇文章,分别学习:数组分三块&#…...
spark3 sql优化:同一个表关联多次,优化方案
目录 1.合并查询2.使用 JOIN 条件的过滤优化3.使用 Map-side Join 或 Broadcast Join4.使用 Partitioning 和 Bucketing5.利用 DataFrame API 进行优化假设 A 和 B 已经加载为 DataFramePerform left joins with specific conditions6.使用缓存或持久化7.避免笛卡尔积总结 1.合…...
JavaWeb学习(4)(四大域、HttpSession原理(面试)、SessionAPI、Session实现验证码功能)
目录 一、web四大域。 (1)基本介绍。 (2)RequestScope。(请求域) (3)SessionScope。(会话域) (4)ApplicationScope。(应用域) (5)PageScope。(页面域) 二、Ht…...
Ubuntu22.04系统源码编译OpenCV 4.10.0(包含opencv_contrib)
因项目需要使用不同版本的OpenCV,而本地的Ubuntu22.04系统装了ROS2自带OpenCV 4.5.4的版本,于是编译一个OpenCV 4.10.0(带opencv_contrib)版本,给特定的项目使用,这就不用换个设备后重新安装OpenCV 了&…...
【Unity高级】在编辑器中如何让物体围绕一个点旋转固定角度
本文介绍如何在编辑器里让物体围绕一个点旋转固定角度,比如上图里的Cube是围绕白色圆盘的中心旋转45度的。 目标: 创建一个在 Unity 编辑器中使用的旋转工具,使开发者能够在编辑模式下快速旋转一个物体。 实现思路: 编辑模式下…...
2024.11.29——[HCTF 2018]WarmUp 1
拿到题,发现是一张图,查看源代码发现了被注释掉的提示 <!-- source.php--> step 1 在url传参看看这个文件,发现了这道题的源码 step 2 开始审计代码,分析关键函数 //mb_strpos($haystack,$needle,$offset,$encoding):int|…...
AGameModeBase和游戏模式方法
AGameModeBase和游戏模式方法有着密切的关系: AGameModeBase是游戏模式的基础类: 它提供了控制游戏规则的基本框架包含了一系列管理游戏流程的核心方法是所有自定义游戏模式类的父类 主要的游戏模式方法包括: // 游戏初始化时调用 virtua…...
Swift 扩展
Swift 扩展 Swift 是一种强大的编程语言,由苹果公司开发,用于iOS、macOS、watchOS和tvOS应用程序的开发。自2014年发布以来,Swift因其易于阅读和编写的语法、现代化的设计以及出色的性能而广受欢迎。本文将探讨Swift的一些关键特性ÿ…...
【NebulaGraph】官方查询语言nGQL教程1 (四)
【NebulaGraph】官方查询语言nGQL教程1 1. 课程信息2. 查找路径FIND PATH2.1 补充说明FIND PATH2.2 例子 1. 课程信息 课程地址: https://www.bilibili.com/video/BV1PT411P7w8/?spm_id_from333.337.search-card.all.click&vd_source240d9002f7c7e3da63cd9a975639409a …...
阿里云负载均衡SLB实践
基于上篇文章继续,如果你使用的是阿里云等云平台,通过配置nginxkeepAlived行不通,因为阿里云服务器不支持你虚拟出ip提供给外部访问,需要使用阿里云的负载均衡产品 对应的产品有三个系列 1、应用场景 ALB: 主要是对应应用层的7层…...
19c补丁后oracle属主变化,导致不能识别磁盘组
补丁后服务器重启,数据库再次无法启动 ORA01017: invalid username/password; logon denied Oracle 19c 在打上 19.23 或以上补丁版本后,存在与用户组权限相关的问题。具体表现为,Oracle 实例的运行用户(oracle)和集…...
从WWDC看苹果产品发展的规律
WWDC 是苹果公司一年一度面向全球开发者的盛会,其主题演讲展现了苹果在产品设计、技术路线、用户体验和生态系统构建上的核心理念与演进脉络。我们借助 ChatGPT Deep Research 工具,对过去十年 WWDC 主题演讲内容进行了系统化分析,形成了这份…...
Debian系统简介
目录 Debian系统介绍 Debian版本介绍 Debian软件源介绍 软件包管理工具dpkg dpkg核心指令详解 安装软件包 卸载软件包 查询软件包状态 验证软件包完整性 手动处理依赖关系 dpkg vs apt Debian系统介绍 Debian 和 Ubuntu 都是基于 Debian内核 的 Linux 发行版ÿ…...
为什么需要建设工程项目管理?工程项目管理有哪些亮点功能?
在建筑行业,项目管理的重要性不言而喻。随着工程规模的扩大、技术复杂度的提升,传统的管理模式已经难以满足现代工程的需求。过去,许多企业依赖手工记录、口头沟通和分散的信息管理,导致效率低下、成本失控、风险频发。例如&#…...
微服务商城-商品微服务
数据表 CREATE TABLE product (id bigint(20) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT 商品id,cateid smallint(6) UNSIGNED NOT NULL DEFAULT 0 COMMENT 类别Id,name varchar(100) NOT NULL DEFAULT COMMENT 商品名称,subtitle varchar(200) NOT NULL DEFAULT COMMENT 商…...
什么是EULA和DPA
文章目录 EULA(End User License Agreement)DPA(Data Protection Agreement)一、定义与背景二、核心内容三、法律效力与责任四、实际应用与意义 EULA(End User License Agreement) 定义: EULA即…...
Web中间件--tomcat学习
Web中间件–tomcat Java虚拟机详解 什么是JAVA虚拟机 Java虚拟机是一个抽象的计算机,它可以执行Java字节码。Java虚拟机是Java平台的一部分,Java平台由Java语言、Java API和Java虚拟机组成。Java虚拟机的主要作用是将Java字节码转换为机器代码&#x…...
android RelativeLayout布局
<?xml version"1.0" encoding"utf-8"?> <RelativeLayout xmlns:android"http://schemas.android.com/apk/res/android"android:layout_width"match_parent"android:layout_height"match_parent"android:gravity&…...
Cilium动手实验室: 精通之旅---13.Cilium LoadBalancer IPAM and L2 Service Announcement
Cilium动手实验室: 精通之旅---13.Cilium LoadBalancer IPAM and L2 Service Announcement 1. LAB环境2. L2公告策略2.1 部署Death Star2.2 访问服务2.3 部署L2公告策略2.4 服务宣告 3. 可视化 ARP 流量3.1 部署新服务3.2 准备可视化3.3 再次请求 4. 自动IPAM4.1 IPAM Pool4.2 …...
yaml读取写入常见错误 (‘cannot represent an object‘, 117)
错误一:yaml.representer.RepresenterError: (‘cannot represent an object’, 117) 出现这个问题一直没找到原因,后面把yaml.safe_dump直接替换成yaml.dump,确实能保存,但出现乱码: 放弃yaml.dump,又切…...
