從 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>
112 lines
3.1 KiB
Go
112 lines
3.1 KiB
Go
package deps
|
||
|
||
import (
|
||
"os"
|
||
"os/exec"
|
||
"path/filepath"
|
||
"strings"
|
||
)
|
||
|
||
// Dependency describes an external CLI tool the platform may use.
|
||
type Dependency struct {
|
||
Name string `json:"name"`
|
||
Available bool `json:"available"`
|
||
Version string `json:"version,omitempty"`
|
||
Path string `json:"path,omitempty"`
|
||
Required bool `json:"required"`
|
||
InstallHint string `json:"installHint,omitempty"`
|
||
}
|
||
|
||
// CheckAll probes all known external dependencies.
|
||
//
|
||
// 偵測順序(由 lookupBin 實作):
|
||
// 1. $VISIONA_BUNDLE_BIN_DIR/<name>(Wails shell 啟動 server 時設定)
|
||
// 2. exec.LookPath(<name>)(系統 PATH)
|
||
func CheckAll() []Dependency {
|
||
return []Dependency{
|
||
check("ffmpeg", false,
|
||
"macOS: brew install ffmpeg | Windows: winget install Gyan.FFmpeg",
|
||
"-version"),
|
||
check("ffprobe", false,
|
||
"macOS: brew install ffmpeg | Windows: winget install Gyan.FFmpeg",
|
||
"-version"),
|
||
check("python3", false,
|
||
"Required only for Kneron KL720 hardware. macOS: brew install python3",
|
||
"--version"),
|
||
}
|
||
}
|
||
|
||
// lookupBin 依序在 bundle bin dir 與 PATH 中找 binary。
|
||
//
|
||
// bundle bin dir 來自 env var VISIONA_BUNDLE_BIN_DIR(由 Wails shell 於啟動
|
||
// server 時注入,指向 .app/Contents/Resources/bin/ 或開發模式 payload/darwin/bin/)。
|
||
func lookupBin(name string) (string, bool) {
|
||
if dir := strings.TrimSpace(os.Getenv("VISIONA_BUNDLE_BIN_DIR")); dir != "" {
|
||
candidate := filepath.Join(dir, name)
|
||
if info, err := os.Stat(candidate); err == nil && !info.IsDir() {
|
||
return candidate, true
|
||
}
|
||
}
|
||
if p, err := exec.LookPath(name); err == nil {
|
||
return p, true
|
||
}
|
||
return "", false
|
||
}
|
||
|
||
func check(name string, required bool, hint string, args ...string) Dependency {
|
||
d := Dependency{
|
||
Name: name,
|
||
Required: required,
|
||
InstallHint: hint,
|
||
}
|
||
path, ok := lookupBin(name)
|
||
if !ok {
|
||
return d
|
||
}
|
||
d.Available = true
|
||
d.Path = path
|
||
|
||
// 效能:bundle 內的 binary 冷啟動可能較慢(尤其 PyInstaller),
|
||
// bundle binary 已知良好,跳過 version 查詢以加速啟動。
|
||
// 若之後需要版本字串,handler 可 lazy 再打一次。
|
||
if strings.HasPrefix(path, strings.TrimSpace(os.Getenv("VISIONA_BUNDLE_BIN_DIR"))) &&
|
||
os.Getenv("VISIONA_BUNDLE_BIN_DIR") != "" {
|
||
d.Version = "(bundled)"
|
||
return d
|
||
}
|
||
|
||
out, err := exec.Command(path, args...).Output()
|
||
if err == nil {
|
||
lines := strings.SplitN(string(out), "\n", 2)
|
||
if len(lines) > 0 {
|
||
d.Version = strings.TrimSpace(lines[0])
|
||
}
|
||
}
|
||
return d
|
||
}
|
||
|
||
// Logger is the minimal interface used for startup reporting.
|
||
type Logger interface {
|
||
Info(msg string, args ...interface{})
|
||
}
|
||
|
||
// PrintStartupReport logs the status of every external dependency.
|
||
func PrintStartupReport(logger Logger) {
|
||
deps := CheckAll()
|
||
logger.Info("External dependency check:")
|
||
if dir := os.Getenv("VISIONA_BUNDLE_BIN_DIR"); dir != "" {
|
||
logger.Info(" (bundle bin dir: %s)", dir)
|
||
}
|
||
for _, d := range deps {
|
||
if d.Available {
|
||
logger.Info(" [OK] %s: %s (%s)", d.Name, d.Version, d.Path)
|
||
} else {
|
||
tag := "OPTIONAL"
|
||
if d.Required {
|
||
tag = "MISSING"
|
||
}
|
||
logger.Info(" [%s] %s: not found — %s", tag, d.Name, d.InstallHint)
|
||
}
|
||
}
|
||
}
|