Files
spark-store/docs/superpowers/specs/2026-04-15-update-ignore-design.md

5.7 KiB

更新忽略配置迁移设计

背景

Electron 更新中心已经具备忽略状态的数据通路,但默认仍写入 /etc/spark-store/ignored_apps.conf。老 Qt 更新器也沿用同一路径。新架构下更新器不再以 root 身份启动,因此 GUI 无法稳定写入 /etc。与此同时,ss-update-notifier.sh 以 root systemd 服务运行,若直接使用 ~/ 会错误落到 /root

本次改动的目标是在不重做更新链路的前提下,把“忽略更新”统一改为用户级配置,并让 Electron、老 Qt 更新器和 notifier 对同一份规则生效。

目标

  1. 忽略配置统一迁移到用户目录 ~/.config/spark-store/ignored_apps.conf
  2. Electron 更新中心支持显式忽略和取消忽略操作。
  3. 老 Qt 更新器改为读写同一份用户级忽略配置。
  4. ss-update-notifier.sh 在 root systemd 环境下也能读取用户级忽略配置。
  5. 忽略规则同时作用于 Spark 与 APM 更新项。
  6. 忽略规则按 pkgname|version 精确匹配,被忽略的旧版本在后续出现新版本时应重新提醒。

非目标

  1. 不兼容旧的 /etc/spark-store/ignored_apps.conf
  2. 不改变更新下载、安装和迁移逻辑。
  3. 不把忽略配置升级为 JSON 或数据库格式。
  4. 不修改 AmberPM 侧的 amber-pm-upgrade-notifier

方案概览

本次实现由三部分组成:

  1. Electron 主进程把忽略配置路径切到用户目录,渲染层补齐“忽略 / 取消忽略”入口,并把已忽略项排在后面展示。
  2. 老 Qt 更新器的 IgnoreConfig 改为使用 QStandardPaths::ConfigLocation 下的 spark-store/ignored_apps.conf,同时将忽略键统一为“包名 + 新版本”。
  3. ss-update-notifier.sh 新增用户配置定位与扫描逻辑,在 root systemd 环境下优先识别活动桌面用户,失败时回退扫描 /home/*/.config/spark-store/ignored_apps.conf 并合并忽略集合。

配置文件设计

路径

  • 统一路径:~/.config/spark-store/ignored_apps.conf
  • Electron 通过当前进程用户的 home 解析该路径。
  • Qt 通过 QStandardPaths::writableLocation(QStandardPaths::ConfigLocation) 解析该路径。
  • notifier 不直接依赖 ~/,而是根据目标 home 拼出 <home>/.config/spark-store/ignored_apps.conf

格式

继续沿用现有纯文本格式,每行一条:

pkgname|version

其中 version 统一表示“待更新到的新版本”,而不是当前已安装版本。

匹配语义

  1. 仅当 pkgnameversion 同时匹配时,视为被忽略。
  2. 忽略规则不区分 spark / apm 来源。相同包名与目标版本的更新,在两侧都应被同一条规则命中。
  3. 某版本被忽略后,未来出现更高版本时,不自动继承忽略状态。

Electron 更新中心

主进程

electron/main/backend/update-center/ignore-config.ts 保持文本解析逻辑不变,只修改默认配置路径到用户目录。

electron/main/backend/update-center/service.ts 的默认读写也改用新路径,并在刷新结果上做一次稳定排序:

  1. 正常更新项在前。
  2. 已忽略项在后。
  3. 同组内保持原有顺序,避免不必要的 UI 抖动。

渲染层交互

更新中心列表项新增两个互斥操作:

  1. 未忽略项显示“忽略”按钮。
  2. 已忽略项显示“取消忽略”按钮。

交互规则:

  1. 点击“忽略”后调用 window.updateCenter.ignore({ packageName, newVersion })
  2. 点击“取消忽略”后调用 window.updateCenter.unignore({ packageName, newVersion })
  3. 主进程刷新完成后,渲染层使用推送或返回的新快照更新列表。
  4. 已忽略项继续不可勾选,也不会加入“更新选中”任务。

老 Qt 更新器

配置路径

IgnoreConfig 不再尝试写 /etc,改为:

  1. 使用 QStandardPaths::writableLocation(QStandardPaths::ConfigLocation)
  2. 在其下创建 spark-store/ignored_apps.conf

忽略键统一

Qt 当前交互里,忽略按钮传的是当前版本,检查时也匹配当前版本。这会导致与 Electron 的“目标版本忽略”语义不一致。

本次统一改为:

  1. 点击“忽略”时写入 packageName + newVersion
  2. 刷新列表时,用 packageName + newVersion 判断是否忽略。

取消忽略也改为按包名 + 版本删除对应条目,避免误删同包历史忽略记录。

ss-update-notifier.sh

读取忽略配置

脚本新增两个步骤:

  1. 尝试定位最可能的桌面用户 home。
  2. 如果无法可靠定位,则扫描 /home/*/.config/spark-store/ignored_apps.conf

扫描模式下需要把所有命中的配置文件合并成一个忽略集合,再参与过滤。

过滤规则

脚本当前只按包名过滤,本次改为按 pkgname|newVersion 精确过滤:

  1. ss-do-upgrade-worker.sh upgradable-list 读取 PKG_NAME PKG_NEW_VER PKG_CUR_VER
  2. 构造键 PKG_NAME|PKG_NEW_VER
  3. 若忽略集合中存在该键,则跳过通知计数。

与通知用户识别解耦

通知发送仍然尽量复用现有“找活动用户然后 sudo -u 发送”的策略,但“读取忽略配置”与“给谁发通知”必须解耦:

  1. 即使没有可靠的当前登录用户,也应先完成忽略过滤。
  2. 只有在最终需要发送通知时,再尝试解析实际桌面用户。

验证范围

  1. Electron 单元测试覆盖新路径常量、忽略排序与忽略按钮交互。
  2. Electron 手动验证更新中心忽略 / 取消忽略流程。
  3. Qt 手动验证忽略后重新打开更新器仍保留状态。
  4. 手动执行 ss-update-notifier.sh,验证 root 环境下能命中用户级忽略配置且按版本精确过滤。