1
0
mirror of https://github.com/AvengeMedia/DankMaterialShell.git synced 2025-12-10 07:25:37 -05:00
Files
DankMaterialShell/backend/internal/server/network/backend_networkmanager_wifi.go
2025-11-12 17:18:45 -05:00

719 lines
18 KiB
Go

package network
import (
"bytes"
"fmt"
"sort"
"github.com/AvengeMedia/DankMaterialShell/backend/internal/log"
"github.com/Wifx/gonetworkmanager/v2"
)
func (b *NetworkManagerBackend) GetWiFiEnabled() (bool, error) {
nm := b.nmConn.(gonetworkmanager.NetworkManager)
return nm.GetPropertyWirelessEnabled()
}
func (b *NetworkManagerBackend) SetWiFiEnabled(enabled bool) error {
nm := b.nmConn.(gonetworkmanager.NetworkManager)
err := nm.SetPropertyWirelessEnabled(enabled)
if err != nil {
return fmt.Errorf("failed to set WiFi enabled: %w", err)
}
b.stateMutex.Lock()
b.state.WiFiEnabled = enabled
b.stateMutex.Unlock()
if b.onStateChange != nil {
b.onStateChange()
}
return nil
}
func (b *NetworkManagerBackend) ScanWiFi() error {
if b.wifiDevice == nil {
return fmt.Errorf("no WiFi device available")
}
b.stateMutex.RLock()
enabled := b.state.WiFiEnabled
b.stateMutex.RUnlock()
if !enabled {
return fmt.Errorf("WiFi is disabled")
}
if err := b.ensureWiFiDevice(); err != nil {
return err
}
w := b.wifiDev.(gonetworkmanager.DeviceWireless)
err := w.RequestScan()
if err != nil {
return fmt.Errorf("scan request failed: %w", err)
}
_, err = b.updateWiFiNetworks()
return err
}
func (b *NetworkManagerBackend) GetWiFiNetworkDetails(ssid string) (*NetworkInfoResponse, error) {
if b.wifiDevice == nil {
return nil, fmt.Errorf("no WiFi device available")
}
if err := b.ensureWiFiDevice(); err != nil {
return nil, err
}
wifiDev := b.wifiDev
w := wifiDev.(gonetworkmanager.DeviceWireless)
apPaths, err := w.GetAccessPoints()
if err != nil {
return nil, fmt.Errorf("failed to get access points: %w", err)
}
s := b.settings
if s == nil {
s, err = gonetworkmanager.NewSettings()
if err != nil {
return nil, fmt.Errorf("failed to get settings: %w", err)
}
b.settings = s
}
settingsMgr := s.(gonetworkmanager.Settings)
connections, err := settingsMgr.ListConnections()
if err != nil {
return nil, fmt.Errorf("failed to get connections: %w", err)
}
savedSSIDs := make(map[string]bool)
autoconnectMap := make(map[string]bool)
for _, conn := range connections {
connSettings, err := conn.GetSettings()
if err != nil {
continue
}
if connMeta, ok := connSettings["connection"]; ok {
if connType, ok := connMeta["type"].(string); ok && connType == "802-11-wireless" {
if wifiSettings, ok := connSettings["802-11-wireless"]; ok {
if ssidBytes, ok := wifiSettings["ssid"].([]byte); ok {
savedSSID := string(ssidBytes)
savedSSIDs[savedSSID] = true
autoconnect := true
if ac, ok := connMeta["autoconnect"].(bool); ok {
autoconnect = ac
}
autoconnectMap[savedSSID] = autoconnect
}
}
}
}
}
b.stateMutex.RLock()
currentSSID := b.state.WiFiSSID
currentBSSID := b.state.WiFiBSSID
b.stateMutex.RUnlock()
var bands []WiFiNetwork
for _, ap := range apPaths {
apSSID, err := ap.GetPropertySSID()
if err != nil || apSSID != ssid {
continue
}
strength, _ := ap.GetPropertyStrength()
flags, _ := ap.GetPropertyFlags()
wpaFlags, _ := ap.GetPropertyWPAFlags()
rsnFlags, _ := ap.GetPropertyRSNFlags()
freq, _ := ap.GetPropertyFrequency()
maxBitrate, _ := ap.GetPropertyMaxBitrate()
bssid, _ := ap.GetPropertyHWAddress()
mode, _ := ap.GetPropertyMode()
secured := flags != uint32(gonetworkmanager.Nm80211APFlagsNone) ||
wpaFlags != uint32(gonetworkmanager.Nm80211APSecNone) ||
rsnFlags != uint32(gonetworkmanager.Nm80211APSecNone)
enterprise := (rsnFlags&uint32(gonetworkmanager.Nm80211APSecKeyMgmt8021X) != 0) ||
(wpaFlags&uint32(gonetworkmanager.Nm80211APSecKeyMgmt8021X) != 0)
var modeStr string
switch mode {
case gonetworkmanager.Nm80211ModeAdhoc:
modeStr = "adhoc"
case gonetworkmanager.Nm80211ModeInfra:
modeStr = "infrastructure"
case gonetworkmanager.Nm80211ModeAp:
modeStr = "ap"
default:
modeStr = "unknown"
}
channel := frequencyToChannel(freq)
network := WiFiNetwork{
SSID: ssid,
BSSID: bssid,
Signal: strength,
Secured: secured,
Enterprise: enterprise,
Connected: ssid == currentSSID && bssid == currentBSSID,
Saved: savedSSIDs[ssid],
Autoconnect: autoconnectMap[ssid],
Frequency: freq,
Mode: modeStr,
Rate: maxBitrate / 1000,
Channel: channel,
}
bands = append(bands, network)
}
if len(bands) == 0 {
return nil, fmt.Errorf("network not found: %s", ssid)
}
sort.Slice(bands, func(i, j int) bool {
if bands[i].Connected && !bands[j].Connected {
return true
}
if !bands[i].Connected && bands[j].Connected {
return false
}
return bands[i].Signal > bands[j].Signal
})
return &NetworkInfoResponse{
SSID: ssid,
Bands: bands,
}, nil
}
func (b *NetworkManagerBackend) ConnectWiFi(req ConnectionRequest) error {
if b.wifiDevice == nil {
return fmt.Errorf("no WiFi device available")
}
b.stateMutex.RLock()
alreadyConnected := b.state.WiFiConnected && b.state.WiFiSSID == req.SSID
b.stateMutex.RUnlock()
if alreadyConnected && !req.Interactive {
return nil
}
b.stateMutex.Lock()
b.state.IsConnecting = true
b.state.ConnectingSSID = req.SSID
b.state.LastError = ""
b.stateMutex.Unlock()
if b.onStateChange != nil {
b.onStateChange()
}
nm := b.nmConn.(gonetworkmanager.NetworkManager)
existingConn, err := b.findConnection(req.SSID)
if err == nil && existingConn != nil {
dev := b.wifiDevice.(gonetworkmanager.Device)
_, err := nm.ActivateConnection(existingConn, dev, nil)
if err != nil {
log.Warnf("[ConnectWiFi] Failed to activate existing connection: %v", err)
b.stateMutex.Lock()
b.state.IsConnecting = false
b.state.ConnectingSSID = ""
b.state.LastError = fmt.Sprintf("failed to activate connection: %v", err)
b.stateMutex.Unlock()
if b.onStateChange != nil {
b.onStateChange()
}
return fmt.Errorf("failed to activate connection: %w", err)
}
return nil
}
if err := b.createAndConnectWiFi(req); err != nil {
log.Warnf("[ConnectWiFi] Failed to create and connect: %v", err)
b.stateMutex.Lock()
b.state.IsConnecting = false
b.state.ConnectingSSID = ""
b.state.LastError = err.Error()
b.stateMutex.Unlock()
if b.onStateChange != nil {
b.onStateChange()
}
return err
}
return nil
}
func (b *NetworkManagerBackend) DisconnectWiFi() error {
if b.wifiDevice == nil {
return fmt.Errorf("no WiFi device available")
}
dev := b.wifiDevice.(gonetworkmanager.Device)
err := dev.Disconnect()
if err != nil {
return fmt.Errorf("failed to disconnect: %w", err)
}
b.updateWiFiState()
b.updatePrimaryConnection()
if b.onStateChange != nil {
b.onStateChange()
}
return nil
}
func (b *NetworkManagerBackend) ForgetWiFiNetwork(ssid string) error {
conn, err := b.findConnection(ssid)
if err != nil {
return fmt.Errorf("connection not found: %w", err)
}
b.stateMutex.RLock()
currentSSID := b.state.WiFiSSID
isConnected := b.state.WiFiConnected
b.stateMutex.RUnlock()
err = conn.Delete()
if err != nil {
return fmt.Errorf("failed to delete connection: %w", err)
}
if isConnected && currentSSID == ssid {
b.stateMutex.Lock()
b.state.WiFiConnected = false
b.state.WiFiSSID = ""
b.state.WiFiBSSID = ""
b.state.WiFiSignal = 0
b.state.WiFiIP = ""
b.state.NetworkStatus = StatusDisconnected
b.stateMutex.Unlock()
}
b.updateWiFiNetworks()
if b.onStateChange != nil {
b.onStateChange()
}
return nil
}
func (b *NetworkManagerBackend) IsConnectingTo(ssid string) bool {
b.stateMutex.RLock()
defer b.stateMutex.RUnlock()
return b.state.IsConnecting && b.state.ConnectingSSID == ssid
}
func (b *NetworkManagerBackend) updateWiFiNetworks() ([]WiFiNetwork, error) {
if b.wifiDevice == nil {
return nil, fmt.Errorf("no WiFi device available")
}
if err := b.ensureWiFiDevice(); err != nil {
return nil, err
}
wifiDev := b.wifiDev
w := wifiDev.(gonetworkmanager.DeviceWireless)
apPaths, err := w.GetAccessPoints()
if err != nil {
return nil, fmt.Errorf("failed to get access points: %w", err)
}
s := b.settings
if s == nil {
s, err = gonetworkmanager.NewSettings()
if err != nil {
return nil, fmt.Errorf("failed to get settings: %w", err)
}
b.settings = s
}
settingsMgr := s.(gonetworkmanager.Settings)
connections, err := settingsMgr.ListConnections()
if err != nil {
return nil, fmt.Errorf("failed to get connections: %w", err)
}
savedSSIDs := make(map[string]bool)
autoconnectMap := make(map[string]bool)
for _, conn := range connections {
connSettings, err := conn.GetSettings()
if err != nil {
continue
}
if connMeta, ok := connSettings["connection"]; ok {
if connType, ok := connMeta["type"].(string); ok && connType == "802-11-wireless" {
if wifiSettings, ok := connSettings["802-11-wireless"]; ok {
if ssidBytes, ok := wifiSettings["ssid"].([]byte); ok {
ssid := string(ssidBytes)
savedSSIDs[ssid] = true
autoconnect := true
if ac, ok := connMeta["autoconnect"].(bool); ok {
autoconnect = ac
}
autoconnectMap[ssid] = autoconnect
}
}
}
}
}
b.stateMutex.RLock()
currentSSID := b.state.WiFiSSID
b.stateMutex.RUnlock()
seenSSIDs := make(map[string]*WiFiNetwork)
networks := []WiFiNetwork{}
for _, ap := range apPaths {
ssid, err := ap.GetPropertySSID()
if err != nil || ssid == "" {
continue
}
if existing, exists := seenSSIDs[ssid]; exists {
strength, _ := ap.GetPropertyStrength()
if strength > existing.Signal {
existing.Signal = strength
freq, _ := ap.GetPropertyFrequency()
existing.Frequency = freq
bssid, _ := ap.GetPropertyHWAddress()
existing.BSSID = bssid
}
continue
}
strength, _ := ap.GetPropertyStrength()
flags, _ := ap.GetPropertyFlags()
wpaFlags, _ := ap.GetPropertyWPAFlags()
rsnFlags, _ := ap.GetPropertyRSNFlags()
freq, _ := ap.GetPropertyFrequency()
maxBitrate, _ := ap.GetPropertyMaxBitrate()
bssid, _ := ap.GetPropertyHWAddress()
mode, _ := ap.GetPropertyMode()
secured := flags != uint32(gonetworkmanager.Nm80211APFlagsNone) ||
wpaFlags != uint32(gonetworkmanager.Nm80211APSecNone) ||
rsnFlags != uint32(gonetworkmanager.Nm80211APSecNone)
enterprise := (rsnFlags&uint32(gonetworkmanager.Nm80211APSecKeyMgmt8021X) != 0) ||
(wpaFlags&uint32(gonetworkmanager.Nm80211APSecKeyMgmt8021X) != 0)
var modeStr string
switch mode {
case gonetworkmanager.Nm80211ModeAdhoc:
modeStr = "adhoc"
case gonetworkmanager.Nm80211ModeInfra:
modeStr = "infrastructure"
case gonetworkmanager.Nm80211ModeAp:
modeStr = "ap"
default:
modeStr = "unknown"
}
channel := frequencyToChannel(freq)
network := WiFiNetwork{
SSID: ssid,
BSSID: bssid,
Signal: strength,
Secured: secured,
Enterprise: enterprise,
Connected: ssid == currentSSID,
Saved: savedSSIDs[ssid],
Autoconnect: autoconnectMap[ssid],
Frequency: freq,
Mode: modeStr,
Rate: maxBitrate / 1000,
Channel: channel,
}
seenSSIDs[ssid] = &network
networks = append(networks, network)
}
sortWiFiNetworks(networks)
b.stateMutex.Lock()
b.state.WiFiNetworks = networks
b.stateMutex.Unlock()
return networks, nil
}
func (b *NetworkManagerBackend) findConnection(ssid string) (gonetworkmanager.Connection, error) {
s := b.settings
if s == nil {
var err error
s, err = gonetworkmanager.NewSettings()
if err != nil {
return nil, err
}
b.settings = s
}
settings := s.(gonetworkmanager.Settings)
connections, err := settings.ListConnections()
if err != nil {
return nil, err
}
ssidBytes := []byte(ssid)
for _, conn := range connections {
connSettings, err := conn.GetSettings()
if err != nil {
continue
}
if connMeta, ok := connSettings["connection"]; ok {
if connType, ok := connMeta["type"].(string); ok && connType == "802-11-wireless" {
if wifiSettings, ok := connSettings["802-11-wireless"]; ok {
if candidateSSID, ok := wifiSettings["ssid"].([]byte); ok {
if bytes.Equal(candidateSSID, ssidBytes) {
return conn, nil
}
}
}
}
}
}
return nil, fmt.Errorf("connection not found")
}
func (b *NetworkManagerBackend) createAndConnectWiFi(req ConnectionRequest) error {
if b.wifiDevice == nil {
return fmt.Errorf("no WiFi device available")
}
nm := b.nmConn.(gonetworkmanager.NetworkManager)
dev := b.wifiDevice.(gonetworkmanager.Device)
if err := b.ensureWiFiDevice(); err != nil {
return err
}
wifiDev := b.wifiDev
w := wifiDev.(gonetworkmanager.DeviceWireless)
apPaths, err := w.GetAccessPoints()
if err != nil {
return fmt.Errorf("failed to get access points: %w", err)
}
var targetAP gonetworkmanager.AccessPoint
for _, ap := range apPaths {
ssid, err := ap.GetPropertySSID()
if err != nil || ssid != req.SSID {
continue
}
targetAP = ap
break
}
if targetAP == nil {
return fmt.Errorf("access point not found: %s", req.SSID)
}
flags, _ := targetAP.GetPropertyFlags()
wpaFlags, _ := targetAP.GetPropertyWPAFlags()
rsnFlags, _ := targetAP.GetPropertyRSNFlags()
const KeyMgmt8021x = uint32(512)
const KeyMgmtPsk = uint32(256)
const KeyMgmtSae = uint32(1024)
isEnterprise := (wpaFlags&KeyMgmt8021x) != 0 || (rsnFlags&KeyMgmt8021x) != 0
isPsk := (wpaFlags&KeyMgmtPsk) != 0 || (rsnFlags&KeyMgmtPsk) != 0
isSae := (wpaFlags&KeyMgmtSae) != 0 || (rsnFlags&KeyMgmtSae) != 0
secured := flags != uint32(gonetworkmanager.Nm80211APFlagsNone) ||
wpaFlags != uint32(gonetworkmanager.Nm80211APSecNone) ||
rsnFlags != uint32(gonetworkmanager.Nm80211APSecNone)
if isEnterprise {
log.Infof("[createAndConnectWiFi] Enterprise network detected (802.1x) - SSID: %s, interactive: %v",
req.SSID, req.Interactive)
}
settings := make(map[string]map[string]interface{})
settings["connection"] = map[string]interface{}{
"id": req.SSID,
"type": "802-11-wireless",
"autoconnect": true,
}
settings["ipv4"] = map[string]interface{}{"method": "auto"}
settings["ipv6"] = map[string]interface{}{"method": "auto"}
if secured {
settings["802-11-wireless"] = map[string]interface{}{
"ssid": []byte(req.SSID),
"mode": "infrastructure",
"security": "802-11-wireless-security",
}
switch {
case isEnterprise || req.Username != "":
settings["802-11-wireless-security"] = map[string]interface{}{
"key-mgmt": "wpa-eap",
}
x := map[string]interface{}{
"eap": []string{"peap"},
"phase2-auth": "mschapv2",
"system-ca-certs": false,
"password-flags": uint32(0),
}
if req.Username != "" {
x["identity"] = req.Username
}
if req.Password != "" {
x["password"] = req.Password
}
if req.AnonymousIdentity != "" {
x["anonymous-identity"] = req.AnonymousIdentity
}
if req.DomainSuffixMatch != "" {
x["domain-suffix-match"] = req.DomainSuffixMatch
}
settings["802-1x"] = x
log.Infof("[createAndConnectWiFi] WPA-EAP settings: eap=peap, phase2-auth=mschapv2, identity=%s, interactive=%v, system-ca-certs=%v, domain-suffix-match=%q",
req.Username, req.Interactive, x["system-ca-certs"], req.DomainSuffixMatch)
case isPsk:
sec := map[string]interface{}{
"key-mgmt": "wpa-psk",
"psk-flags": uint32(0),
}
if !req.Interactive {
sec["psk"] = req.Password
}
settings["802-11-wireless-security"] = sec
case isSae:
sec := map[string]interface{}{
"key-mgmt": "sae",
"pmf": int32(3),
"psk-flags": uint32(0),
}
if !req.Interactive {
sec["psk"] = req.Password
}
settings["802-11-wireless-security"] = sec
default:
return fmt.Errorf("secured network but not SAE/PSK/802.1X (rsn=0x%x wpa=0x%x)", rsnFlags, wpaFlags)
}
} else {
settings["802-11-wireless"] = map[string]interface{}{
"ssid": []byte(req.SSID),
"mode": "infrastructure",
}
}
if req.Interactive {
s := b.settings
if s == nil {
var settingsErr error
s, settingsErr = gonetworkmanager.NewSettings()
if settingsErr != nil {
return fmt.Errorf("failed to get settings manager: %w", settingsErr)
}
b.settings = s
}
settingsMgr := s.(gonetworkmanager.Settings)
conn, err := settingsMgr.AddConnection(settings)
if err != nil {
return fmt.Errorf("failed to add connection: %w", err)
}
if isEnterprise {
log.Infof("[createAndConnectWiFi] Enterprise connection added, activating (secret agent will be called)")
}
_, err = nm.ActivateWirelessConnection(conn, dev, targetAP)
if err != nil {
return fmt.Errorf("failed to activate connection: %w", err)
}
log.Infof("[createAndConnectWiFi] Connection activation initiated, waiting for NetworkManager state changes...")
} else {
_, err = nm.AddAndActivateWirelessConnection(settings, dev, targetAP)
if err != nil {
return fmt.Errorf("failed to connect: %w", err)
}
log.Infof("[createAndConnectWiFi] Connection activation initiated, waiting for NetworkManager state changes...")
}
return nil
}
func (b *NetworkManagerBackend) SetWiFiAutoconnect(ssid string, autoconnect bool) error {
conn, err := b.findConnection(ssid)
if err != nil {
return fmt.Errorf("connection not found: %w", err)
}
settings, err := conn.GetSettings()
if err != nil {
return fmt.Errorf("failed to get connection settings: %w", err)
}
if connMeta, ok := settings["connection"]; ok {
connMeta["autoconnect"] = autoconnect
} else {
return fmt.Errorf("connection metadata not found")
}
if ipv4, ok := settings["ipv4"]; ok {
delete(ipv4, "addresses")
delete(ipv4, "routes")
delete(ipv4, "dns")
}
if ipv6, ok := settings["ipv6"]; ok {
delete(ipv6, "addresses")
delete(ipv6, "routes")
delete(ipv6, "dns")
}
err = conn.Update(settings)
if err != nil {
return fmt.Errorf("failed to update connection: %w", err)
}
b.updateWiFiNetworks()
if b.onStateChange != nil {
b.onStateChange()
}
return nil
}