# 桌面端打包说明 (Electron + React UI) 本项目可打包为桌面应用,使用 Electron 作为桌面壳,`apps/dsa-web` 的 React UI 作为界面。 ## 架构说明 - React UI(Vite 构建)由本地 FastAPI 服务托管 - Electron 启动时自动拉起后端服务,等待 `/api/health` 就绪后加载 UI - Windows 便携/安装模式下,用户配置文件 `.env` 和数据库放在 exe 同级目录;macOS 打包版使用 Electron 用户数据目录保存运行时配置 ## 本地开发 一键启动(开发模式): ```bash powershell -ExecutionPolicy Bypass -File scripts\run-desktop.ps1 ``` 或手动执行: 1) 构建 React UI(输出到 `static/`) ```bash cd apps/dsa-web npm install npm run build ``` 2) 启动 Electron 应用(自动拉起后端) ```bash cd apps/dsa-desktop npm install npm run dev ``` 首次运行时会自动从 `.env.example` 复制生成 `.env`。 ## 打包 (Windows) ### 前置条件 - Node.js 18+ - Python 3.10+ - 开启 Windows 开发者模式(electron-builder 需要创建符号链接) - 设置 -> 隐私和安全性 -> 开发者选项 -> 开发者模式 ### 一键打包 ```bash powershell -ExecutionPolicy Bypass -File scripts\build-all.ps1 ``` 该脚本会依次执行: 1. 构建 React UI 2. 安装 Python 依赖 3. PyInstaller 打包后端 4. electron-builder 打包桌面应用 当前 Windows 安装包使用 NSIS 向导式安装流程,仅支持当前用户安装且已禁用管理员提权,安装时可手动选择目标目录(例如非 C 盘)。安装器通过 NSIS `.onVerifyInstDir` 回调在安装器层面阻止选择 `Program Files`、`Windows` 等系统保护目录——选择这些路径时"下一步"按钮会被自动禁用。安装完成后,桌面端仍会按现有逻辑在安装目录旁生成/读取 `.env`、`data/stock_analysis.db`(含 `data/stock_analysis.db-wal` / `data/stock_analysis.db-shm`)和 `logs/desktop.log`。推荐使用默认的 per-user 安装目录。如果不想安装,仍可继续分发 `win-unpacked` 免安装包。 ## GitHub CI 自动打包并发布 Release 仓库已支持通过 GitHub Actions 自动构建桌面端并上传到 GitHub Releases: - 工作流:`.github/workflows/desktop-release.yml` - 触发方式: - 推送语义化 tag(如 `v3.2.12`)后自动触发 - 在 Actions 页面手动触发并指定 `release_tag` - 产物: - Windows 安装包:Release 附件和本地 `apps/dsa-desktop/dist/` 中统一为 `daily-stock-analysis-windows-installer-.exe` - Windows 自动更新元数据:Release 附件会额外保留 `latest.yml` 和 `*.blockmap`,供安装版桌面端后台下载与校验更新;普通用户无需手动下载这些元数据。下载完成后用户确认“重启安装”时,桌面端会先停止内置后端、备份运行时文件,并以静默模式执行安装器。 - Windows 免安装包:`daily-stock-analysis-windows-noinstall-.zip` - macOS Intel:`daily-stock-analysis-macos-x64-.dmg` - macOS Apple Silicon:`daily-stock-analysis-macos-arm64-.dmg` 建议发布流程: 1. 合并代码到 `main` 2. 由自动打 tag 工作流生成版本(或手动创建 tag) 3. `desktop-release` 工作流自动构建并把两个平台安装包附加到对应 GitHub Release ## 发版前可复现验证(桌面更新链路) 桌面端自动更新链路依赖 Windows NSIS 安装产物、`latest.yml` 与 `*.blockmap` 元数据。当前桌面 CI 不覆盖 `desktop-release` 打包产物可发布链路,提交前建议补充如下本地验证: 说明:该清单专注于 Windows NSIS 安装版与 `electron-updater` 发布元数据。当前 Linux 环境无法直接产出 Windows 安装包和 updater 元数据(`latest.yml` / `*.blockmap`),此类链路需在 Windows 发布执行器或 Windows 本机环境复核。 若在非 Windows 环境无法完成上述验证,请在 PR 验收说明中明确补齐 Windows 发布链路复核人、复核时间窗及 `desktop-release` 产物检查结果(release/tag 与 `daily-stock-analysis-windows-installer-.exe`、`latest.yml`、`*.blockmap` 版本一致性与可下载性)。 1. 先构建 Web 静态产物(桌面端主窗口与设置页入口依赖) ```bash cd apps/dsa-web npm ci npm run lint npm run build ``` 2. 回到桌面端,补齐依赖、运行 preload 单测、再执行 Electron 打包 ```bash cd ../dsa-desktop npm ci npm test npm run build ``` 在 Windows 发布复核环境,还可额外执行: ```powershell ./scripts/verify-desktop-updater-artifacts.ps1 -ReleaseTag v$(node -p "require('./apps/dsa-desktop/package.json').version") ``` > 预期当前执行环境不支持生成 Windows NSIS 安装器时,请在交付说明中明确注明平台限制,并要求指定的 Windows 发布链路复核人补齐该项验证。 3. 检查更新元数据是否产出 ```bash ls -1 dist | sort ls -1 dist/*.yml dist/*.blockmap 2>/dev/null || true ``` 4. 强制对齐版本与发布附件(可在 Windows 环境或能产出 NSIS 产物的执行器上复核) ```bash RELEASE_TAG="v$(node -p \"require('./package.json').version\")" REPO="ZhuLinsen/daily_stock_analysis" for f in dist/*latest.yml dist/*.blockmap dist/daily-stock-analysis-windows-installer-*.exe; do [ -f \"$f\" ] && echo \"[FOUND] $f\" done if [ -f dist/latest.yml ]; then echo \"---- latest.yml 版本片段 ----\" grep -E \"^version:|^files:|^sha512:\" dist/latest.yml fi echo \"---- Release 清单(人工核对)----\" echo \"Release Tag: $RELEASE_TAG\" echo \"Release 地址: https://github.com/$REPO/releases/tag/$RELEASE_TAG\" echo \"应核对附件是否包含:\" echo \"- daily-stock-analysis-windows-installer-*.exe\" echo \"- latest.yml\" echo \"- *.blockmap\" echo \"并确保 latest.yml 中 version 与 tag 的语义化版本一致,path/url 与安装包附件名一致\" ``` 5a. 建议在 PR 描述里记录的“可复核输出”(Windows): ```bash echo "release-tag=${RELEASE_TAG}" echo "latest.yml version:" grep -E "^version:" dist/latest.yml echo "latest.yml files:" sed -n '1,80p' dist/latest.yml echo "packaging artifacts:" ls -1 dist/*.yml dist/*.blockmap dist/*installer*.exe 2>/dev/null | sort ``` Windows 发布链路复核清单(在 PR 后由发布团队/维护者执行): - release/tag 与 `daily-stock-analysis-windows-installer-.exe` 的版本号一致; - `latest.yml`、`daily-stock-analysis-windows-installer-.exe`、`*.blockmap` 同 tag 同步出现且可下载; - `latest.yml` 中 `version` 与 Release tag 语义一致(去掉 `v` 前缀后比对),且 `path` / `files.url` 与安装包附件名一致; - 如缺少上述文件或 `release-tag` 不匹配,需标注阻断并补齐 `desktop-release` 打包流程。 5. Windows/NSIS 产物与发布附件一致性请在 Windows 环境手动验证(可人工触发发布流程),并在升级后核对运行时文件留存: 1. 安装前后分别记录安装目录中的 `.env`、`data/stock_analysis.db`、`data/stock_analysis.db-wal`、`data/stock_analysis.db-shm`、`logs/desktop.log` 的 SHA256; 2. 确认桌面端下一次启动后,上述文件仍存在且与安装前记录一致; 3. 如不一致,可在应用退出后检查用户数据目录中的 `.dsa-desktop-update-backup` 是否清理完整,并结合最新日志串联排查。 Windows 平台建议使用 PowerShell 执行: ```bash Get-FileHash .env,data\\stock_analysis.db,data\\stock_analysis.db-wal,data\\stock_analysis.db-shm,logs\\desktop.log -Algorithm SHA256 ``` 说明:应用已在 Windows NSIS 安装版的“重启安装”前停止内置后端、备份安装目录旁上述运行时文件,并以静默模式运行更新安装器,目的是避免安装向导抢先覆盖仍在运行的桌面端进程,同时降低更新过程中文件丢失风险;若恢复失败,桌面端会显示更新安装错误并保留手动下载路径供回退处理。此次修复仅改动 Windows 更新安装链路与内置后端进程生命周期处理,不涉及设置保存语义、模型运行时清理策略或配置迁移行为。 ### 分步打包 1) 构建 React UI ```bash cd apps/dsa-web npm install npm run build ``` 2) 打包 Python 后端 ```bash pip install pyinstaller pip install -r requirements.txt python -m PyInstaller --name stock_analysis --onefile --noconsole --add-data "static;static" --hidden-import=multipart --hidden-import=multipart.multipart main.py ``` 将生成的 exe 复制到 `dist/backend/`: ```bash mkdir dist\backend copy dist\stock_analysis.exe dist\backend\stock_analysis.exe ``` 3) 打包 Electron 桌面应用 ```bash cd apps/dsa-desktop npm install npm run build ``` 打包产物位于 `apps/dsa-desktop/dist/`。Windows 安装器会生成 `daily-stock-analysis-windows-installer-.exe`,安装向导中可选择安装目录。 ## 目录结构 Windows 安装包模式下,安装器仅支持当前用户安装且已禁用管理员提权,用户可在安装向导中选择安装目录;安装器会在安装器层面阻止选择 `Program Files`、`Windows` 等系统保护目录(选择时"下一步"按钮自动禁用),安装完成后,应用会在安装目录旁生成/读取 `.env`、`data/stock_analysis.db`(含 `data/stock_analysis.db-wal` / `data/stock_analysis.db-shm`)和 `logs/desktop.log`。请保留默认的 per-user 安装位置或选择其他用户可写目录。 `win-unpacked` 免安装模式下,目录结构如下: ``` win-unpacked/ Daily Stock Analysis.exe <- 双击启动 .env <- 用户配置文件(首次启动自动生成) data/ stock_analysis.db <- 数据库主文件 stock_analysis.db-wal <- WAL 日志文件(更新备份/恢复) stock_analysis.db-shm <- WAL 共享元文件(更新备份/恢复) logs/ desktop.log <- 运行日志 resources/ .env.example <- 配置模板 backend/ stock_analysis.exe <- 后端服务 ``` ## 配置文件说明 - Windows 桌面端的 `.env` 放在 exe 同目录下 - macOS 打包版的 `.env`、`data/` 和 `logs/` 放在 Electron 用户数据目录,避免替换 `.app` 时丢失 - 首次启动时自动从 `.env.example` 复制生成 - 从旧版本升级时,如果旧 `.app` 包内部的 `.env`、`data/stock_analysis.db` 或日志文件仍可访问,新版本会在目标文件不存在时自动迁移到用户数据目录;已有目标文件不会被覆盖 - 用户需要编辑 `.env` 配置以下内容: - `GEMINI_API_KEY` 或 `OPENAI_API_KEY`:AI 分析必需 - `STOCK_LIST`:自选股列表(逗号分隔) - 其他可选配置参考 `.env.example` ### 配置备份 / 恢复 `.env` - WebUI 与桌面端都可以从 `系统设置 -> 配置备份` 看到 `导出 .env` 和 `导入 .env` 按钮 - WebUI 非桌面运行时需要先开启管理员认证并完成登录;未开启认证时按钮会禁用,API 返回 `403` - `导出 .env` 会导出当前**已保存**的 `.env` 备份文件;页面上尚未点击“保存配置”的本地草稿不会被导出 - `导入 .env` 会读取备份文件中的键值并合并到当前配置中,导入后会立即触发配置重载 - 导入是“键级覆盖”而不是整文件替换:备份文件中出现的键会覆盖当前值,未出现的键保持不变 - 如果当前页面还有未保存草稿,导入前会先提示确认,避免把本地草稿和已保存配置混在一起 - Web 端默认 `ADMIN_AUTH_ENABLED=false` 时,设置页会展示按钮为禁用态并提示先启用管理员鉴权;桌面端不受该配置影响,仍可直接使用配置备份/恢复能力。 > 建议:从旧版本升级的 macOS 用户仍可在升级前执行一次 `导出 .env` 作为保险;如果旧 `.app` 已经被整体替换,包内旧文件无法凭空恢复,只能通过备份导入。 ### 设置页版本信息 - `系统设置 -> 版本信息` 中的“桌面端版本”由 Electron 主进程的 `app.getVersion()` 提供,并通过 preload bridge 暴露给前端 - 开发态 `npm run dev` 与打包态 `npm run build` / 安装包都会复用同一条版本注入链路,不再在 `preload.js` 里维护独立硬编码版本号 - `README.md` 继续保留安装和运行入口说明;这类桌面端运行时细节统一落在本专题文档维护,避免入门文档膨胀 ### 桌面端更新提醒 - 应用在主界面加载完成后会后台检查 GitHub Releases 的最新正式版,并与当前 `app.getVersion()` 做语义化版本比较 - Windows NSIS 安装版会通过内置 GitHub 更新源自动下载新版本;下载完成后弹出一次性提醒,用户确认后静默重启并安装 - `系统设置 -> 版本信息` 中的“桌面端更新”区域可手动检查更新;若更新已下载,会展示“重启安装”操作 - Windows 免安装包、开发态和 macOS DMG 仍保持“提醒 + 跳转下载页”的兼容路径,不会因为网络失败而阻断桌面端启动 - 版本检查失败、GitHub API 超时、更新元数据缺失或下载安装异常时,会记录到 `logs/desktop.log`,设置页手动检查时会展示错误状态 ## 常见问题 ### 启动后一直显示 "Preparing backend..." 1. 检查 `logs/desktop.log` 查看错误信息 2. 确认 `.env` 文件存在且配置正确 3. 确认端口 8000-8100 未被占用 ### 后端启动报 ModuleNotFoundError PyInstaller 打包时缺少模块,需要在 `scripts/build-backend.ps1` 中增加 `--hidden-import`。 ### UI 加载空白 确认 `static/index.html` 存在,如不存在需重新构建 React UI。 ### macOS 升级后配置迁移 旧版本曾把运行时 `.env`、数据库和日志写在 `.app` 包内部。新版本改为使用 Electron 用户数据目录,并在旧 `.app` 包内文件仍可访问时做一次性迁移。迁移规则是“目标不存在才复制”,避免覆盖用户已经在新版本中保存的配置。 如果旧 `.app` 已经被整体替换,旧包内 `.env` 无法由新版本自动恢复。此时可使用升级前导出的 `.env` 在 `系统设置 -> 配置备份` 中手动导入;完成一次迁移或重新配置后,后续版本会继续复用用户数据目录,不再随 `.app` 替换丢失。 ## 分发给用户 Windows 分发现在有两种方式: 1. 安装包:分发 `apps/dsa-desktop/dist/` 下的 `daily-stock-analysis-windows-installer-.exe`,用户安装时可自行选择目标目录 2. 免安装包:将 `apps/dsa-desktop/dist/win-unpacked/` 整个文件夹打包发给用户 使用 `win-unpacked` 免安装包时,用户只需: 1. 解压文件夹 2. 编辑 `.env` 配置 API Key 和股票列表 3. 双击 `Daily Stock Analysis.exe` 启动