api.go 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826
  1. package main
  2. import (
  3. "bufio"
  4. "context"
  5. "encoding/hex"
  6. "encoding/json"
  7. "fmt"
  8. "log"
  9. "net/http"
  10. "os"
  11. "os/exec"
  12. "runtime"
  13. "strconv"
  14. "strings"
  15. "sync"
  16. "time"
  17. "github.com/gorilla/websocket"
  18. )
  19. // APIServer handles HTTP API requests
  20. type APIServer struct {
  21. daemon *Daemon
  22. upgrader websocket.Upgrader
  23. // HTTP server
  24. httpServer *http.Server
  25. serverMu sync.Mutex
  26. // Recent events (ring buffer)
  27. recentBLE []interface{}
  28. recentWiFi []interface{}
  29. recentMu sync.RWMutex
  30. // WebSocket clients
  31. wsClients map[*websocket.Conn]bool
  32. wsClientsMu sync.Mutex
  33. // Session management
  34. sessions map[string]time.Time
  35. sessionsMu sync.RWMutex
  36. }
  37. // StatusResponse is the response for /api/status
  38. type StatusResponse struct {
  39. DeviceID string `json:"device_id"`
  40. Registered bool `json:"registered"`
  41. Mode string `json:"mode"`
  42. Uptime int64 `json:"uptime_sec"`
  43. Network NetworkStatus `json:"network"`
  44. Scanners ScannerStatus `json:"scanners"`
  45. Counters CounterStatus `json:"counters"`
  46. ServerOK bool `json:"server_ok"`
  47. }
  48. type NetworkStatus struct {
  49. Eth0IP string `json:"eth0_ip,omitempty"`
  50. Eth0RX int64 `json:"eth0_rx,omitempty"`
  51. Eth0TX int64 `json:"eth0_tx,omitempty"`
  52. Wlan0IP string `json:"wlan0_ip,omitempty"`
  53. Wlan0SSID string `json:"wlan0_ssid,omitempty"`
  54. Wlan0Signal int `json:"wlan0_signal,omitempty"`
  55. Wlan0Channel int `json:"wlan0_channel,omitempty"`
  56. Wlan0Gateway string `json:"wlan0_gateway,omitempty"`
  57. Wlan0DNS string `json:"wlan0_dns,omitempty"`
  58. Wlan0RX int64 `json:"wlan0_rx,omitempty"`
  59. Wlan0TX int64 `json:"wlan0_tx,omitempty"`
  60. Gateway string `json:"gateway,omitempty"`
  61. DNS string `json:"dns,omitempty"`
  62. NTP string `json:"ntp,omitempty"`
  63. APActive bool `json:"ap_active"`
  64. }
  65. type ScannerStatus struct {
  66. BLERunning bool `json:"ble_running"`
  67. WiFiRunning bool `json:"wifi_running"`
  68. }
  69. type CounterStatus struct {
  70. BLEEvents uint64 `json:"ble_events"`
  71. WiFiEvents uint64 `json:"wifi_events"`
  72. Uploads uint64 `json:"uploads"`
  73. Errors uint64 `json:"errors"`
  74. }
  75. // MetricsResponse is the response for /api/metrics
  76. type MetricsResponse struct {
  77. CPUPercent float64 `json:"cpu_percent"`
  78. MemUsedMB float64 `json:"mem_used_mb"`
  79. MemTotalMB float64 `json:"mem_total_mb"`
  80. Temperature float64 `json:"temperature"`
  81. LoadAvg float64 `json:"load_avg"`
  82. }
  83. // SettingsRequest is the request for /api/settings
  84. type SettingsRequest struct {
  85. Password string `json:"password"`
  86. Settings SettingsPayload `json:"settings"`
  87. }
  88. // SettingsPayload contains the actual settings
  89. type SettingsPayload struct {
  90. Mode string `json:"mode"`
  91. WifiSSID string `json:"wifi_ssid"`
  92. WifiPSK string `json:"wifi_psk"`
  93. Eth0Mode string `json:"eth0_mode"`
  94. Eth0IP string `json:"eth0_ip"`
  95. Eth0Gateway string `json:"eth0_gateway"`
  96. Eth0DNS string `json:"eth0_dns"`
  97. NTPServers string `json:"ntp_servers"`
  98. }
  99. // NewAPIServer creates a new API server
  100. func NewAPIServer(daemon *Daemon) *APIServer {
  101. return &APIServer{
  102. daemon: daemon,
  103. upgrader: websocket.Upgrader{
  104. CheckOrigin: func(r *http.Request) bool {
  105. return true // Allow all origins for local access
  106. },
  107. },
  108. recentBLE: make([]interface{}, 0, 100),
  109. recentWiFi: make([]interface{}, 0, 100),
  110. wsClients: make(map[*websocket.Conn]bool),
  111. sessions: make(map[string]time.Time),
  112. }
  113. }
  114. // Start starts the HTTP server
  115. func (s *APIServer) Start(addr string) error {
  116. s.serverMu.Lock()
  117. defer s.serverMu.Unlock()
  118. if s.httpServer != nil {
  119. return fmt.Errorf("HTTP server already running")
  120. }
  121. mux := http.NewServeMux()
  122. // API endpoints
  123. mux.HandleFunc("/api/status", s.handleStatus)
  124. mux.HandleFunc("/api/metrics", s.handleMetrics)
  125. mux.HandleFunc("/api/ble/recent", s.handleBLERecent)
  126. mux.HandleFunc("/api/wifi/recent", s.handleWiFiRecent)
  127. mux.HandleFunc("/api/config", s.handleConfig)
  128. mux.HandleFunc("/api/settings", s.handleSettings)
  129. mux.HandleFunc("/api/unlock", s.handleUnlock)
  130. mux.HandleFunc("/api/logs", s.handleLogs)
  131. mux.HandleFunc("/api/ws", s.handleWebSocket)
  132. // Serve static files for dashboard with SPA fallback
  133. mux.Handle("/", s.spaHandler("/opt/mybeacon/www"))
  134. s.httpServer = &http.Server{
  135. Addr: addr,
  136. Handler: s.corsMiddleware(mux),
  137. }
  138. log.Printf("[api] Starting HTTP server on %s", addr)
  139. return s.httpServer.ListenAndServe()
  140. }
  141. // Stop stops the HTTP server
  142. func (s *APIServer) Stop() error {
  143. s.serverMu.Lock()
  144. defer s.serverMu.Unlock()
  145. if s.httpServer == nil {
  146. return nil // Already stopped
  147. }
  148. log.Println("[api] Stopping HTTP server...")
  149. ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
  150. defer cancel()
  151. err := s.httpServer.Shutdown(ctx)
  152. s.httpServer = nil
  153. return err
  154. }
  155. // IsRunning returns true if HTTP server is running
  156. func (s *APIServer) IsRunning() bool {
  157. s.serverMu.Lock()
  158. defer s.serverMu.Unlock()
  159. return s.httpServer != nil
  160. }
  161. // spaHandler serves static files with SPA fallback
  162. func (s *APIServer) spaHandler(staticPath string) http.Handler {
  163. fileServer := http.FileServer(http.Dir(staticPath))
  164. return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
  165. // Check if file exists
  166. path := staticPath + r.URL.Path
  167. _, err := os.Stat(path)
  168. // If file doesn't exist and it's not the root, serve index.html (SPA routing)
  169. if os.IsNotExist(err) && r.URL.Path != "/" {
  170. // Serve index.html for SPA routing
  171. http.ServeFile(w, r, staticPath+"/index.html")
  172. return
  173. }
  174. // Serve the file normally
  175. fileServer.ServeHTTP(w, r)
  176. })
  177. }
  178. // corsMiddleware adds CORS headers
  179. func (s *APIServer) corsMiddleware(next http.Handler) http.Handler {
  180. return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
  181. w.Header().Set("Access-Control-Allow-Origin", "*")
  182. w.Header().Set("Access-Control-Allow-Methods", "GET, POST, OPTIONS")
  183. w.Header().Set("Access-Control-Allow-Headers", "Content-Type, Authorization")
  184. if r.Method == "OPTIONS" {
  185. w.WriteHeader(http.StatusOK)
  186. return
  187. }
  188. next.ServeHTTP(w, r)
  189. })
  190. }
  191. // handleStatus returns device status
  192. func (s *APIServer) handleStatus(w http.ResponseWriter, r *http.Request) {
  193. s.daemon.mu.Lock()
  194. defer s.daemon.mu.Unlock()
  195. eth0rx, eth0tx := getInterfaceStats("eth0")
  196. wlan0rx, wlan0tx := getInterfaceStats("wlan0")
  197. wlanInfo := getWlanInfo()
  198. // Gateway and DNS are shared, but wlan0 might have its own
  199. gateway := getDefaultGateway()
  200. dns := getDNS()
  201. status := StatusResponse{
  202. DeviceID: s.daemon.state.DeviceID,
  203. Registered: s.daemon.state.DeviceToken != "",
  204. Mode: "cloud", // TODO: implement mode switching
  205. Uptime: getUptime(),
  206. Network: NetworkStatus{
  207. Eth0IP: getInterfaceIP("eth0"),
  208. Eth0RX: eth0rx,
  209. Eth0TX: eth0tx,
  210. Wlan0IP: getInterfaceIP("wlan0"),
  211. Wlan0SSID: wlanInfo.ssid,
  212. Wlan0Signal: wlanInfo.signal,
  213. Wlan0Channel: wlanInfo.channel,
  214. Wlan0Gateway: gateway, // same gateway for now
  215. Wlan0DNS: dns, // same DNS for now
  216. Wlan0RX: wlan0rx,
  217. Wlan0TX: wlan0tx,
  218. Gateway: gateway,
  219. DNS: dns,
  220. NTP: "pool.ntp.org",
  221. },
  222. Scanners: ScannerStatus{
  223. BLERunning: s.daemon.scanners.IsBLERunning(),
  224. WiFiRunning: s.daemon.scanners.IsWiFiRunning(),
  225. },
  226. ServerOK: s.daemon.state.DeviceToken != "",
  227. }
  228. s.jsonResponse(w, status)
  229. }
  230. // handleMetrics returns system metrics
  231. func (s *APIServer) handleMetrics(w http.ResponseWriter, r *http.Request) {
  232. metrics := MetricsResponse{
  233. CPUPercent: getCPUPercent(),
  234. MemUsedMB: getMemUsedMB(),
  235. MemTotalMB: getMemTotalMB(),
  236. Temperature: getTemperature(),
  237. LoadAvg: getLoadAvg(),
  238. }
  239. s.jsonResponse(w, metrics)
  240. }
  241. // handleBLERecent returns recent BLE events
  242. func (s *APIServer) handleBLERecent(w http.ResponseWriter, r *http.Request) {
  243. s.recentMu.RLock()
  244. events := make([]interface{}, len(s.recentBLE))
  245. copy(events, s.recentBLE)
  246. s.recentMu.RUnlock()
  247. s.jsonResponse(w, events)
  248. }
  249. // handleWiFiRecent returns recent WiFi events
  250. func (s *APIServer) handleWiFiRecent(w http.ResponseWriter, r *http.Request) {
  251. s.recentMu.RLock()
  252. events := make([]interface{}, len(s.recentWiFi))
  253. copy(events, s.recentWiFi)
  254. s.recentMu.RUnlock()
  255. s.jsonResponse(w, events)
  256. }
  257. // handleLogs returns recent daemon log lines
  258. func (s *APIServer) handleLogs(w http.ResponseWriter, r *http.Request) {
  259. logFile := "/var/log/mybeacon.log"
  260. data, err := os.ReadFile(logFile)
  261. if err != nil {
  262. s.jsonResponse(w, []string{})
  263. return
  264. }
  265. lines := strings.Split(string(data), "\n")
  266. // Return last 500 lines
  267. if len(lines) > 500 {
  268. lines = lines[len(lines)-500:]
  269. }
  270. // Filter out empty lines
  271. var result []string
  272. for _, line := range lines {
  273. if strings.TrimSpace(line) != "" {
  274. result = append(result, line)
  275. }
  276. }
  277. s.jsonResponse(w, result)
  278. }
  279. // handleConfig returns current config (without secrets)
  280. func (s *APIServer) handleConfig(w http.ResponseWriter, r *http.Request) {
  281. s.daemon.mu.Lock()
  282. cfg := *s.daemon.cfg
  283. s.daemon.mu.Unlock()
  284. // Remove secrets
  285. cfg.SSHTunnel.KeyPath = ""
  286. s.jsonResponse(w, cfg)
  287. }
  288. // handleSettings handles settings updates (requires password)
  289. func (s *APIServer) handleSettings(w http.ResponseWriter, r *http.Request) {
  290. if r.Method != "POST" {
  291. http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
  292. return
  293. }
  294. var req SettingsRequest
  295. if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
  296. http.Error(w, "Invalid JSON", http.StatusBadRequest)
  297. return
  298. }
  299. // Verify password
  300. if !s.verifyPassword(req.Password) {
  301. http.Error(w, "Invalid password", http.StatusUnauthorized)
  302. return
  303. }
  304. s.daemon.mu.Lock()
  305. // Apply mode change
  306. if req.Settings.Mode != "" && (req.Settings.Mode == "cloud" || req.Settings.Mode == "lan") {
  307. if s.daemon.cfg.Mode != req.Settings.Mode {
  308. log.Printf("[api] Mode changed: %s -> %s", s.daemon.cfg.Mode, req.Settings.Mode)
  309. s.daemon.cfg.Mode = req.Settings.Mode
  310. }
  311. }
  312. // Apply NTP settings
  313. if req.Settings.NTPServers != "" {
  314. servers := strings.Split(req.Settings.NTPServers, ",")
  315. for i := range servers {
  316. servers[i] = strings.TrimSpace(servers[i])
  317. }
  318. s.daemon.cfg.Network.NTPServers = servers
  319. log.Printf("[api] NTP servers updated: %v", servers)
  320. }
  321. // Apply eth0 settings (always local, never from server)
  322. if req.Settings.Eth0Mode == "static" {
  323. s.daemon.cfg.Network.Eth0.Static = true
  324. s.daemon.cfg.Network.Eth0.Address = req.Settings.Eth0IP
  325. s.daemon.cfg.Network.Eth0.Gateway = req.Settings.Eth0Gateway
  326. s.daemon.cfg.Network.Eth0.DNS = req.Settings.Eth0DNS
  327. log.Printf("[api] eth0 static IP configured: %s", req.Settings.Eth0IP)
  328. } else if req.Settings.Eth0Mode == "dhcp" {
  329. s.daemon.cfg.Network.Eth0.Static = false
  330. log.Printf("[api] eth0 set to DHCP")
  331. }
  332. // Save local config
  333. SaveConfig(s.daemon.configPath, s.daemon.cfg)
  334. s.daemon.mu.Unlock()
  335. // Apply WiFi settings if provided (this may take time)
  336. if req.Settings.WifiSSID != "" {
  337. // Also save to local config
  338. s.daemon.mu.Lock()
  339. s.daemon.cfg.WiFi.SSID = req.Settings.WifiSSID
  340. s.daemon.cfg.WiFi.PSK = req.Settings.WifiPSK
  341. s.daemon.cfg.WiFi.ClientEnabled = true
  342. SaveConfig(s.daemon.configPath, s.daemon.cfg)
  343. s.daemon.mu.Unlock()
  344. if err := applyWiFiSettings(req.Settings.WifiSSID, req.Settings.WifiPSK); err != nil {
  345. log.Printf("[api] Failed to apply WiFi settings: %v", err)
  346. http.Error(w, "Failed to apply WiFi settings: "+err.Error(), http.StatusInternalServerError)
  347. return
  348. }
  349. log.Printf("[api] WiFi settings applied: SSID=%s", req.Settings.WifiSSID)
  350. }
  351. s.jsonResponse(w, map[string]bool{"success": true})
  352. }
  353. // handleUnlock verifies password and creates session
  354. func (s *APIServer) handleUnlock(w http.ResponseWriter, r *http.Request) {
  355. if r.Method != "POST" {
  356. http.Error(w, "Method not allowed", http.StatusMethodNotAllowed)
  357. return
  358. }
  359. var req struct {
  360. Password string `json:"password"`
  361. }
  362. if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
  363. http.Error(w, "Invalid JSON", http.StatusBadRequest)
  364. return
  365. }
  366. if !s.verifyPassword(req.Password) {
  367. http.Error(w, "Invalid password", http.StatusUnauthorized)
  368. return
  369. }
  370. // Create session token (simple implementation)
  371. token := generateToken(32)
  372. s.sessionsMu.Lock()
  373. s.sessions[token] = time.Now().Add(30 * time.Minute)
  374. s.sessionsMu.Unlock()
  375. s.jsonResponse(w, map[string]string{"token": token})
  376. }
  377. // handleWebSocket handles WebSocket connections for live updates
  378. func (s *APIServer) handleWebSocket(w http.ResponseWriter, r *http.Request) {
  379. conn, err := s.upgrader.Upgrade(w, r, nil)
  380. if err != nil {
  381. log.Printf("[api] WebSocket upgrade error: %v", err)
  382. return
  383. }
  384. defer conn.Close()
  385. s.wsClientsMu.Lock()
  386. s.wsClients[conn] = true
  387. s.wsClientsMu.Unlock()
  388. defer func() {
  389. s.wsClientsMu.Lock()
  390. delete(s.wsClients, conn)
  391. s.wsClientsMu.Unlock()
  392. }()
  393. // Keep connection alive and read messages
  394. for {
  395. _, _, err := conn.ReadMessage()
  396. if err != nil {
  397. break
  398. }
  399. }
  400. }
  401. // AddBLEEvent adds a BLE event to recent list and broadcasts to WebSocket
  402. func (s *APIServer) AddBLEEvent(event interface{}) {
  403. s.recentMu.Lock()
  404. s.recentBLE = append(s.recentBLE, event)
  405. if len(s.recentBLE) > 100 {
  406. s.recentBLE = s.recentBLE[1:]
  407. }
  408. s.recentMu.Unlock()
  409. s.broadcast(map[string]interface{}{
  410. "type": "ble",
  411. "event": event,
  412. })
  413. }
  414. // AddWiFiEvent adds a WiFi event to recent list and broadcasts to WebSocket
  415. func (s *APIServer) AddWiFiEvent(event interface{}) {
  416. s.recentMu.Lock()
  417. s.recentWiFi = append(s.recentWiFi, event)
  418. if len(s.recentWiFi) > 100 {
  419. s.recentWiFi = s.recentWiFi[1:]
  420. }
  421. s.recentMu.Unlock()
  422. s.broadcast(map[string]interface{}{
  423. "type": "wifi",
  424. "event": event,
  425. })
  426. }
  427. // broadcast sends a message to all WebSocket clients
  428. func (s *APIServer) broadcast(msg interface{}) {
  429. data, err := json.Marshal(msg)
  430. if err != nil {
  431. return
  432. }
  433. s.wsClientsMu.Lock()
  434. defer s.wsClientsMu.Unlock()
  435. for conn := range s.wsClients {
  436. if err := conn.WriteMessage(websocket.TextMessage, data); err != nil {
  437. conn.Close()
  438. delete(s.wsClients, conn)
  439. }
  440. }
  441. }
  442. // verifyPassword checks if password matches device password
  443. func (s *APIServer) verifyPassword(password string) bool {
  444. s.daemon.mu.Lock()
  445. devicePassword := s.daemon.state.DevicePassword
  446. s.daemon.mu.Unlock()
  447. // Use default password "admin" if no password set (device not registered)
  448. if devicePassword == "" {
  449. devicePassword = "admin"
  450. }
  451. return password != "" && password == devicePassword
  452. }
  453. // jsonResponse sends a JSON response
  454. func (s *APIServer) jsonResponse(w http.ResponseWriter, data interface{}) {
  455. w.Header().Set("Content-Type", "application/json")
  456. json.NewEncoder(w).Encode(data)
  457. }
  458. // Helper functions for system metrics
  459. func getUptime() int64 {
  460. data, err := os.ReadFile("/proc/uptime")
  461. if err != nil {
  462. return 0
  463. }
  464. var uptime float64
  465. if err := parseFloat(string(data), &uptime); err != nil {
  466. return 0
  467. }
  468. return int64(uptime)
  469. }
  470. func parseFloat(s string, f *float64) error {
  471. parts := strings.Fields(s)
  472. if len(parts) == 0 {
  473. return nil
  474. }
  475. return json.Unmarshal([]byte(parts[0]), f)
  476. }
  477. func getCPUPercent() float64 {
  478. // Simplified - read from /proc/stat
  479. return 0 // TODO: implement proper CPU usage
  480. }
  481. func getMemUsedMB() float64 {
  482. var m runtime.MemStats
  483. runtime.ReadMemStats(&m)
  484. return float64(m.Alloc) / 1024 / 1024
  485. }
  486. func getMemTotalMB() float64 {
  487. data, err := os.ReadFile("/proc/meminfo")
  488. if err != nil {
  489. return 0
  490. }
  491. scanner := bufio.NewScanner(strings.NewReader(string(data)))
  492. for scanner.Scan() {
  493. line := scanner.Text()
  494. if strings.HasPrefix(line, "MemTotal:") {
  495. var total int64
  496. if err := parseMemInfo(line, &total); err == nil {
  497. return float64(total) / 1024
  498. }
  499. }
  500. }
  501. return 0
  502. }
  503. func parseMemInfo(line string, value *int64) error {
  504. parts := strings.Fields(line)
  505. if len(parts) >= 2 {
  506. return json.Unmarshal([]byte(parts[1]), value)
  507. }
  508. return nil
  509. }
  510. func getTemperature() float64 {
  511. data, err := os.ReadFile("/sys/class/thermal/thermal_zone0/temp")
  512. if err != nil {
  513. return 0
  514. }
  515. var temp int64
  516. if err := json.Unmarshal([]byte(strings.TrimSpace(string(data))), &temp); err != nil {
  517. return 0
  518. }
  519. return float64(temp) / 1000
  520. }
  521. func getLoadAvg() float64 {
  522. data, err := os.ReadFile("/proc/loadavg")
  523. if err != nil {
  524. return 0
  525. }
  526. var load float64
  527. if err := parseFloat(string(data), &load); err != nil {
  528. return 0
  529. }
  530. return load
  531. }
  532. func generateToken(length int) string {
  533. const chars = "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789"
  534. b := make([]byte, length)
  535. for i := range b {
  536. b[i] = chars[time.Now().UnixNano()%int64(len(chars))]
  537. time.Sleep(time.Nanosecond)
  538. }
  539. return string(b)
  540. }
  541. func getInterfaceStats(name string) (rx, tx int64) {
  542. rxPath := "/sys/class/net/" + name + "/statistics/rx_bytes"
  543. txPath := "/sys/class/net/" + name + "/statistics/tx_bytes"
  544. if data, err := os.ReadFile(rxPath); err == nil {
  545. json.Unmarshal([]byte(strings.TrimSpace(string(data))), &rx)
  546. }
  547. if data, err := os.ReadFile(txPath); err == nil {
  548. json.Unmarshal([]byte(strings.TrimSpace(string(data))), &tx)
  549. }
  550. return
  551. }
  552. func getDefaultGateway() string {
  553. data, err := os.ReadFile("/proc/net/route")
  554. if err != nil {
  555. return ""
  556. }
  557. scanner := bufio.NewScanner(strings.NewReader(string(data)))
  558. for scanner.Scan() {
  559. fields := strings.Fields(scanner.Text())
  560. if len(fields) >= 3 && fields[1] == "00000000" {
  561. gw := fields[2]
  562. if len(gw) == 8 {
  563. b, _ := hex.DecodeString(gw)
  564. if len(b) == 4 {
  565. // Little-endian: reverse bytes
  566. return fmt.Sprintf("%d.%d.%d.%d", b[3], b[2], b[1], b[0])
  567. }
  568. }
  569. }
  570. }
  571. return ""
  572. }
  573. func getDNS() string {
  574. data, err := os.ReadFile("/etc/resolv.conf")
  575. if err != nil {
  576. return ""
  577. }
  578. var servers []string
  579. scanner := bufio.NewScanner(strings.NewReader(string(data)))
  580. for scanner.Scan() {
  581. line := scanner.Text()
  582. if strings.HasPrefix(line, "nameserver ") {
  583. servers = append(servers, strings.TrimPrefix(line, "nameserver "))
  584. }
  585. }
  586. return strings.Join(servers, ", ")
  587. }
  588. type wlanInfoResult struct {
  589. ssid string
  590. signal int
  591. channel int
  592. }
  593. func getWlanInfo() wlanInfoResult {
  594. var info wlanInfoResult
  595. // Get SSID, signal, channel from iw dev wlan0 link
  596. out, err := exec.Command("iw", "dev", "wlan0", "link").Output()
  597. if err == nil {
  598. scanner := bufio.NewScanner(strings.NewReader(string(out)))
  599. for scanner.Scan() {
  600. line := strings.TrimSpace(scanner.Text())
  601. if strings.HasPrefix(line, "SSID:") {
  602. info.ssid = strings.TrimSpace(strings.TrimPrefix(line, "SSID:"))
  603. } else if strings.HasPrefix(line, "signal:") {
  604. // signal: -30 dBm
  605. parts := strings.Fields(line)
  606. if len(parts) >= 2 {
  607. if v, err := strconv.Atoi(parts[1]); err == nil {
  608. info.signal = v
  609. }
  610. }
  611. } else if strings.HasPrefix(line, "freq:") {
  612. // freq: 2462
  613. parts := strings.Fields(line)
  614. if len(parts) >= 2 {
  615. if f, err := strconv.Atoi(parts[1]); err == nil {
  616. info.channel = freqToChannel(f)
  617. }
  618. }
  619. }
  620. }
  621. }
  622. return info
  623. }
  624. // freqToChannel converts WiFi frequency (MHz) to channel number
  625. func freqToChannel(freq int) int {
  626. if freq >= 2412 && freq <= 2484 {
  627. if freq == 2484 {
  628. return 14
  629. }
  630. return (freq - 2407) / 5
  631. }
  632. if freq >= 5170 && freq <= 5825 {
  633. return (freq - 5000) / 5
  634. }
  635. return 0
  636. }
  637. // applyWiFiSettings configures wpa_supplicant and connects to WiFi
  638. func applyWiFiSettings(ssid, psk string) error {
  639. // Retry the whole connection up to 3 times
  640. var lastErr error
  641. for attempt := 1; attempt <= 3; attempt++ {
  642. log.Printf("[wifi] Connection attempt %d/3 to SSID=%s", attempt, ssid)
  643. if err := tryWiFiConnect(ssid, psk); err != nil {
  644. lastErr = err
  645. log.Printf("[wifi] Attempt %d failed: %v", attempt, err)
  646. if attempt < 3 {
  647. log.Printf("[wifi] Waiting 5s before retry...")
  648. time.Sleep(5 * time.Second)
  649. }
  650. continue
  651. }
  652. log.Println("[wifi] WiFi client connected successfully")
  653. return nil
  654. }
  655. return fmt.Errorf("all connection attempts failed: %w", lastErr)
  656. }
  657. func tryWiFiConnect(ssid, psk string) error {
  658. // Use external shell script for WiFi connection (more reliable)
  659. scriptPath := "/opt/mybeacon/bin/wifi-connect.sh"
  660. cmd := exec.Command(scriptPath, ssid, psk)
  661. output, err := cmd.CombinedOutput()
  662. if err != nil {
  663. return fmt.Errorf("wifi-connect.sh failed: %w (output: %s)", err, string(output))
  664. }
  665. log.Printf("[wifi] %s", strings.TrimSpace(string(output)))
  666. // Ensure wlan0 is configured in /etc/network/interfaces (for manual use)
  667. ensureWlan0Interface()
  668. return nil
  669. }
  670. func ensureWlan0Interface() error {
  671. interfacesPath := "/etc/network/interfaces"
  672. data, err := os.ReadFile(interfacesPath)
  673. if err != nil {
  674. return err
  675. }
  676. content := string(data)
  677. // Check if wlan0 is already configured (not commented out)
  678. lines := strings.Split(content, "\n")
  679. for _, line := range lines {
  680. trimmed := strings.TrimSpace(line)
  681. if !strings.HasPrefix(trimmed, "#") && strings.Contains(trimmed, "iface wlan0 inet") {
  682. return nil // Already configured
  683. }
  684. }
  685. // Add wlan0 configuration
  686. wlan0Config := `
  687. # WiFi (managed by beacon-daemon)
  688. auto wlan0
  689. iface wlan0 inet dhcp
  690. pre-up wpa_supplicant -B -i wlan0 -c /etc/wpa_supplicant/wpa_supplicant.conf
  691. post-down killall wpa_supplicant 2>/dev/null || true
  692. `
  693. if err := os.WriteFile(interfacesPath, []byte(content+wlan0Config), 0644); err != nil {
  694. return err
  695. }
  696. log.Println("[wifi] Added wlan0 configuration to /etc/network/interfaces")
  697. return nil
  698. }
  699. // disconnectWiFiClient disconnects from WiFi and stops wpa_supplicant
  700. func disconnectWiFiClient() {
  701. log.Println("[wifi] Disconnecting WiFi client...")
  702. exec.Command("killall", "wpa_supplicant", "udhcpc", "dhcpcd").Run()
  703. exec.Command("ip", "addr", "flush", "dev", "wlan0").Run()
  704. os.RemoveAll("/var/run/wpa_supplicant/wlan0")
  705. log.Println("[wifi] WiFi client disconnected")
  706. }