從 local-tool 複製出獨立的「visionA Agent」桌面應用(A3 純橋樑: tunnel client + 配對 UI + 設定,不開 HTTP port、不做本機裝置/推論 UI)。 Bundle ID 與 local-tool 不同(com.innovedus.visiona-agent vs visiona-local), 雙 app 可共存。fork 後不主動 sync,需要時手動 cherry-pick。 Backend / Wails Go(AB1-AB13): - internal/tunnel:6 狀態機(Idle/Connecting/Connected/Reconnecting/Failed/Stopped) + Pair/Unpair/Reconnect/Disconnect binding + ClientHooks event - internal/auth:encrypted file token store(AES-GCM + scrypt + machineID fallback salt + 13 tests) - internal/config:YAML validation + atomic write + 11 tests - internal/log:ring buffer + ExportLog 升級 zip - visionA-backend /api/pairing/exchange:SessionTokenStore + 17 new tests - 三平台 build 驗證(macOS DMG 160 MB / Windows EXE / Linux AppImage) - end-to-end 5 milestone 全綠(pairing → tunnel → forward → reuse 防護 → tunnel drop failover) Frontend / Next.js(AF1-AF7,沿用 visionA-frontend 基礎): - AppShell + Header + TabNav(StatusView / PairView / SettingsView 三 tab) - ConnectionStatusBadge 5 種狀態 - TokenInput regex 驗證 + 7 種錯誤 + 0.5s auto-switch 到狀態頁 - 設定頁 4 區塊(含重新配對 AlertDialog) - agent-api.ts 封裝 Wails bindings(mock/real 雙實作)+ 90 tests Phase 0.7 review-driven fix(Round 2): - A1 Session fixation 防護(RotateSessionID) - A3 mock pairing 預設改 false(必須明確 opt-in)+ startup log - A4 Pair 失敗後 state 清理矩陣(exchange/Save/Start fail 各自終態) - A5 Pair/Unpair/Reconnect lifecycleMu + 50 goroutine race test - F1 重新配對次按鈕 / F2 PairView Esc cancel / F3 Wails BrowserOpenURL / F4 Settings draft 持久 + 未儲存 badge 驗證:agent backend go test -race -count=3 ./... 4 packages 全綠 / agent frontend pnpm test 119 tests 全綠 Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
204 lines
7.9 KiB
Bash
Executable File
204 lines
7.9 KiB
Bash
Executable File
#!/usr/bin/env bash
|
||
# visionA Agent Linux AppImage builder
|
||
#
|
||
# 必須在 Linux x86_64 跑(GitHub Actions ubuntu-latest 即可)。
|
||
# 需要的工具:appimagetool、wails CLI、go 1.26+、pnpm、curl
|
||
#
|
||
# 前置:payload/linux/ 應已由 `make payload-linux` 產出,
|
||
# visiona-agent/build/bin/visiona-agent(Wails app binary)應已由 `make wails-linux` 產出。
|
||
#
|
||
# 用法:
|
||
# VERSION=0.1.0 bash installer/linux/build-appimage.sh
|
||
#
|
||
# 產物:dist/visiona-agent-${VERSION}-linux-x64.AppImage
|
||
|
||
set -euo pipefail
|
||
|
||
PROJECT_ROOT="$(cd "$(dirname "$0")/../.." && pwd)"
|
||
APPDIR="$PROJECT_ROOT/dist/visiona-agent.AppDir"
|
||
VERSION="${VERSION:-0.1.0}"
|
||
|
||
# 平台守門:只能在 Linux 跑
|
||
if [ "$(uname -s)" != "Linux" ]; then
|
||
echo "❌ build-appimage.sh 只能在 Linux 上執行(目前:$(uname -s))"
|
||
exit 1
|
||
fi
|
||
|
||
echo "==> 清理舊 AppDir"
|
||
rm -rf "$APPDIR"
|
||
mkdir -p "$APPDIR/usr/bin" \
|
||
"$APPDIR/usr/lib/visiona-agent/data" \
|
||
"$APPDIR/usr/lib/visiona-agent/scripts" \
|
||
"$APPDIR/usr/lib/visiona-agent/python" \
|
||
"$APPDIR/usr/lib/visiona-agent/wheels" \
|
||
"$APPDIR/usr/share/applications" \
|
||
"$APPDIR/usr/share/icons/hicolor/256x256/apps"
|
||
|
||
echo "==> 複製 Wails app binary"
|
||
WAILS_BIN="$PROJECT_ROOT/visiona-agent/build/bin/visiona-agent"
|
||
if [ ! -f "$WAILS_BIN" ]; then
|
||
echo "❌ 找不到 Wails binary:$WAILS_BIN"
|
||
echo " 請先執行:make wails-linux"
|
||
exit 1
|
||
fi
|
||
cp "$WAILS_BIN" "$APPDIR/usr/bin/visiona-agent"
|
||
chmod +x "$APPDIR/usr/bin/visiona-agent"
|
||
|
||
echo "==> 複製 server binary + 工具(從 payload/linux/)"
|
||
PAYLOAD_LINUX="$PROJECT_ROOT/payload/linux"
|
||
if [ ! -d "$PAYLOAD_LINUX" ]; then
|
||
echo "❌ 找不到 payload/linux/,請先執行:make payload-linux"
|
||
exit 1
|
||
fi
|
||
|
||
# server binary
|
||
if [ -f "$PAYLOAD_LINUX/bin/visiona-agent-server" ]; then
|
||
cp "$PAYLOAD_LINUX/bin/visiona-agent-server" "$APPDIR/usr/bin/visiona-agent-server"
|
||
else
|
||
echo "⚠️ payload/linux/bin/visiona-agent-server 不存在(需要在 Linux 上 go build server)"
|
||
fi
|
||
|
||
# ffmpeg / ffprobe
|
||
# ffmpeg 為 LGPL v3 build(BtbN n7.1,v2 TDD §4)
|
||
for tool in ffmpeg ffprobe; do
|
||
if [ -f "$PAYLOAD_LINUX/bin/$tool" ]; then
|
||
cp "$PAYLOAD_LINUX/bin/$tool" "$APPDIR/usr/bin/$tool"
|
||
else
|
||
echo "⚠️ payload/linux/bin/$tool 不存在"
|
||
fi
|
||
done
|
||
|
||
chmod +x "$APPDIR/usr/bin/"* 2>/dev/null || true
|
||
|
||
# 把 ffmpeg 授權條款放到 AppImage 的 share/doc/
|
||
mkdir -p "$APPDIR/usr/share/doc/visiona-agent"
|
||
if [ -f "$PAYLOAD_LINUX/bin/ffmpeg-LICENSE.txt" ]; then
|
||
cp "$PAYLOAD_LINUX/bin/ffmpeg-LICENSE.txt" "$APPDIR/usr/share/doc/visiona-agent/ffmpeg-LICENSE.txt"
|
||
fi
|
||
|
||
echo "==> 檢查 Python runtime + wheels(bundled Python 必備,缺就自動補)"
|
||
# Python tarball
|
||
if [ ! -f "$PAYLOAD_LINUX/python/python.tar.gz" ]; then
|
||
echo "⚠️ payload/linux/python/python.tar.gz 不存在,嘗試自動補(make vendor-python-linux + payload-linux)"
|
||
(cd "$PROJECT_ROOT" && make vendor-python-linux payload-linux)
|
||
fi
|
||
if [ ! -f "$PAYLOAD_LINUX/python/python.tar.gz" ]; then
|
||
echo "❌ python tarball 仍缺失:$PAYLOAD_LINUX/python/python.tar.gz"
|
||
echo " AppImage 必須內嵌 Python runtime(Linux 強制 bundled mode),無此檔無法 build。"
|
||
echo " 請手動執行:make vendor-python-linux"
|
||
exit 1
|
||
fi
|
||
|
||
# Wheels(至少要有 numpy / opencv / pyusb / kp 4 個關鍵 wheel,bridge scan 才能跑)
|
||
wheel_count=$(ls -1 "$PAYLOAD_LINUX/wheels/"*.whl 2>/dev/null | wc -l)
|
||
if [ "$wheel_count" -lt 4 ]; then
|
||
echo "⚠️ payload/linux/wheels 只找到 $wheel_count 個 wheel(預期至少 4),嘗試自動補(make vendor-wheels-linux + payload-linux)"
|
||
(cd "$PROJECT_ROOT" && make vendor-wheels-linux payload-linux)
|
||
wheel_count=$(ls -1 "$PAYLOAD_LINUX/wheels/"*.whl 2>/dev/null | wc -l)
|
||
fi
|
||
if [ "$wheel_count" -lt 4 ]; then
|
||
echo "❌ wheels 數量仍不足:$PAYLOAD_LINUX/wheels/ 只有 $wheel_count 個 .whl(預期至少 4:numpy / opencv / pyusb / kp)"
|
||
echo " 請檢查 vendor/wheels/linux/ 或 visiona-agent/wheels/linux/ 是否有 KneronPLUS wheel"
|
||
exit 1
|
||
fi
|
||
|
||
# 驗證關鍵 wheel 名稱存在(避免只下載到其他 wheel 卻缺 kp 等)
|
||
# 大小寫不敏感:KneronPLUS wheel 檔名是大寫開頭,pip 下載的是小寫
|
||
missing=()
|
||
for pkg in numpy opencv pyusb KneronPLUS; do
|
||
if ! ls "$PAYLOAD_LINUX/wheels/" 2>/dev/null | grep -iq "$pkg"; then
|
||
missing+=("$pkg")
|
||
fi
|
||
done
|
||
if [ ${#missing[@]} -gt 0 ]; then
|
||
echo "❌ 缺少關鍵 wheel:${missing[*]}"
|
||
echo " 目前 wheels 目錄內容:"
|
||
ls -1 "$PAYLOAD_LINUX/wheels/" 2>/dev/null | sed 's/^/ /'
|
||
echo " KneronPLUS wheel 需手動放到 visiona-agent/wheels/linux/(非公開套件)"
|
||
exit 1
|
||
fi
|
||
|
||
echo "==> ✅ python tarball + $wheel_count 個 wheel 齊備"
|
||
|
||
echo "==> 複製資料、腳本、Python runtime、wheels"
|
||
[ -d "$PAYLOAD_LINUX/data" ] && cp -R "$PAYLOAD_LINUX/data/." "$APPDIR/usr/lib/visiona-agent/data/" || true
|
||
[ -d "$PAYLOAD_LINUX/scripts" ] && cp -R "$PAYLOAD_LINUX/scripts/." "$APPDIR/usr/lib/visiona-agent/scripts/" || true
|
||
cp "$PAYLOAD_LINUX/python/python.tar.gz" "$APPDIR/usr/lib/visiona-agent/python/"
|
||
cp "$PAYLOAD_LINUX/wheels/"*.whl "$APPDIR/usr/lib/visiona-agent/wheels/"
|
||
|
||
# 複製 udev rule 到 lib 供 Wails app 首次啟動時使用
|
||
if [ -f "$PROJECT_ROOT/installer/linux/99-kneron.rules" ]; then
|
||
cp "$PROJECT_ROOT/installer/linux/99-kneron.rules" "$APPDIR/usr/lib/visiona-agent/"
|
||
fi
|
||
if [ -f "$PROJECT_ROOT/installer/linux/install-udev.sh" ]; then
|
||
cp "$PROJECT_ROOT/installer/linux/install-udev.sh" "$APPDIR/usr/lib/visiona-agent/"
|
||
chmod +x "$APPDIR/usr/lib/visiona-agent/install-udev.sh"
|
||
fi
|
||
|
||
echo "==> 寫入 .desktop 與 icon"
|
||
cat > "$APPDIR/visiona-agent.desktop" <<'DESKTOP'
|
||
[Desktop Entry]
|
||
Type=Application
|
||
Name=visionA Agent
|
||
Comment=Local-first Edge AI development tool for Kneron KL520/KL720
|
||
Exec=visiona-agent
|
||
Icon=visiona-agent
|
||
Categories=Development;
|
||
Terminal=false
|
||
StartupWMClass=visionA Agent
|
||
DESKTOP
|
||
cp "$APPDIR/visiona-agent.desktop" "$APPDIR/usr/share/applications/"
|
||
|
||
# Icon (256x256 PNG),從 visiona-agent/build/appicon.png 複製
|
||
APPICON="$PROJECT_ROOT/visiona-agent/build/appicon.png"
|
||
if [ -f "$APPICON" ]; then
|
||
cp "$APPICON" "$APPDIR/visiona-agent.png"
|
||
cp "$APPICON" "$APPDIR/usr/share/icons/hicolor/256x256/apps/visiona-agent.png"
|
||
else
|
||
echo "⚠️ 找不到 appicon.png,AppImage 將無 icon"
|
||
fi
|
||
|
||
echo "==> 寫入 AppRun"
|
||
cat > "$APPDIR/AppRun" <<'APPRUN'
|
||
#!/bin/bash
|
||
# AppRun — AppImage 啟動 entry point
|
||
HERE="$(dirname "$(readlink -f "${0}")")"
|
||
export PATH="${HERE}/usr/bin:${PATH}"
|
||
export LD_LIBRARY_PATH="${HERE}/usr/lib:${LD_LIBRARY_PATH:-}"
|
||
|
||
# visionA Agent 專屬環境變數:讓 Wails app 定位 bundle 內檔案
|
||
export VISIONA_BUNDLE_BIN_DIR="${HERE}/usr/bin"
|
||
export VISIONA_BUNDLE_LIB_DIR="${HERE}/usr/lib/visiona-agent"
|
||
|
||
# Linux AppImage 強制走 bundled Python(內嵌 python-build-standalone + wheels),
|
||
# 避免 system Python 缺 numpy / kp / pyusb 導致 bridge scan 失敗。
|
||
export VISIONA_PYTHON_MODE=bundled
|
||
|
||
exec "${HERE}/usr/bin/visiona-agent" "$@"
|
||
APPRUN
|
||
chmod +x "$APPDIR/AppRun"
|
||
|
||
echo "==> 檢查 appimagetool"
|
||
if ! command -v appimagetool &> /dev/null; then
|
||
echo "❌ appimagetool 未安裝。"
|
||
echo ""
|
||
echo " 一鍵安裝(推薦):"
|
||
echo " curl -fsSL https://github.com/AppImage/appimagetool/releases/download/continuous/appimagetool-x86_64.AppImage -o /tmp/appimagetool \\"
|
||
echo " && chmod +x /tmp/appimagetool \\"
|
||
echo " && sudo mv /tmp/appimagetool /usr/local/bin/appimagetool"
|
||
echo ""
|
||
echo " 或重跑 bootstrap-linux.sh(會自動安裝所有依賴):"
|
||
echo " bash scripts/bootstrap-linux.sh"
|
||
echo ""
|
||
exit 1
|
||
fi
|
||
|
||
echo "==> 用 appimagetool 打包"
|
||
mkdir -p "$PROJECT_ROOT/dist"
|
||
OUTPUT="$PROJECT_ROOT/dist/visiona-agent-${VERSION}-linux-x64.AppImage"
|
||
ARCH=x86_64 appimagetool "$APPDIR" "$OUTPUT"
|
||
|
||
echo ""
|
||
echo "==> 完成:$OUTPUT"
|
||
ls -lh "$OUTPUT"
|