從 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>
84 lines
2.2 KiB
Go
84 lines
2.2 KiB
Go
package ws
|
||
|
||
// system_ws_integration_test.go — MAJ-4 補丁:/ws/system 整合 smoke test
|
||
//
|
||
// 啟一個 httptest server 掛 SystemEventsHandler,真的用 gorilla WebSocket client
|
||
// 連進去,然後呼叫 hub.BroadcastToRoom("system", ...),驗證 client 收到訊息。
|
||
//
|
||
// 這個測試取代外部 websocat / wscat 的需求,讓 smoke test 在 CI 就能跑。
|
||
|
||
import (
|
||
"encoding/json"
|
||
"net/http"
|
||
"net/http/httptest"
|
||
"strings"
|
||
"testing"
|
||
"time"
|
||
|
||
"github.com/gin-gonic/gin"
|
||
"github.com/gorilla/websocket"
|
||
)
|
||
|
||
func TestSystemEventsHandler_ReceivesBroadcast(t *testing.T) {
|
||
gin.SetMode(gin.TestMode)
|
||
|
||
hub := NewHub()
|
||
go hub.Run()
|
||
|
||
r := gin.New()
|
||
r.GET("/ws/system", SystemEventsHandler(hub))
|
||
|
||
srv := httptest.NewServer(r)
|
||
defer srv.Close()
|
||
|
||
wsURL := "ws" + strings.TrimPrefix(srv.URL, "http") + "/ws/system"
|
||
dialer := websocket.DefaultDialer
|
||
conn, _, err := dialer.Dial(wsURL, http.Header{})
|
||
if err != nil {
|
||
t.Fatalf("dial: %v", err)
|
||
}
|
||
defer conn.Close()
|
||
|
||
// 給 Hub.register channel 一點時間處理 client 加入 room
|
||
// (RegisterSync 會同步等完,但我們這邊是透過 HTTP upgrade 流程,
|
||
// client 先連上後才 RegisterSync — 需要等 handler 執行到那一行)
|
||
// 改用 poll:持續廣播直到收到或 timeout。
|
||
_ = conn.SetReadDeadline(time.Now().Add(2 * time.Second))
|
||
|
||
// 等 hub 吸收 Register(最多 500 ms)
|
||
deadline := time.Now().Add(500 * time.Millisecond)
|
||
for time.Now().Before(deadline) {
|
||
hub.mu.RLock()
|
||
n := len(hub.rooms["system"])
|
||
hub.mu.RUnlock()
|
||
if n > 0 {
|
||
break
|
||
}
|
||
time.Sleep(10 * time.Millisecond)
|
||
}
|
||
|
||
// 廣播 shutdown-imminent
|
||
hub.BroadcastToRoom("system", map[string]interface{}{
|
||
"type": "server:shutdown-imminent",
|
||
"reason": "quit",
|
||
"ts": time.Now().UnixMilli(),
|
||
})
|
||
|
||
// Read 第一則訊息
|
||
_, data, err := conn.ReadMessage()
|
||
if err != nil {
|
||
t.Fatalf("read: %v", err)
|
||
}
|
||
|
||
var got map[string]interface{}
|
||
if err := json.Unmarshal(data, &got); err != nil {
|
||
t.Fatalf("json: %v; raw=%s", err, string(data))
|
||
}
|
||
if got["type"] != "server:shutdown-imminent" {
|
||
t.Errorf("wrong type: %+v", got)
|
||
}
|
||
if got["reason"] != "quit" {
|
||
t.Errorf("wrong reason: %+v", got)
|
||
}
|
||
}
|